Ndbif.cpp
上传用户:romrleung
上传日期:2022-05-23
资源大小:18897k
文件大小:42k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2003 MySQL AB
  2.    This program is free software; you can redistribute it and/or modify
  3.    it under the terms of the GNU General Public License as published by
  4.    the Free Software Foundation; either version 2 of the License, or
  5.    (at your option) any later version.
  6.    This program is distributed in the hope that it will be useful,
  7.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  8.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  9.    GNU General Public License for more details.
  10.    You should have received a copy of the GNU General Public License
  11.    along with this program; if not, write to the Free Software
  12.    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
  13. #include <ndb_global.h>
  14. #include "NdbApiSignal.hpp"
  15. #include "NdbImpl.hpp"
  16. #include "NdbOperation.hpp"
  17. #include "NdbIndexOperation.hpp"
  18. #include "NdbScanOperation.hpp"
  19. #include "NdbConnection.hpp"
  20. #include "NdbRecAttr.hpp"
  21. #include "NdbReceiver.hpp"
  22. #include "API.hpp"
  23. #include <signaldata/TcCommit.hpp>
  24. #include <signaldata/TcKeyFailConf.hpp>
  25. #include <signaldata/TcKeyConf.hpp>
  26. #include <signaldata/TestOrd.hpp>
  27. #include <signaldata/CreateIndx.hpp>
  28. #include <signaldata/DropIndx.hpp>
  29. #include <signaldata/TcIndx.hpp>
  30. #include <signaldata/TransIdAI.hpp>
  31. #include <signaldata/ScanFrag.hpp>
  32. #include <signaldata/ScanTab.hpp>
  33. #include <ndb_limits.h>
  34. #include <NdbOut.hpp>
  35. #include <NdbTick.h>
  36. /******************************************************************************
  37.  * int init( int aNrOfCon, int aNrOfOp );
  38.  *
  39.  * Return Value:   Return 0 : init was successful.
  40.  *                Return -1: In all other case.  
  41.  * Parameters: aNrOfCon : Number of connections offered to the application.
  42.  * aNrOfOp : Number of operations offered to the application.
  43.  * Remark: Create pointers and idle list Synchronous.
  44.  ****************************************************************************/ 
  45. int
  46. Ndb::init(int aMaxNoOfTransactions)
  47. {
  48.   DBUG_ENTER("Ndb::init");
  49.   int i;
  50.   int aNrOfCon;
  51.   int aNrOfOp;
  52.   int tMaxNoOfTransactions;
  53.   NdbApiSignal* tSignal[16]; // Initiate free list of 16 signal objects
  54.   if (theInitState != NotInitialised) {
  55.     switch(theInitState){
  56.     case InitConfigError:
  57.       theError.code = 4117;
  58.       break;
  59.     default:
  60.       theError.code = 4104;
  61.       break;
  62.     }
  63.     DBUG_RETURN(-1);
  64.   }//if
  65.   theInitState = StartingInit;
  66.   TransporterFacade * theFacade =  TransporterFacade::instance();
  67.   theFacade->lock_mutex();
  68.   
  69.   const int tBlockNo = theFacade->open(this,
  70.                                        executeMessage, 
  71.                                        statusMessage);  
  72.   if ( tBlockNo == -1 ) {
  73.     theError.code = 4105;
  74.     theFacade->unlock_mutex();
  75.     DBUG_RETURN(-1); // no more free blocknumbers
  76.   }//if
  77.   
  78.   theNdbBlockNumber = tBlockNo;
  79.   theFacade->unlock_mutex();
  80.   
  81.   theDictionary->setTransporter(this, theFacade);
  82.   
  83.   aNrOfCon = theImpl->theNoOfDBnodes;
  84.   aNrOfOp = 2*theImpl->theNoOfDBnodes;
  85.   
  86.   // Create connection object in a linked list 
  87.   if((createConIdleList(aNrOfCon)) == -1){
  88.     theError.code = 4000;
  89.     goto error_handler;
  90.   }
  91.   
  92.   // Create operations in a linked list
  93.   if((createOpIdleList(aNrOfOp)) == -1){       
  94.     theError.code = 4000;
  95.     goto error_handler;
  96.   }
  97.   
  98.   tMaxNoOfTransactions = aMaxNoOfTransactions * 3;
  99.   if (tMaxNoOfTransactions > 1024) {
  100.     tMaxNoOfTransactions = 1024;
  101.   }//if
  102.   theMaxNoOfTransactions = tMaxNoOfTransactions;
  103.   
  104.   thePreparedTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
  105.   theSentTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
  106.   theCompletedTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
  107.   
  108.   if ((thePreparedTransactionsArray == NULL) ||
  109.       (theSentTransactionsArray == NULL) ||
  110.       (theCompletedTransactionsArray == NULL)) {
  111.     goto error_handler;
  112.   }//if
  113.   
  114.   for (i = 0; i < tMaxNoOfTransactions; i++) {
  115.     thePreparedTransactionsArray[i] = NULL;
  116.     theSentTransactionsArray[i] = NULL;
  117.     theCompletedTransactionsArray[i] = NULL;
  118.   }//for     
  119.   for (i = 0; i < 16; i++){
  120.     tSignal[i] = getSignal();
  121.     if(tSignal[i] == NULL) {
  122.       theError.code = 4000;
  123.       goto error_handler;
  124.     }
  125.   }
  126.   for (i = 0; i < 16; i++)
  127.     releaseSignal(tSignal[i]);
  128.   theInitState = Initialised; 
  129.   DBUG_RETURN(0);
  130.   
  131. error_handler:
  132.   ndbout << "error_handler" << endl;
  133.   releaseTransactionArrays();
  134.   delete theDictionary;
  135.   TransporterFacade::instance()->close(theNdbBlockNumber, 0);
  136.   DBUG_RETURN(-1);
  137. }
  138. void
  139. Ndb::releaseTransactionArrays()
  140. {
  141.   DBUG_ENTER("Ndb::releaseTransactionArrays");
  142.   if (thePreparedTransactionsArray != NULL) {
  143.     delete [] thePreparedTransactionsArray;
  144.   }//if
  145.   if (theSentTransactionsArray != NULL) {
  146.     delete [] theSentTransactionsArray;
  147.   }//if
  148.   if (theCompletedTransactionsArray != NULL) {
  149.     delete [] theCompletedTransactionsArray;
  150.   }//if
  151.   DBUG_VOID_RETURN;
  152. }//Ndb::releaseTransactionArrays()
  153. void
  154. Ndb::executeMessage(void* NdbObject,
  155.                     NdbApiSignal * aSignal,
  156.                     LinearSectionPtr ptr[3])
  157. {
  158.   Ndb* tNdb = (Ndb*)NdbObject;
  159.   tNdb->handleReceivedSignal(aSignal, ptr);
  160. }
  161. void Ndb::connected(Uint32 ref)
  162. {
  163.   theMyRef= ref;
  164.   Uint32 tmpTheNode= refToNode(ref);
  165.   Uint64 tBlockNo= refToBlock(ref);
  166.   if (theNdbBlockNumber >= 0){
  167.     assert(theMyRef == numberToRef(theNdbBlockNumber, tmpTheNode));
  168.   }
  169.   
  170.   TransporterFacade * theFacade =  TransporterFacade::instance();
  171.   int i, n= 0;
  172.   for (i = 1; i < MAX_NDB_NODES; i++){
  173.     if (theFacade->getIsDbNode(i)){
  174.       theImpl->theDBnodes[n] = i;
  175.       n++;
  176.     }
  177.   }
  178.   theImpl->theNoOfDBnodes= n;
  179.   theFirstTransId = ((Uint64)tBlockNo << 52)+
  180.     ((Uint64)tmpTheNode << 40);
  181.   theFirstTransId += theFacade->m_max_trans_id;
  182.   //      assert(0);
  183.   DBUG_PRINT("info",("connected with ref=%x, id=%d, no_db_nodes=%d, first_trans_id=%lx",
  184.      theMyRef,
  185.      tmpTheNode,
  186.      theImpl->theNoOfDBnodes,
  187.      theFirstTransId));
  188.   theCommitAckSignal = new NdbApiSignal(theMyRef);
  189.   theDictionary->m_receiver.m_reference= theMyRef;
  190.   theNode= tmpTheNode; // flag that Ndb object is initialized
  191. }
  192. void
  193. Ndb::statusMessage(void* NdbObject, Uint32 a_node, bool alive, bool nfComplete)
  194. {
  195.   DBUG_ENTER("Ndb::statusMessage");
  196.   Ndb* tNdb = (Ndb*)NdbObject;
  197.   if (alive) {
  198.     if (nfComplete) {
  199.       tNdb->connected(a_node);
  200.       DBUG_VOID_RETURN;
  201.     }//if
  202.   } else {
  203.     if (nfComplete) {
  204.       tNdb->report_node_failure_completed(a_node);
  205.     } else {
  206.       tNdb->report_node_failure(a_node);
  207.     }//if
  208.   }//if
  209.   NdbDictInterface::execNodeStatus(&tNdb->theDictionary->m_receiver,
  210.    a_node, alive, nfComplete);
  211.   DBUG_VOID_RETURN;
  212. }
  213. void
  214. Ndb::report_node_failure(Uint32 node_id)
  215. {
  216.   /**
  217.    * We can only set the state here since this object can execute 
  218.    * simultaneously. 
  219.    * 
  220.    * This method is only called by ClusterMgr (via lots of methods)
  221.    */
  222.   theImpl->the_release_ind[node_id] = 1;
  223.   // must come after
  224.   theImpl->the_release_ind[0] = 1;
  225.   theImpl->theWaiter.nodeFail(node_id);
  226.   return;
  227. }//Ndb::report_node_failure()
  228. void
  229. Ndb::report_node_failure_completed(Uint32 node_id)
  230. {
  231.   abortTransactionsAfterNodeFailure(node_id);
  232. }//Ndb::report_node_failure_completed()
  233. /***************************************************************************
  234. void abortTransactionsAfterNodeFailure();
  235. Remark:   Abort all transactions in theSentTransactionsArray after connection 
  236.           to one node has failed
  237. ****************************************************************************/
  238. void
  239. Ndb::abortTransactionsAfterNodeFailure(Uint16 aNodeId)
  240. {  
  241.   Uint32 tNoSentTransactions = theNoOfSentTransactions;
  242.   for (int i = tNoSentTransactions - 1; i >= 0; i--) {
  243.     NdbConnection* localCon = theSentTransactionsArray[i];
  244.     if (localCon->getConnectedNodeId() == aNodeId) {
  245.       const NdbConnection::SendStatusType sendStatus = localCon->theSendStatus;
  246.       if (sendStatus == NdbConnection::sendTC_OP || 
  247.   sendStatus == NdbConnection::sendTC_COMMIT) {
  248.         /*
  249.         A transaction was interrupted in the prepare phase by a node
  250.         failure. Since the transaction was not found in the phase
  251.         after the node failure it cannot have been committed and
  252.         we report a normal node failure abort.
  253.         */
  254. localCon->setOperationErrorCodeAbort(4010);
  255.         localCon->theCompletionStatus = NdbConnection::CompletedFailure;
  256.       } else if (sendStatus == NdbConnection::sendTC_ROLLBACK) {
  257.         /*
  258.         We aimed for abort and abort we got even if it was by a node
  259.         failure. We will thus report it as a success.
  260.         */
  261.         localCon->theCompletionStatus = NdbConnection::CompletedSuccess;
  262.       } else {
  263. #ifdef VM_TRACE
  264.         printState("abortTransactionsAfterNodeFailure %x", this);
  265.         abort();
  266. #endif
  267.       }
  268.       /*
  269.       All transactions arriving here have no connection to the kernel
  270.       intact since the node was failing and they were aborted. Thus we
  271.       set commit state to Aborted and set state to release on close.
  272.       */
  273.       localCon->theCommitStatus = NdbConnection::Aborted;
  274.       localCon->theReleaseOnClose = true;
  275.       completedTransaction(localCon);
  276.     }
  277.     else if(localCon->report_node_failure(aNodeId))
  278.     {
  279.       completedTransaction(localCon);
  280.     }
  281.   }//for
  282.   return;
  283. }//Ndb::abortTransactionsAfterNodeFailure()
  284. /****************************************************************************
  285. void handleReceivedSignal(NdbApiSignal* aSignal);
  286. Parameters:     aSignal: The signal object.
  287. Remark:         Send all operations belonging to this connection. 
  288. *****************************************************************************/
  289. void
  290. Ndb::handleReceivedSignal(NdbApiSignal* aSignal, LinearSectionPtr ptr[3])
  291. {
  292.   NdbOperation* tOp;
  293.   NdbIndexOperation* tIndexOp;
  294.   NdbConnection* tCon;
  295.   int tReturnCode = -1;
  296.   const Uint32* tDataPtr = aSignal->getDataPtr();
  297.   const Uint32 tWaitState = theImpl->theWaiter.m_state;
  298.   const Uint32 tSignalNumber = aSignal->readSignalNumber();
  299.   const Uint32 tFirstData = *tDataPtr;
  300.   const Uint32 tLen = aSignal->getLength();
  301.   void * tFirstDataPtr;
  302.   /*
  303.     In order to support 64 bit processes in the application we need to use
  304.     id's rather than a direct pointer to the object used. It is also a good
  305.     idea that one cannot corrupt the application code by sending a corrupt
  306.     memory pointer.
  307.     
  308.     All signals received by the API requires the first data word to be such
  309.     an id to the receiving object.
  310.   */
  311.   
  312.   switch (tSignalNumber){
  313.   case GSN_TCKEYCONF:
  314.     {
  315.       tFirstDataPtr = int2void(tFirstData);
  316.       if (tFirstDataPtr == 0) goto InvalidSignal;
  317.       const TcKeyConf * const keyConf = (TcKeyConf *)tDataPtr;
  318.       const BlockReference aTCRef = aSignal->theSendersBlockRef;
  319.       tCon = void2con(tFirstDataPtr);
  320.       if ((tCon->checkMagicNumber() == 0) &&
  321.           (tCon->theSendStatus == NdbConnection::sendTC_OP)) {
  322.         tReturnCode = tCon->receiveTCKEYCONF(keyConf, tLen);
  323.         if (tReturnCode != -1) {
  324.           completedTransaction(tCon);
  325.         }//if
  326. if(TcKeyConf::getMarkerFlag(keyConf->confInfo)){
  327.   NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
  328.    keyConf->transId1, 
  329.    keyConf->transId2,
  330.    aTCRef);
  331. }
  332.       
  333. return;
  334.       }//if
  335.       goto InvalidSignal;
  336.       
  337.       return;
  338.     }
  339.   case GSN_TRANSID_AI:{
  340.     tFirstDataPtr = int2void(tFirstData);
  341.     NdbReceiver* tRec;
  342.     if (tFirstDataPtr && (tRec = void2rec(tFirstDataPtr)) && 
  343. tRec->checkMagicNumber() && (tCon = tRec->getTransaction()) &&
  344. tCon->checkState_TransId(((const TransIdAI*)tDataPtr)->transId)){
  345.       Uint32 com;
  346.       if(aSignal->m_noOfSections > 0){
  347. com = tRec->execTRANSID_AI(ptr[0].p, ptr[0].sz);
  348.       } else {
  349. com = tRec->execTRANSID_AI(tDataPtr + TransIdAI::HeaderLength, 
  350.    tLen - TransIdAI::HeaderLength);
  351.       }
  352.       
  353.       if(com == 1){
  354. switch(tRec->getType()){
  355. case NdbReceiver::NDB_OPERATION:
  356. case NdbReceiver::NDB_INDEX_OPERATION:
  357.   if(tCon->OpCompleteSuccess() != -1){
  358.     completedTransaction(tCon);
  359.     return;
  360.   }
  361.   break;
  362. case NdbReceiver::NDB_SCANRECEIVER:
  363.   tCon->theScanningOp->receiver_delivered(tRec);
  364.   theImpl->theWaiter.m_state = (((WaitSignalType) tWaitState) == WAIT_SCAN ? 
  365.        (Uint32) NO_WAIT : tWaitState);
  366.   break;
  367. default:
  368.   goto InvalidSignal;
  369. }
  370.       }
  371.       break;
  372.     } else {
  373.       /**
  374.        * This is ok as transaction can have been aborted before TRANSID_AI
  375.        * arrives (if TUP on  other node than TC)
  376.        */
  377.       return;
  378.     }
  379.   }
  380.   case GSN_TCKEY_FAILCONF:
  381.     {
  382.       tFirstDataPtr = int2void(tFirstData);
  383.       const TcKeyFailConf * failConf = (TcKeyFailConf *)tDataPtr;
  384.       const BlockReference aTCRef = aSignal->theSendersBlockRef;
  385.       if (tFirstDataPtr != 0){
  386. tOp = void2rec_op(tFirstDataPtr);
  387. if (tOp->checkMagicNumber(false) == 0) {
  388.   tCon = tOp->theNdbCon;
  389.   if (tCon != NULL) {
  390.     if ((tCon->theSendStatus == NdbConnection::sendTC_OP) ||
  391. (tCon->theSendStatus == NdbConnection::sendTC_COMMIT)) {
  392.       tReturnCode = tCon->receiveTCKEY_FAILCONF(failConf);
  393.       if (tReturnCode != -1) {
  394. completedTransaction(tCon);
  395.       }//if
  396.     }//if
  397.   }
  398. }
  399.       } else {
  400. #ifdef VM_TRACE
  401. ndbout_c("Recevied TCKEY_FAILCONF wo/ operation");
  402. #endif
  403.       }
  404.       if(tFirstData & 1){
  405. NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
  406.  failConf->transId1, 
  407.  failConf->transId2,
  408.  aTCRef);
  409.       }
  410.       return;
  411.     }
  412.   case GSN_TCKEY_FAILREF:
  413.     {
  414.       tFirstDataPtr = int2void(tFirstData);
  415.       if(tFirstDataPtr != 0){
  416. tOp = void2rec_op(tFirstDataPtr);
  417. if (tOp->checkMagicNumber(false) == 0) {
  418.   tCon = tOp->theNdbCon;
  419.   if (tCon != NULL) {
  420.     if ((tCon->theSendStatus == NdbConnection::sendTC_OP) ||
  421. (tCon->theSendStatus == NdbConnection::sendTC_ROLLBACK)) {
  422.       tReturnCode = tCon->receiveTCKEY_FAILREF(aSignal);
  423.       if (tReturnCode != -1) {
  424. completedTransaction(tCon);
  425. return;
  426.       }//if
  427.     }//if
  428.   }//if
  429. }//if
  430.       }
  431. #ifdef VM_TRACE
  432.       ndbout_c("Recevied TCKEY_FAILREF wo/ operation");
  433. #endif
  434.       return;
  435.       break;
  436.     }
  437.   case GSN_TCKEYREF:
  438.     {
  439.       tFirstDataPtr = int2void(tFirstData);
  440.       if (tFirstDataPtr == 0) goto InvalidSignal;
  441.       tOp = void2rec_op(tFirstDataPtr);
  442.       if (tOp->checkMagicNumber() == 0) {
  443. tCon = tOp->theNdbCon;
  444. if (tCon != NULL) {
  445.   if (tCon->theSendStatus == NdbConnection::sendTC_OP) {
  446.     tReturnCode = tOp->receiveTCKEYREF(aSignal);
  447.     if (tReturnCode != -1) {
  448.       completedTransaction(tCon);
  449.       return;
  450.     }//if
  451.     break;
  452.   }//if
  453. }//if
  454.       } //if
  455.       goto InvalidSignal;
  456.       return;
  457.     } 
  458.   case GSN_TC_COMMITCONF:
  459.     {
  460.       tFirstDataPtr = int2void(tFirstData);
  461.       if (tFirstDataPtr == 0) goto InvalidSignal;
  462.       const TcCommitConf * const commitConf = (TcCommitConf *)tDataPtr;
  463.       const BlockReference aTCRef = aSignal->theSendersBlockRef;
  464.       
  465.       tCon = void2con(tFirstDataPtr);
  466.       if ((tCon->checkMagicNumber() == 0) &&
  467.   (tCon->theSendStatus == NdbConnection::sendTC_COMMIT)) {
  468. tReturnCode = tCon->receiveTC_COMMITCONF(commitConf);
  469. if (tReturnCode != -1) {
  470.   completedTransaction(tCon);
  471. }//if
  472. if(tFirstData & 1){
  473.   NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
  474.    commitConf->transId1, 
  475.    commitConf->transId2,
  476.    aTCRef);
  477. }
  478. return;
  479.       }
  480.       goto InvalidSignal;
  481.       return;
  482.     }
  483.   case GSN_TC_COMMITREF:
  484.     {
  485.       tFirstDataPtr = int2void(tFirstData);
  486.       if (tFirstDataPtr == 0) goto InvalidSignal;
  487.       tCon = void2con(tFirstDataPtr);
  488.       if ((tCon->checkMagicNumber() == 0) &&
  489.   (tCon->theSendStatus == NdbConnection::sendTC_COMMIT)) {
  490. tReturnCode = tCon->receiveTC_COMMITREF(aSignal);
  491. if (tReturnCode != -1) {
  492.   completedTransaction(tCon);
  493. }//if
  494.       }//if
  495.       return;
  496.     }
  497.   case GSN_TCROLLBACKCONF:
  498.     {
  499.       tFirstDataPtr = int2void(tFirstData);
  500.       if (tFirstDataPtr == 0) goto InvalidSignal;
  501.       tCon = void2con(tFirstDataPtr);
  502.       if ((tCon->checkMagicNumber() == 0) &&
  503.   (tCon->theSendStatus == NdbConnection::sendTC_ROLLBACK)) {
  504. tReturnCode = tCon->receiveTCROLLBACKCONF(aSignal);
  505. if (tReturnCode != -1) {
  506.   completedTransaction(tCon);
  507. }//if
  508.       }//if
  509.       return;
  510.     }
  511.   case GSN_TCROLLBACKREF:
  512.     {
  513.       tFirstDataPtr = int2void(tFirstData);
  514.       if (tFirstDataPtr == 0) goto InvalidSignal;
  515.       tCon = void2con(tFirstDataPtr);
  516.       if ((tCon->checkMagicNumber() == 0) &&
  517.   (tCon->theSendStatus == NdbConnection::sendTC_ROLLBACK)) {
  518. tReturnCode = tCon->receiveTCROLLBACKREF(aSignal);
  519. if (tReturnCode != -1) {
  520.   completedTransaction(tCon);
  521. }//if
  522.       }//if
  523.       return;
  524.     }
  525.   case GSN_TCROLLBACKREP:
  526.     {
  527.       tFirstDataPtr = int2void(tFirstData);
  528.       if (tFirstDataPtr == 0) goto InvalidSignal;
  529.       tCon = void2con(tFirstDataPtr);
  530.       if (tCon->checkMagicNumber() == 0) {
  531. tReturnCode = tCon->receiveTCROLLBACKREP(aSignal);
  532. if (tReturnCode != -1) {
  533.   completedTransaction(tCon);
  534. }//if
  535.       }//if
  536.       return;
  537.     }
  538.   case GSN_TCSEIZECONF:
  539.     {
  540.       tFirstDataPtr = int2void(tFirstData);
  541.       if (tFirstDataPtr == 0) goto InvalidSignal;
  542.       if (tWaitState != WAIT_TC_SEIZE) {
  543. goto InvalidSignal;
  544.       }//if
  545.       tCon = void2con(tFirstDataPtr);
  546.       if (tCon->checkMagicNumber() != 0) {
  547. goto InvalidSignal;
  548.       }//if
  549.       tReturnCode = tCon->receiveTCSEIZECONF(aSignal);
  550.       if (tReturnCode != -1) {
  551. theImpl->theWaiter.m_state = NO_WAIT;
  552.       } else {
  553. goto InvalidSignal;
  554.       }//if
  555.       break;
  556.     }
  557.   case GSN_TCSEIZEREF:
  558.     {
  559.       tFirstDataPtr = int2void(tFirstData);
  560.       if (tFirstDataPtr == 0) goto InvalidSignal;
  561.       if (tWaitState != WAIT_TC_SEIZE) {
  562. return;
  563.       }//if
  564.       tCon = void2con(tFirstDataPtr);
  565.       if (tCon->checkMagicNumber() != 0) {
  566. return;
  567.       }//if
  568.       tReturnCode = tCon->receiveTCSEIZEREF(aSignal);
  569.       if (tReturnCode != -1) {
  570. theImpl->theWaiter.m_state = NO_WAIT;
  571.       } else {
  572.         return;
  573.       }//if
  574.       break;
  575.     }
  576.   case GSN_TCRELEASECONF:
  577.     {
  578.       tFirstDataPtr = int2void(tFirstData);
  579.       if (tFirstDataPtr == 0) goto InvalidSignal;
  580.       if (tWaitState != WAIT_TC_RELEASE) {
  581. goto InvalidSignal;
  582.       }//if
  583.       tCon = void2con(tFirstDataPtr);
  584.       if (tCon->checkMagicNumber() != 0) {
  585. goto InvalidSignal;
  586.       }//if
  587.       tReturnCode = tCon->receiveTCRELEASECONF(aSignal);
  588.       if (tReturnCode != -1) {
  589. theImpl->theWaiter.m_state = NO_WAIT;
  590.       }//if
  591.       break;
  592.     } 
  593.   case GSN_TCRELEASEREF:
  594.     {
  595.       tFirstDataPtr = int2void(tFirstData);
  596.       if (tFirstDataPtr == 0) goto InvalidSignal;
  597.       if (tWaitState != WAIT_TC_RELEASE) {
  598. goto InvalidSignal;
  599.       }//if
  600.       tCon = void2con(tFirstDataPtr);
  601.       if (tCon->checkMagicNumber() != 0) {
  602. goto InvalidSignal;
  603.       }//if
  604.       tReturnCode = tCon->receiveTCRELEASEREF(aSignal);
  605.       if (tReturnCode != -1) {
  606. theImpl->theWaiter.m_state = NO_WAIT;
  607.       }//if
  608.       break;
  609.     }
  610.       
  611.   case GSN_GET_TABINFOREF:
  612.   case GSN_GET_TABINFO_CONF:
  613.   case GSN_CREATE_TABLE_REF:
  614.   case GSN_CREATE_TABLE_CONF:
  615.   case GSN_DROP_TABLE_CONF:
  616.   case GSN_DROP_TABLE_REF:
  617.   case GSN_ALTER_TABLE_CONF:
  618.   case GSN_ALTER_TABLE_REF:
  619.   case GSN_CREATE_INDX_CONF:
  620.   case GSN_CREATE_INDX_REF:
  621.   case GSN_DROP_INDX_CONF:
  622.   case GSN_DROP_INDX_REF:
  623.   case GSN_CREATE_EVNT_CONF:
  624.   case GSN_CREATE_EVNT_REF:
  625.   case GSN_DROP_EVNT_CONF:
  626.   case GSN_DROP_EVNT_REF:
  627.   case GSN_LIST_TABLES_CONF:
  628.     NdbDictInterface::execSignal(&theDictionary->m_receiver,
  629.  aSignal, ptr);
  630.     break;
  631.     
  632.   case GSN_SUB_META_DATA:
  633.   case GSN_SUB_REMOVE_CONF:
  634.   case GSN_SUB_REMOVE_REF:
  635.     break; // ignore these signals
  636.   case GSN_SUB_GCP_COMPLETE_REP:
  637.   case GSN_SUB_START_CONF:
  638.   case GSN_SUB_START_REF:
  639.   case GSN_SUB_TABLE_DATA:
  640.   case GSN_SUB_STOP_CONF:
  641.   case GSN_SUB_STOP_REF:
  642.     NdbDictInterface::execSignal(&theDictionary->m_receiver,
  643.  aSignal, ptr);
  644.     break;
  645.   case GSN_DIHNDBTAMPER:
  646.     {
  647.       tFirstDataPtr = int2void(tFirstData);
  648.       if (tFirstDataPtr == 0) goto InvalidSignal;
  649.       
  650.       if (tWaitState != WAIT_NDB_TAMPER)
  651. return;
  652.       tCon = void2con(tFirstDataPtr);
  653.       if (tCon->checkMagicNumber() != 0)
  654. return;
  655.       tReturnCode = tCon->receiveDIHNDBTAMPER(aSignal);
  656.       if (tReturnCode != -1)
  657. theImpl->theWaiter.m_state = NO_WAIT;
  658.       break;
  659.     }
  660.   case GSN_SCAN_TABCONF:
  661.     {
  662.       tFirstDataPtr = int2void(tFirstData);
  663.       assert(tFirstDataPtr);
  664.       assert(void2con(tFirstDataPtr));
  665.       assert(void2con(tFirstDataPtr)->checkMagicNumber() == 0);
  666.       if(tFirstDataPtr && 
  667.  (tCon = void2con(tFirstDataPtr)) && (tCon->checkMagicNumber() == 0)){
  668. if(aSignal->m_noOfSections > 0){
  669.   tReturnCode = tCon->receiveSCAN_TABCONF(aSignal, 
  670.   ptr[0].p, ptr[0].sz);
  671. } else {
  672.   tReturnCode = 
  673.     tCon->receiveSCAN_TABCONF(aSignal, 
  674.       tDataPtr + ScanTabConf::SignalLength, 
  675.       tLen - ScanTabConf::SignalLength);
  676. }
  677. if (tReturnCode != -1 && tWaitState == WAIT_SCAN)
  678.   theImpl->theWaiter.m_state = NO_WAIT;
  679. break;
  680.       } else {
  681. goto InvalidSignal;
  682.       }
  683.     }
  684.   case GSN_SCAN_TABREF:
  685.     {
  686.       tFirstDataPtr = int2void(tFirstData);
  687.       if (tFirstDataPtr == 0) goto InvalidSignal;
  688.       
  689.       tCon = void2con(tFirstDataPtr);
  690.       
  691.       assert(tFirstDataPtr != 0 && 
  692.      void2con(tFirstDataPtr)->checkMagicNumber() == 0);
  693.       
  694.       if (tCon->checkMagicNumber() == 0){
  695. tReturnCode = tCon->receiveSCAN_TABREF(aSignal);
  696. if (tReturnCode != -1 && tWaitState == WAIT_SCAN){
  697.   theImpl->theWaiter.m_state = NO_WAIT;
  698. }
  699. break;
  700.       }
  701.       goto InvalidSignal;
  702.     }
  703.   case GSN_KEYINFO20: {
  704.     tFirstDataPtr = int2void(tFirstData);
  705.     NdbReceiver* tRec;
  706.     if (tFirstDataPtr && (tRec = void2rec(tFirstDataPtr)) &&
  707. tRec->checkMagicNumber() && (tCon = tRec->getTransaction()) &&
  708. tCon->checkState_TransId(&((const KeyInfo20*)tDataPtr)->transId1)){
  709.       
  710.       Uint32 len = ((const KeyInfo20*)tDataPtr)->keyLen;
  711.       Uint32 info = ((const KeyInfo20*)tDataPtr)->scanInfo_Node;
  712.       int com = -1;
  713.       if(aSignal->m_noOfSections > 0 && len == ptr[0].sz){
  714. com = tRec->execKEYINFO20(info, ptr[0].p, len);
  715.       } else if(len == tLen - KeyInfo20::HeaderLength){
  716. com = tRec->execKEYINFO20(info, tDataPtr+KeyInfo20::HeaderLength, len);
  717.       }
  718.       
  719.       switch(com){
  720.       case 1:
  721. tCon->theScanningOp->receiver_delivered(tRec);
  722. theImpl->theWaiter.m_state = (((WaitSignalType) tWaitState) == WAIT_SCAN ? 
  723.       (Uint32) NO_WAIT : tWaitState);
  724. break;
  725.       case 0:
  726. break;
  727.       case -1:
  728. goto InvalidSignal;
  729.       }
  730.       break;
  731.     } else {
  732.       /**
  733.        * This is ok as transaction can have been aborted before KEYINFO20
  734.        * arrives (if TUP on  other node than TC)
  735.        */
  736.       return;
  737.     }
  738.   }
  739.   case GSN_TCINDXCONF:{
  740.     tFirstDataPtr = int2void(tFirstData);
  741.     if (tFirstDataPtr == 0) goto InvalidSignal;
  742.     const TcIndxConf * const indxConf = (TcIndxConf *)tDataPtr;
  743.     const BlockReference aTCRef = aSignal->theSendersBlockRef;
  744.     tCon = void2con(tFirstDataPtr);
  745.     if ((tCon->checkMagicNumber() == 0) &&
  746. (tCon->theSendStatus == NdbConnection::sendTC_OP)) {
  747.       tReturnCode = tCon->receiveTCINDXCONF(indxConf, tLen);
  748.       if (tReturnCode != -1) { 
  749. completedTransaction(tCon);
  750.       }//if
  751.     }//if
  752.     
  753.     if(TcIndxConf::getMarkerFlag(indxConf->confInfo)){
  754.       NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
  755.        indxConf->transId1, 
  756.        indxConf->transId2,
  757.        aTCRef);
  758.     }
  759.     return;
  760.   }
  761.   case GSN_TCINDXREF:{
  762.     tFirstDataPtr = int2void(tFirstData);
  763.     if (tFirstDataPtr == 0) goto InvalidSignal;
  764.     tIndexOp = void2rec_iop(tFirstDataPtr);
  765.     if (tIndexOp->checkMagicNumber() == 0) {
  766.       tCon = tIndexOp->theNdbCon;
  767.       if (tCon != NULL) {
  768. if (tCon->theSendStatus == NdbConnection::sendTC_OP) {
  769.   tReturnCode = tIndexOp->receiveTCINDXREF(aSignal);
  770.   if (tReturnCode != -1) {
  771.     completedTransaction(tCon);
  772.   }//if
  773.   return;
  774. }//if
  775.       }//if
  776.     }//if
  777.     goto InvalidSignal;
  778.     return;
  779.   } 
  780.   default:
  781.     goto InvalidSignal;
  782.   }//switch
  783.   
  784.   if (theImpl->theWaiter.m_state == NO_WAIT) {
  785.     // Wake up the thread waiting for response
  786.     NdbCondition_Signal(theImpl->theWaiter.m_condition);
  787.   }//if
  788.   return;
  789.  InvalidSignal:
  790. #ifdef VM_TRACE
  791.   ndbout_c("Ndbif: Error Ndb::handleReceivedSignal "
  792.    "(GSN=%d, theImpl->theWaiter.m_state=%d)"
  793.    " sender = (Block: %d Node: %d)",
  794.    tSignalNumber,
  795.    tWaitState,
  796.    refToBlock(aSignal->theSendersBlockRef),
  797.    refToNode(aSignal->theSendersBlockRef));
  798. #endif
  799. #ifdef NDB_NO_DROPPED_SIGNAL
  800.   abort();
  801. #endif
  802.   
  803.   return;
  804. }//Ndb::handleReceivedSignal()
  805. /*****************************************************************************
  806. void completedTransaction(NdbConnection* aCon);
  807. Remark:   One transaction has been completed.
  808.           Remove it from send array and put it into the completed
  809.           transaction array. Finally check if it is time to wake
  810.           up a poller.
  811. ******************************************************************************/
  812. void
  813. Ndb::completedTransaction(NdbConnection* aCon)
  814. {
  815.   Uint32 tTransArrayIndex = aCon->theTransArrayIndex;
  816.   Uint32 tNoSentTransactions = theNoOfSentTransactions;
  817.   Uint32 tNoCompletedTransactions = theNoOfCompletedTransactions;
  818.   if ((tNoSentTransactions > 0) && (aCon->theListState == NdbConnection::InSendList) &&
  819.       (tTransArrayIndex < tNoSentTransactions)) {
  820.     NdbConnection* tMoveCon = theSentTransactionsArray[tNoSentTransactions - 1];
  821.     theCompletedTransactionsArray[tNoCompletedTransactions] = aCon;
  822.     aCon->theTransArrayIndex = tNoCompletedTransactions;
  823.     if (tMoveCon != aCon) {
  824.       tMoveCon->theTransArrayIndex = tTransArrayIndex;
  825.       theSentTransactionsArray[tTransArrayIndex] = tMoveCon;
  826.     }//if
  827.     theSentTransactionsArray[tNoSentTransactions - 1] = NULL;
  828.     theNoOfCompletedTransactions = tNoCompletedTransactions + 1;
  829.     theNoOfSentTransactions = tNoSentTransactions - 1;
  830.     aCon->theListState = NdbConnection::InCompletedList;
  831.     aCon->handleExecuteCompletion();
  832.     if ((theMinNoOfEventsToWakeUp != 0) &&
  833.         (theNoOfCompletedTransactions >= theMinNoOfEventsToWakeUp)) {
  834.       theMinNoOfEventsToWakeUp = 0;
  835.       NdbCondition_Signal(theImpl->theWaiter.m_condition);
  836.       return;
  837.     }//if
  838.   } else {
  839.     ndbout << "theNoOfSentTransactions = " << (int) theNoOfSentTransactions;
  840.     ndbout << " theListState = " << (int) aCon->theListState;
  841.     ndbout << " theTransArrayIndex = " << aCon->theTransArrayIndex;
  842.     ndbout << endl << flush;
  843. #ifdef VM_TRACE
  844.     printState("completedTransaction abort");
  845.     abort();
  846. #endif
  847.   }//if
  848. }//Ndb::completedTransaction()
  849. /*****************************************************************************
  850. void reportCallback(NdbConnection** aCopyArray, Uint32 aNoOfCompletedTrans);
  851. Remark:   Call the callback methods of the completed transactions.
  852. ******************************************************************************/
  853. void
  854. Ndb::reportCallback(NdbConnection** aCopyArray, Uint32 aNoOfCompletedTrans)
  855. {
  856.   Uint32         i;
  857.   if (aNoOfCompletedTrans > 0) {
  858.     for (i = 0; i < aNoOfCompletedTrans; i++) {
  859.       void* anyObject = aCopyArray[i]->theCallbackObject;
  860.       NdbAsynchCallback aCallback = aCopyArray[i]->theCallbackFunction;
  861.       int tResult = 0;
  862.       if (aCallback != NULL) {
  863.         if (aCopyArray[i]->theReturnStatus == NdbConnection::ReturnFailure) {
  864.           tResult = -1;
  865.         }//if
  866.         (*aCallback)(tResult, aCopyArray[i], anyObject);
  867.       }//if
  868.     }//for
  869.   }//if
  870. }//Ndb::reportCallback()
  871. /*****************************************************************************
  872. Uint32 pollCompleted(NdbConnection** aCopyArray);
  873. Remark:   Transfer the data from the completed transaction to a local array.
  874.           This support is used by a number of the poll-methods.
  875. ******************************************************************************/
  876. Uint32
  877. Ndb::pollCompleted(NdbConnection** aCopyArray)
  878. {
  879.   check_send_timeout();
  880.   Uint32         i;
  881.   Uint32 tNoCompletedTransactions = theNoOfCompletedTransactions;
  882.   if (tNoCompletedTransactions > 0) {
  883.     for (i = 0; i < tNoCompletedTransactions; i++) {
  884.       aCopyArray[i] = theCompletedTransactionsArray[i];
  885.       if (aCopyArray[i]->theListState != NdbConnection::InCompletedList) {
  886.         ndbout << "pollCompleted error ";
  887.         ndbout << (int) aCopyArray[i]->theListState << endl;
  888. abort();
  889.       }//if
  890.       theCompletedTransactionsArray[i] = NULL;
  891.       aCopyArray[i]->theListState = NdbConnection::NotInList;
  892.     }//for
  893.   }//if
  894.   theNoOfCompletedTransactions = 0;
  895.   return tNoCompletedTransactions;
  896. }//Ndb::pollCompleted()
  897. void
  898. Ndb::check_send_timeout()
  899. {
  900.   NDB_TICKS current_time = NdbTick_CurrentMillisecond();
  901.   if (current_time - the_last_check_time > 1000) {
  902.     the_last_check_time = current_time;
  903.     Uint32 no_of_sent = theNoOfSentTransactions;
  904.     for (Uint32 i = 0; i < no_of_sent; i++) {
  905.       NdbConnection* a_con = theSentTransactionsArray[i];
  906.       if ((current_time - a_con->theStartTransTime) >
  907.           WAITFOR_RESPONSE_TIMEOUT) {
  908. #ifdef VM_TRACE
  909.         a_con->printState();
  910. Uint32 t1 = a_con->theTransactionId;
  911. Uint32 t2 = a_con->theTransactionId >> 32;
  912. ndbout_c("[%.8x %.8x]", t1, t2);
  913. abort();
  914. #endif
  915.         a_con->setOperationErrorCodeAbort(4012);
  916.         a_con->theCommitStatus = NdbConnection::Aborted;
  917.         a_con->theCompletionStatus = NdbConnection::CompletedFailure;
  918.         a_con->handleExecuteCompletion();
  919.         remove_sent_list(i);
  920.         insert_completed_list(a_con);
  921.         no_of_sent--;
  922.         i--;
  923.       }//if
  924.     }//for
  925.   }//if
  926. }
  927. void
  928. Ndb::remove_sent_list(Uint32 list_index)
  929. {
  930.   Uint32 last_index = theNoOfSentTransactions - 1;
  931.   if (list_index < last_index) {
  932.     NdbConnection* t_con = theSentTransactionsArray[last_index];
  933.     theSentTransactionsArray[list_index] = t_con;
  934.   }//if
  935.   theNoOfSentTransactions = last_index;
  936.   theSentTransactionsArray[last_index] = 0;
  937. }
  938. Uint32
  939. Ndb::insert_completed_list(NdbConnection* a_con)
  940. {
  941.   Uint32 no_of_comp = theNoOfCompletedTransactions;
  942.   theCompletedTransactionsArray[no_of_comp] = a_con;
  943.   theNoOfCompletedTransactions = no_of_comp + 1;
  944.   a_con->theListState = NdbConnection::InCompletedList;
  945.   a_con->theTransArrayIndex = no_of_comp;
  946.   return no_of_comp;
  947. }
  948. Uint32
  949. Ndb::insert_sent_list(NdbConnection* a_con)
  950. {
  951.   Uint32 no_of_sent = theNoOfSentTransactions;
  952.   theSentTransactionsArray[no_of_sent] = a_con;
  953.   theNoOfSentTransactions = no_of_sent + 1;
  954.   a_con->theListState = NdbConnection::InSendList;
  955.   a_con->theTransArrayIndex = no_of_sent;
  956.   return no_of_sent;
  957. }
  958. /*****************************************************************************
  959. void sendPrepTrans(int forceSend);
  960. Remark: Send a batch of transactions prepared for sending to the NDB kernel.  
  961. ******************************************************************************/
  962. void
  963. Ndb::sendPrepTrans(int forceSend)
  964. {
  965.   // Always called when holding mutex on TransporterFacade
  966.   /*
  967.      We will send a list of transactions to the NDB kernel. Before
  968.      sending we check the following.
  969.      1) Node connected to is still alive
  970.         Checked by both checking node status and node sequence
  971.      2) Send buffer can handle the size of messages we are planning to send
  972.         So far this is just a fake check but will soon be a real check
  973.      When the connected node has failed we abort the transaction without
  974.      responding anymore to the node since the kernel will clean up
  975.      automatically.
  976.      When sendBuffer cannot handle anymore messages then we will also abort
  977.      transaction but by communicating to the kernel since it is still alive
  978.      and we keep a small space for messages like that.
  979.   */
  980.   Uint32 i;
  981.   TransporterFacade* tp = TransporterFacade::instance();
  982.   Uint32 no_of_prep_trans = theNoOfPreparedTransactions;
  983.   for (i = 0; i < no_of_prep_trans; i++) {
  984.     NdbConnection * a_con = thePreparedTransactionsArray[i];
  985.     thePreparedTransactionsArray[i] = NULL;
  986.     Uint32 node_id = a_con->getConnectedNodeId();
  987.     if ((tp->getNodeSequence(node_id) == a_con->theNodeSequence) &&
  988.          tp->get_node_alive(node_id) ||
  989.          (tp->get_node_stopping(node_id) && 
  990.          ((a_con->theSendStatus == NdbConnection::sendABORT) ||
  991.           (a_con->theSendStatus == NdbConnection::sendABORTfail) ||
  992.           (a_con->theSendStatus == NdbConnection::sendCOMMITstate) ||
  993.           (a_con->theSendStatus == NdbConnection::sendCompleted)))) {
  994.       /*
  995.       We will send if
  996.       1) Node is alive and sequences are correct OR
  997.       2) Node is stopping and we only want to commit or abort
  998.       In a graceful stop situation we want to ensure quick aborts
  999.       of all transactions and commits and thus we allow aborts and
  1000.       commits to continue but not normal operations.
  1001.       */
  1002.       if (tp->check_send_size(node_id, a_con->get_send_size())) {
  1003.         if (a_con->doSend() == 0) {
  1004.           NDB_TICKS current_time = NdbTick_CurrentMillisecond();
  1005.           a_con->theStartTransTime = current_time;
  1006.           continue;
  1007.         } else {
  1008.           /*
  1009.           Although all precautions we did not manage to send the operations
  1010.           Must have been a dropped connection on the transporter side.
  1011.           We don't expect to be able to continue using this connection so
  1012.           we will treat it as a node failure.
  1013.           */
  1014.           TRACE_DEBUG("Send problem even after checking node status");
  1015.         }//if
  1016.       } else {
  1017.         /*
  1018.         The send buffer is currently full or at least close to. We will
  1019.         not allow a send to continue. We will set the connection so that
  1020.         it is indicated that we need to abort the transaction. If we were
  1021.         trying to commit or abort and got a send buffer we will not try
  1022.         again and will thus set the state to Aborted to avoid a more or
  1023.         less eternal loop of tries.
  1024.         */
  1025.         if (a_con->theSendStatus == NdbConnection::sendOperations) {
  1026.           a_con->setOperationErrorCodeAbort(4021);
  1027.           a_con->theCommitStatus = NdbConnection::NeedAbort;
  1028.           TRACE_DEBUG("Send buffer full and sendOperations");
  1029.         } else {
  1030.           a_con->setOperationErrorCodeAbort(4026);
  1031.           a_con->theCommitStatus = NdbConnection::Aborted;
  1032.           TRACE_DEBUG("Send buffer full, set state to Aborted");
  1033.         }//if
  1034.       }//if
  1035.     } else {
  1036. #ifdef VM_TRACE
  1037.       a_con->printState();
  1038. #endif
  1039.       if ((tp->getNodeSequence(node_id) == a_con->theNodeSequence) &&
  1040.           tp->get_node_stopping(node_id)) {
  1041.         /*
  1042.         The node we are connected to is currently in an early stopping phase
  1043.         of a graceful stop. We will not send the prepared transactions. We
  1044.         will simply refuse and let the application code handle the abort.
  1045.         */
  1046.         TRACE_DEBUG("Abort a transaction when stopping a node");
  1047.         a_con->setOperationErrorCodeAbort(4023);
  1048.         a_con->theCommitStatus = NdbConnection::NeedAbort;
  1049.       } else {
  1050.         /*
  1051.         The node is hard dead and we cannot continue. We will also release
  1052.         the connection to the free pool.
  1053.         */
  1054.         TRACE_DEBUG("The node was stone dead, inform about abort");
  1055.         a_con->setOperationErrorCodeAbort(4025);
  1056.         a_con->theReleaseOnClose = true;
  1057.         a_con->theTransactionIsStarted = false;
  1058.         a_con->theCommitStatus = NdbConnection::Aborted;
  1059.       }//if
  1060.     }//if
  1061.     a_con->theCompletionStatus = NdbConnection::CompletedFailure;
  1062.     a_con->handleExecuteCompletion();
  1063.     insert_completed_list(a_con);
  1064.   }//for
  1065.   theNoOfPreparedTransactions = 0;
  1066.   if (forceSend == 0) {
  1067.      tp->checkForceSend(theNdbBlockNumber);
  1068.   } else if (forceSend == 1) {
  1069.      tp->forceSend(theNdbBlockNumber);
  1070.   }//if
  1071.   return;
  1072. }//Ndb::sendPrepTrans()
  1073. /*****************************************************************************
  1074. void waitCompletedTransactions(int aMilliSecondsToWait, int noOfEventsToWaitFor);
  1075. Remark:   First send all prepared operations and then check if there are any
  1076.           transactions already completed. Do not wait for not completed
  1077.           transactions.
  1078. ******************************************************************************/
  1079. void
  1080. Ndb::waitCompletedTransactions(int aMilliSecondsToWait, 
  1081.        int noOfEventsToWaitFor)
  1082. {
  1083.   theImpl->theWaiter.m_state = NO_WAIT; 
  1084.   /**
  1085.    * theImpl->theWaiter.m_state = NO_WAIT; 
  1086.    * To ensure no messup with synchronous node fail handling
  1087.    * (see ReportFailure)
  1088.    */
  1089.   int waitTime = aMilliSecondsToWait;
  1090.   NDB_TICKS maxTime = NdbTick_CurrentMillisecond() + (NDB_TICKS)waitTime;
  1091.   theMinNoOfEventsToWakeUp = noOfEventsToWaitFor;
  1092.   do {
  1093.     if (waitTime < 1000) waitTime = 1000;
  1094.     NdbCondition_WaitTimeout(theImpl->theWaiter.m_condition,
  1095.      (NdbMutex*)theImpl->theWaiter.m_mutex,
  1096.      waitTime);
  1097.     if (theNoOfCompletedTransactions >= (Uint32)noOfEventsToWaitFor) {
  1098.       break;
  1099.     }//if
  1100.     theMinNoOfEventsToWakeUp = noOfEventsToWaitFor;
  1101.     waitTime = (int)(maxTime - NdbTick_CurrentMillisecond());
  1102.   } while (waitTime > 0);
  1103.   return;
  1104. }//Ndb::waitCompletedTransactions()
  1105. /*****************************************************************************
  1106. void sendPreparedTransactions(int forceSend = 0);
  1107. Remark:   First send all prepared operations and then check if there are any
  1108.           transactions already completed. Do not wait for not completed
  1109.           transactions.
  1110. ******************************************************************************/
  1111. void
  1112. Ndb::sendPreparedTransactions(int forceSend)
  1113. {
  1114.   TransporterFacade::instance()->lock_mutex();
  1115.   sendPrepTrans(forceSend);
  1116.   TransporterFacade::instance()->unlock_mutex();
  1117.   return;
  1118. }//Ndb::sendPreparedTransactions()
  1119. /*****************************************************************************
  1120. int sendPollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup = 1, int forceSend = 0);
  1121. Remark:   First send all prepared operations and then check if there are any
  1122.           transactions already completed. Wait for not completed
  1123.           transactions until the specified number have completed or until the
  1124.           timeout has occured. Timeout zero means no waiting time.
  1125. ******************************************************************************/
  1126. int
  1127. Ndb::sendPollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup, int forceSend)
  1128. {
  1129.   NdbConnection* tConArray[1024];
  1130.   Uint32         tNoCompletedTransactions;
  1131.   //theCurrentConnectCounter = 0;
  1132.   //theCurrentConnectIndex++;
  1133.   TransporterFacade::instance()->lock_mutex();
  1134.   sendPrepTrans(forceSend);
  1135.   if ((minNoOfEventsToWakeup <= 0) ||
  1136.       ((Uint32)minNoOfEventsToWakeup > theNoOfSentTransactions)) {
  1137.     minNoOfEventsToWakeup = theNoOfSentTransactions;
  1138.   }//if
  1139.   if ((theNoOfCompletedTransactions < (Uint32)minNoOfEventsToWakeup) &&
  1140.       (aMillisecondNumber > 0)) {
  1141.     waitCompletedTransactions(aMillisecondNumber, minNoOfEventsToWakeup);
  1142.     tNoCompletedTransactions = pollCompleted(tConArray);
  1143.   } else {
  1144.     tNoCompletedTransactions = pollCompleted(tConArray);
  1145.   }//if
  1146.   TransporterFacade::instance()->unlock_mutex();
  1147.   reportCallback(tConArray, tNoCompletedTransactions);
  1148.   return tNoCompletedTransactions;
  1149. }//Ndb::sendPollNdb()
  1150. /*****************************************************************************
  1151. int pollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup);
  1152. Remark:   Check if there are any transactions already completed. Wait for not
  1153.           completed transactions until the specified number have completed or
  1154.           until the timeout has occured. Timeout zero means no waiting time.
  1155. ******************************************************************************/
  1156. int
  1157. Ndb::pollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup)
  1158. {
  1159.   NdbConnection* tConArray[1024];
  1160.   Uint32         tNoCompletedTransactions;
  1161.   //theCurrentConnectCounter = 0;
  1162.   //theCurrentConnectIndex++;
  1163.   TransporterFacade::instance()->lock_mutex();
  1164.   if ((minNoOfEventsToWakeup == 0) ||
  1165.       ((Uint32)minNoOfEventsToWakeup > theNoOfSentTransactions)) {
  1166.     minNoOfEventsToWakeup = theNoOfSentTransactions;
  1167.   }//if
  1168.   if ((theNoOfCompletedTransactions < (Uint32)minNoOfEventsToWakeup) &&
  1169.       (aMillisecondNumber > 0)) {
  1170.     waitCompletedTransactions(aMillisecondNumber, minNoOfEventsToWakeup);
  1171.     tNoCompletedTransactions = pollCompleted(tConArray);
  1172.   } else {
  1173.     tNoCompletedTransactions = pollCompleted(tConArray);
  1174.   }//if
  1175.   TransporterFacade::instance()->unlock_mutex();
  1176.   reportCallback(tConArray, tNoCompletedTransactions);
  1177.   return tNoCompletedTransactions;
  1178. }//Ndb::sendPollNdbWithoutWait()
  1179. /*****************************************************************************
  1180. int receiveOptimisedResponse();
  1181. Return:  0 - Response received
  1182.         -1 - Timeout occured waiting for response
  1183.         -2 - Node failure interupted wait for response
  1184. ******************************************************************************/
  1185. int
  1186. Ndb::receiveResponse(int waitTime){
  1187.   int tResultCode;
  1188.   TransporterFacade::instance()->checkForceSend(theNdbBlockNumber);
  1189.   
  1190.   theImpl->theWaiter.wait(waitTime);
  1191.   
  1192.   if(theImpl->theWaiter.m_state == NO_WAIT) {
  1193.     tResultCode = 0;
  1194.   } else {
  1195. #ifdef VM_TRACE
  1196.     ndbout << "ERR: receiveResponse - theImpl->theWaiter.m_state = ";
  1197.     ndbout << theImpl->theWaiter.m_state << endl;
  1198. #endif
  1199.     if (theImpl->theWaiter.m_state == WAIT_NODE_FAILURE){
  1200.       tResultCode = -2;
  1201.     } else {
  1202.       tResultCode = -1;
  1203.     }
  1204.     theImpl->theWaiter.m_state = NO_WAIT;
  1205.   }
  1206.   return tResultCode;
  1207. }//Ndb::receiveResponse()
  1208. int
  1209. Ndb::sendRecSignal(Uint16 node_id,
  1210.    Uint32 aWaitState,
  1211.    NdbApiSignal* aSignal,
  1212.                    Uint32 conn_seq)
  1213. {
  1214.   /*
  1215.   In most situations 0 is returned.
  1216.   In error cases we have 5 different cases
  1217.   -1: Send ok, time out in waiting for reply
  1218.   -2: Node has failed
  1219.   -3: Send buffer not full, send failed yet
  1220.   -4: Send buffer full
  1221.   -5: Node is currently stopping
  1222.   */
  1223.   int return_code;
  1224.   TransporterFacade* tp = TransporterFacade::instance();
  1225.   Uint32 send_size = 1; // Always sends one signal only 
  1226.   tp->lock_mutex();
  1227.   // Protected area
  1228.   if ((tp->get_node_alive(node_id)) &&
  1229.       ((tp->getNodeSequence(node_id) == conn_seq) ||
  1230.        (conn_seq == 0))) {
  1231.     if (tp->check_send_size(node_id, send_size)) {
  1232.       return_code = tp->sendSignal(aSignal, node_id);
  1233.       if (return_code != -1) {
  1234.         theImpl->theWaiter.m_node = node_id;
  1235.         theImpl->theWaiter.m_state = aWaitState;
  1236.         return_code = receiveResponse();
  1237.       } else {
  1238. return_code = -3;
  1239.       }
  1240.     } else {
  1241.       return_code = -4;
  1242.     }//if
  1243.   } else {
  1244.     if ((tp->get_node_stopping(node_id)) &&
  1245.         ((tp->getNodeSequence(node_id) == conn_seq) ||
  1246.          (conn_seq == 0))) {
  1247.       return_code = -5;
  1248.     } else {
  1249.       return_code = -2;
  1250.     }//if
  1251.   }//if
  1252.   tp->unlock_mutex();
  1253.   // End of protected area
  1254.   return return_code;
  1255. }//Ndb::sendRecSignal()
  1256. void
  1257. NdbConnection::sendTC_COMMIT_ACK(NdbApiSignal * aSignal,
  1258.  Uint32 transId1, Uint32 transId2, 
  1259.  Uint32 aTCRef){
  1260. #ifdef MARKER_TRACE
  1261.   ndbout_c("Sending TC_COMMIT_ACK(0x%.8x, 0x%.8x) to -> %d",
  1262.    transId1,
  1263.    transId2,
  1264.    refToNode(aTCRef));
  1265. #endif  
  1266.   TransporterFacade *tp = TransporterFacade::instance();
  1267.   aSignal->theTrace                = TestOrd::TraceAPI;
  1268.   aSignal->theReceiversBlockNumber = DBTC;
  1269.   aSignal->theVerId_signalNumber   = GSN_TC_COMMIT_ACK;
  1270.   aSignal->theLength               = 2;
  1271.   Uint32 * dataPtr = aSignal->getDataPtrSend();
  1272.   dataPtr[0] = transId1;
  1273.   dataPtr[1] = transId2;
  1274.   tp->sendSignal(aSignal, refToNode(aTCRef));
  1275. }