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

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. #define DBTUP_C
  14. #include "Dbtup.hpp"
  15. #include <RefConvert.hpp>
  16. #include <ndb_limits.h>
  17. #include <pc.hpp>
  18. #include <AttributeDescriptor.hpp>
  19. #include "AttributeOffset.hpp"
  20. #include <AttributeHeader.hpp>
  21. #include <signaldata/TuxMaint.hpp>
  22. #define ljam() { jamLine(28000 + __LINE__); }
  23. #define ljamEntry() { jamEntryLine(28000 + __LINE__); }
  24. // methods used by ordered index
  25. void
  26. Dbtup::tuxGetTupAddr(Uint32 fragPtrI, Uint32 pageId, Uint32 pageOffset, Uint32& tupAddr)
  27. {
  28.   ljamEntry();
  29.   FragrecordPtr fragPtr;
  30.   fragPtr.i = fragPtrI;
  31.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  32.   TablerecPtr tablePtr;
  33.   tablePtr.i = fragPtr.p->fragTableId;
  34.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  35.   PagePtr pagePtr;
  36.   pagePtr.i = pageId;
  37.   ptrCheckGuard(pagePtr, cnoOfPage, page);
  38.   Uint32 fragPageId = pagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
  39.   Uint32 tupheadsize = tablePtr.p->tupheadsize;
  40.   ndbrequire(pageOffset >= ZPAGE_HEADER_SIZE);
  41.   Uint32 offset = pageOffset - ZPAGE_HEADER_SIZE;
  42.   ndbrequire(offset % tupheadsize == 0);
  43.   Uint32 pageIndex = (offset / tupheadsize) << 1;
  44.   tupAddr = (fragPageId << MAX_TUPLES_BITS) | pageIndex;
  45. }
  46. int
  47. Dbtup::tuxAllocNode(Signal* signal, Uint32 fragPtrI, Uint32& pageId, Uint32& pageOffset, Uint32*& node)
  48. {
  49.   ljamEntry();
  50.   FragrecordPtr fragPtr;
  51.   fragPtr.i = fragPtrI;
  52.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  53.   TablerecPtr tablePtr;
  54.   tablePtr.i = fragPtr.p->fragTableId;
  55.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  56.   PagePtr pagePtr;
  57.   terrorCode = 0;
  58.   if (! allocTh(fragPtr.p, tablePtr.p, NORMAL_PAGE, signal, pageOffset, pagePtr)) {
  59.     ljam();
  60.     ndbrequire(terrorCode != 0);
  61.     return terrorCode;
  62.   }
  63.   pageId = pagePtr.i;
  64.   Uint32 attrDescIndex = tablePtr.p->tabDescriptor + (0 << ZAD_LOG_SIZE);
  65.   Uint32 attrDataOffset = AttributeOffset::getOffset(tableDescriptor[attrDescIndex + 1].tabDescr);
  66.   node = &pagePtr.p->pageWord[pageOffset] + attrDataOffset;
  67.   return 0;
  68. }
  69. void
  70. Dbtup::tuxFreeNode(Signal* signal, Uint32 fragPtrI, Uint32 pageId, Uint32 pageOffset, Uint32* node)
  71. {
  72.   ljamEntry();
  73.   FragrecordPtr fragPtr;
  74.   fragPtr.i = fragPtrI;
  75.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  76.   TablerecPtr tablePtr;
  77.   tablePtr.i = fragPtr.p->fragTableId;
  78.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  79.   PagePtr pagePtr;
  80.   pagePtr.i = pageId;
  81.   ptrCheckGuard(pagePtr, cnoOfPage, page);
  82.   Uint32 attrDescIndex = tablePtr.p->tabDescriptor + (0 << ZAD_LOG_SIZE);
  83.   Uint32 attrDataOffset = AttributeOffset::getOffset(tableDescriptor[attrDescIndex + 1].tabDescr);
  84.   ndbrequire(node == &pagePtr.p->pageWord[pageOffset] + attrDataOffset);
  85.   freeTh(fragPtr.p, tablePtr.p, signal, pagePtr.p, pageOffset);
  86. }
  87. void
  88. Dbtup::tuxGetNode(Uint32 fragPtrI, Uint32 pageId, Uint32 pageOffset, Uint32*& node)
  89. {
  90.   ljamEntry();
  91.   FragrecordPtr fragPtr;
  92.   fragPtr.i = fragPtrI;
  93.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  94.   TablerecPtr tablePtr;
  95.   tablePtr.i = fragPtr.p->fragTableId;
  96.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  97.   PagePtr pagePtr;
  98.   pagePtr.i = pageId;
  99.   ptrCheckGuard(pagePtr, cnoOfPage, page);
  100.   Uint32 attrDescIndex = tablePtr.p->tabDescriptor + (0 << ZAD_LOG_SIZE);
  101.   Uint32 attrDataOffset = AttributeOffset::getOffset(tableDescriptor[attrDescIndex + 1].tabDescr);
  102.   node = &pagePtr.p->pageWord[pageOffset] + attrDataOffset;
  103. }
  104. int
  105. Dbtup::tuxReadAttrs(Uint32 fragPtrI, Uint32 pageId, Uint32 pageOffset, Uint32 tupVersion, const Uint32* attrIds, Uint32 numAttrs, Uint32* dataOut)
  106. {
  107.   ljamEntry();
  108.   // use own variables instead of globals
  109.   FragrecordPtr fragPtr;
  110.   fragPtr.i = fragPtrI;
  111.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  112.   TablerecPtr tablePtr;
  113.   tablePtr.i = fragPtr.p->fragTableId;
  114.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  115.   PagePtr pagePtr;
  116.   pagePtr.i = pageId;
  117.   ptrCheckGuard(pagePtr, cnoOfPage, page);
  118.   // search for tuple version if not original
  119.   if (pagePtr.p->pageWord[pageOffset + 1] != tupVersion) {
  120.     ljam();
  121.     OperationrecPtr opPtr;
  122.     opPtr.i = pagePtr.p->pageWord[pageOffset];
  123.     Uint32 loopGuard = 0;
  124.     while (true) {
  125.       ptrCheckGuard(opPtr, cnoOfOprec, operationrec);
  126.       if (opPtr.p->realPageIdC != RNIL) {
  127.         // update page and offset
  128.         pagePtr.i = opPtr.p->realPageIdC;
  129.         pageOffset = opPtr.p->pageOffsetC;
  130.         ptrCheckGuard(pagePtr, cnoOfPage, page);
  131.         if (pagePtr.p->pageWord[pageOffset + 1] == tupVersion) {
  132.           ljam();
  133.           break;
  134.         }
  135.       }
  136.       ljam();
  137.       opPtr.i = opPtr.p->nextActiveOp;
  138.       ndbrequire(++loopGuard < (1 << ZTUP_VERSION_BITS));
  139.     }
  140.   }
  141.   // read key attributes from found tuple version
  142.   // save globals
  143.   TablerecPtr tabptr_old = tabptr;
  144.   FragrecordPtr fragptr_old = fragptr;
  145.   OperationrecPtr operPtr_old = operPtr;
  146.   // new globals
  147.   tabptr = tablePtr;
  148.   fragptr = fragPtr;
  149.   operPtr.i = RNIL;
  150.   operPtr.p = NULL;
  151.   // do it
  152.   int ret = readAttributes(pagePtr.p, pageOffset, attrIds, numAttrs, dataOut, ZNIL, true);
  153.   // restore globals
  154.   tabptr = tabptr_old;
  155.   fragptr = fragptr_old;
  156.   operPtr = operPtr_old;
  157.   // done
  158.   if (ret == -1) {
  159.     ret = terrorCode ? (-(int)terrorCode) : -1;
  160.   }
  161.   return ret;
  162. }
  163. int
  164. Dbtup::tuxReadPk(Uint32 fragPtrI, Uint32 pageId, Uint32 pageOffset, Uint32* dataOut)
  165. {
  166.   ljamEntry();
  167.   // use own variables instead of globals
  168.   FragrecordPtr fragPtr;
  169.   fragPtr.i = fragPtrI;
  170.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  171.   TablerecPtr tablePtr;
  172.   tablePtr.i = fragPtr.p->fragTableId;
  173.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  174.   PagePtr pagePtr;
  175.   pagePtr.i = pageId;
  176.   ptrCheckGuard(pagePtr, cnoOfPage, page);
  177.   const Uint32 tabDescriptor = tablePtr.p->tabDescriptor;
  178.   const Uint32* attrIds = &tableDescriptor[tablePtr.p->readKeyArray].tabDescr;
  179.   const Uint32 numAttrs = tablePtr.p->noOfKeyAttr;
  180.   // read pk attributes from original tuple
  181.   // save globals
  182.   TablerecPtr tabptr_old = tabptr;
  183.   FragrecordPtr fragptr_old = fragptr;
  184.   OperationrecPtr operPtr_old = operPtr;
  185.   // new globals
  186.   tabptr = tablePtr;
  187.   fragptr = fragPtr;
  188.   operPtr.i = RNIL;
  189.   operPtr.p = NULL;
  190.   // do it
  191.   int ret = readAttributes(pagePtr.p, pageOffset, attrIds,
  192.    numAttrs, dataOut, ZNIL, true);
  193.   // restore globals
  194.   tabptr = tabptr_old;
  195.   fragptr = fragptr_old;
  196.   operPtr = operPtr_old;
  197.   // done
  198.   if (ret != -1) {
  199.     // remove headers
  200.     Uint32 n = 0;
  201.     Uint32 i = 0;
  202.     while (n < numAttrs) {
  203.       const AttributeHeader ah(dataOut[i]);
  204.       Uint32 size = ah.getDataSize();
  205.       ndbrequire(size != 0);
  206.       for (Uint32 j = 0; j < size; j++) {
  207.         dataOut[i + j - n] = dataOut[i + j + 1];
  208.       }
  209.       n += 1;
  210.       i += 1 + size;
  211.     }
  212.     ndbrequire((int)i == ret);
  213.     ret -= numAttrs;
  214.   } else {
  215.     ret = terrorCode ? (-(int)terrorCode) : -1;
  216.   }
  217.   return ret;
  218. }
  219. bool
  220. Dbtup::tuxQueryTh(Uint32 fragPtrI, Uint32 tupAddr, Uint32 tupVersion, Uint32 transId1, Uint32 transId2, Uint32 savePointId)
  221. {
  222.   ljamEntry();
  223.   FragrecordPtr fragPtr;
  224.   fragPtr.i = fragPtrI;
  225.   ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  226.   TablerecPtr tablePtr;
  227.   tablePtr.i = fragPtr.p->fragTableId;
  228.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  229.   // get page
  230.   PagePtr pagePtr;
  231.   Uint32 fragPageId = tupAddr >> MAX_TUPLES_BITS;
  232.   Uint32 pageIndex = tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
  233.   // use temp op rec
  234.   Operationrec tempOp;
  235.   tempOp.fragPageId = fragPageId;
  236.   tempOp.pageIndex = pageIndex;
  237.   tempOp.transid1 = transId1;
  238.   tempOp.transid2 = transId2;
  239.   tempOp.savePointId = savePointId;
  240.   tempOp.optype = ZREAD;
  241.   tempOp.dirtyOp = 1;
  242.   if (getPage(pagePtr, &tempOp, fragPtr.p, tablePtr.p)) {
  243.     /*
  244.     * We use the normal getPage which will return the tuple to be used
  245.     * for this transaction and savepoint id.  If its tuple version
  246.     * equals the requested then we have a visible tuple otherwise not.
  247.     */
  248.     ljam();
  249.     Uint32 read_tupVersion = pagePtr.p->pageWord[tempOp.pageOffset + 1];
  250.     if (read_tupVersion == tupVersion) {
  251.       ljam();
  252.       return true;
  253.     }
  254.   }
  255.   return false;
  256. }
  257. // ordered index build
  258. //#define TIME_MEASUREMENT
  259. #ifdef TIME_MEASUREMENT
  260.   static Uint32 time_events;
  261.   NDB_TICKS tot_time_passed;
  262.   Uint32 number_events;
  263. #endif
  264. void
  265. Dbtup::execBUILDINDXREQ(Signal* signal)
  266. {
  267.   ljamEntry();
  268. #ifdef TIME_MEASUREMENT
  269.   time_events = 0;
  270.   tot_time_passed = 0;
  271.   number_events = 1;
  272. #endif
  273.   // get new operation
  274.   BuildIndexPtr buildPtr;
  275.   if (! c_buildIndexList.seize(buildPtr)) {
  276.     ljam();
  277.     BuildIndexRec buildRec;
  278.     memcpy(buildRec.m_request, signal->theData, sizeof(buildRec.m_request));
  279.     buildRec.m_errorCode = BuildIndxRef::Busy;
  280.     buildIndexReply(signal, &buildRec);
  281.     return;
  282.   }
  283.   memcpy(buildPtr.p->m_request, signal->theData, sizeof(buildPtr.p->m_request));
  284.   // check
  285.   buildPtr.p->m_errorCode = BuildIndxRef::NoError;
  286.   do {
  287.     const BuildIndxReq* buildReq = (const BuildIndxReq*)buildPtr.p->m_request;
  288.     if (buildReq->getTableId() >= cnoOfTablerec) {
  289.       ljam();
  290.       buildPtr.p->m_errorCode = BuildIndxRef::InvalidPrimaryTable;
  291.       break;
  292.     }
  293.     TablerecPtr tablePtr;
  294.     tablePtr.i = buildReq->getTableId();
  295.     ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  296.     if (tablePtr.p->tableStatus != DEFINED) {
  297.       ljam();
  298.       buildPtr.p->m_errorCode = BuildIndxRef::InvalidPrimaryTable;
  299.       break;
  300.     }
  301.     if (! DictTabInfo::isOrderedIndex(buildReq->getIndexType())) {
  302.       ljam();
  303.       buildPtr.p->m_errorCode = BuildIndxRef::InvalidIndexType;
  304.       break;
  305.     }
  306.     const ArrayList<TupTriggerData>& triggerList = tablePtr.p->tuxCustomTriggers;
  307.     TriggerPtr triggerPtr;
  308.     triggerList.first(triggerPtr);
  309.     while (triggerPtr.i != RNIL) {
  310.       if (triggerPtr.p->indexId == buildReq->getIndexId()) {
  311.         ljam();
  312.         break;
  313.       }
  314.       triggerList.next(triggerPtr);
  315.     }
  316.     if (triggerPtr.i == RNIL) {
  317.       ljam();
  318.       // trigger was not created
  319.       buildPtr.p->m_errorCode = BuildIndxRef::InternalError;
  320.       break;
  321.     }
  322.     buildPtr.p->m_triggerPtrI = triggerPtr.i;
  323.     // set to first tuple position
  324.     buildPtr.p->m_fragNo = 0;
  325.     buildPtr.p->m_pageId = 0;
  326.     buildPtr.p->m_tupleNo = 0;
  327.     // start build
  328.     buildIndex(signal, buildPtr.i);
  329.     return;
  330.   } while (0);
  331.   // check failed
  332.   buildIndexReply(signal, buildPtr.p);
  333.   c_buildIndexList.release(buildPtr);
  334. }
  335. void
  336. Dbtup::buildIndex(Signal* signal, Uint32 buildPtrI)
  337. {
  338.   // get build record
  339.   BuildIndexPtr buildPtr;
  340.   buildPtr.i = buildPtrI;
  341.   c_buildIndexList.getPtr(buildPtr);
  342.   const BuildIndxReq* buildReq = (const BuildIndxReq*)buildPtr.p->m_request;
  343.   // get table
  344.   TablerecPtr tablePtr;
  345.   tablePtr.i = buildReq->getTableId();
  346.   ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
  347.   // get trigger
  348.   TriggerPtr triggerPtr;
  349.   triggerPtr.i = buildPtr.p->m_triggerPtrI;
  350.   c_triggerPool.getPtr(triggerPtr);
  351.   ndbrequire(triggerPtr.p->indexId == buildReq->getIndexId());
  352. #ifdef TIME_MEASUREMENT
  353.   MicroSecondTimer start;
  354.   MicroSecondTimer stop;
  355.   NDB_TICKS time_passed;
  356. #endif
  357.   do {
  358.     // get fragment
  359.     FragrecordPtr fragPtr;
  360.     if (buildPtr.p->m_fragNo == 2 * MAX_FRAG_PER_NODE) {
  361.       ljam();
  362.       // build ready
  363.       buildIndexReply(signal, buildPtr.p);
  364.       c_buildIndexList.release(buildPtr);
  365.       return;
  366.     }
  367.     ndbrequire(buildPtr.p->m_fragNo < 2 * MAX_FRAG_PER_NODE);
  368.     fragPtr.i = tablePtr.p->fragrec[buildPtr.p->m_fragNo];
  369.     if (fragPtr.i == RNIL) {
  370.       ljam();
  371.       buildPtr.p->m_fragNo++;
  372.       buildPtr.p->m_pageId = 0;
  373.       buildPtr.p->m_tupleNo = 0;
  374.       break;
  375.     }
  376.     ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
  377.     // get page
  378.     PagePtr pagePtr;
  379.     if (buildPtr.p->m_pageId >= fragPtr.p->noOfPages) {
  380.       ljam();
  381.       buildPtr.p->m_fragNo++;
  382.       buildPtr.p->m_pageId = 0;
  383.       buildPtr.p->m_tupleNo = 0;
  384.       break;
  385.     }
  386.     Uint32 realPageId = getRealpid(fragPtr.p, buildPtr.p->m_pageId);
  387.     pagePtr.i = realPageId;
  388.     ptrCheckGuard(pagePtr, cnoOfPage, page);
  389.     const Uint32 pageState = pagePtr.p->pageWord[ZPAGE_STATE_POS];
  390.     if (pageState != ZTH_MM_FREE &&
  391.         pageState != ZTH_MM_FREE_COPY &&
  392.         pageState != ZTH_MM_FULL &&
  393.         pageState != ZTH_MM_FULL_COPY) {
  394.       ljam();
  395.       buildPtr.p->m_pageId++;
  396.       buildPtr.p->m_tupleNo = 0;
  397.       break;
  398.     }
  399.     // get tuple
  400.     const Uint32 tupheadsize = tablePtr.p->tupheadsize;
  401.     Uint32 pageOffset = ZPAGE_HEADER_SIZE + buildPtr.p->m_tupleNo * tupheadsize;
  402.     if (pageOffset + tupheadsize > ZWORDS_ON_PAGE) {
  403.       ljam();
  404.       buildPtr.p->m_pageId++;
  405.       buildPtr.p->m_tupleNo = 0;
  406.       break;
  407.     }
  408.     // skip over free tuple
  409.     bool isFree = false;
  410.     if (pageState == ZTH_MM_FREE ||
  411.         pageState == ZTH_MM_FREE_COPY) {
  412.       ljam();
  413.       if ((pagePtr.p->pageWord[pageOffset] >> 16) == tupheadsize) {
  414.         // verify it really is free  XXX far too expensive
  415.         Uint32 nextTuple = pagePtr.p->pageWord[ZFREELIST_HEADER_POS] >> 16;
  416.         ndbrequire(nextTuple != 0);
  417.         while (nextTuple != 0) {
  418.           ljam();
  419.           if (nextTuple == pageOffset) {
  420.             ljam();
  421.             isFree = true;
  422.             break;
  423.           }
  424.           nextTuple = pagePtr.p->pageWord[nextTuple] & 0xffff;
  425.         }
  426.       }
  427.     }
  428.     if (isFree) {
  429.       ljam();
  430.       buildPtr.p->m_tupleNo++;
  431.       break;
  432.     }
  433.     Uint32 tupVersion = pagePtr.p->pageWord[pageOffset + 1];
  434.     OperationrecPtr pageOperPtr;
  435.     pageOperPtr.i = pagePtr.p->pageWord[pageOffset];
  436.     if (pageOperPtr.i != RNIL) {
  437.       /*
  438.       If there is an ongoing operation on the tuple then it is either a
  439.       copy tuple or an original tuple with an ongoing transaction. In
  440.       both cases realPageId and pageOffset refer to the original tuple.
  441.       The tuple address stored in TUX will always be the original tuple
  442.       but with the tuple version of the tuple we found.
  443.       This is necessary to avoid having to update TUX at abort of
  444.       update. If an update aborts then the copy tuple is copied to
  445.       the original tuple. The build will however have found that
  446.       tuple as a copy tuple. The original tuple is stable and is thus
  447.       preferrable to store in TUX.
  448.       */
  449.       ljam();
  450.       ptrCheckGuard(pageOperPtr, cnoOfOprec, operationrec);
  451.       realPageId = pageOperPtr.p->realPageId;
  452.       pageOffset = pageOperPtr.p->pageOffset;
  453.     }//if
  454. #ifdef TIME_MEASUREMENT
  455.     NdbTick_getMicroTimer(&start);
  456. #endif
  457.     // add to index
  458.     TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
  459.     req->errorCode = RNIL;
  460.     req->tableId = tablePtr.i;
  461.     req->indexId = triggerPtr.p->indexId;
  462.     req->fragId = tablePtr.p->fragid[buildPtr.p->m_fragNo];
  463.     req->pageId = realPageId;
  464.     req->pageOffset = pageOffset;
  465.     req->tupVersion = tupVersion;
  466.     req->opInfo = TuxMaintReq::OpAdd;
  467.     EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
  468.         signal, TuxMaintReq::SignalLength);
  469.     ljamEntry();
  470.     if (req->errorCode != 0) {
  471.       switch (req->errorCode) {
  472.       case TuxMaintReq::NoMemError:
  473.         ljam();
  474.         buildPtr.p->m_errorCode = BuildIndxRef::AllocationFailure;
  475.         break;
  476.       default:
  477.         ndbrequire(false);
  478.         break;
  479.       }
  480.       buildIndexReply(signal, buildPtr.p);
  481.       c_buildIndexList.release(buildPtr);
  482.       return;
  483.     }
  484. #ifdef TIME_MEASUREMENT
  485.     NdbTick_getMicroTimer(&stop);
  486.     time_passed = NdbTick_getMicrosPassed(start, stop);
  487.     if (time_passed < 1000) {
  488.       time_events++;
  489.       tot_time_passed += time_passed;
  490.       if (time_events == number_events) {
  491.         NDB_TICKS mean_time_passed = tot_time_passed / (NDB_TICKS)number_events;
  492.         ndbout << "Number of events = " << number_events;
  493.         ndbout << " Mean time passed = " << mean_time_passed << endl;
  494.         number_events <<= 1;
  495.         tot_time_passed = (NDB_TICKS)0;
  496.         time_events = 0;
  497.       }//if
  498.     }
  499. #endif
  500.     // next tuple
  501.     buildPtr.p->m_tupleNo++;
  502.     break;
  503.   } while (0);
  504.   signal->theData[0] = ZBUILD_INDEX;
  505.   signal->theData[1] = buildPtr.i;
  506.   sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
  507. }
  508. void
  509. Dbtup::buildIndexReply(Signal* signal, const BuildIndexRec* buildPtrP)
  510. {
  511.   const BuildIndxReq* const buildReq = (const BuildIndxReq*)buildPtrP->m_request;
  512.   // conf is subset of ref
  513.   BuildIndxRef* rep = (BuildIndxRef*)signal->getDataPtr();
  514.   rep->setUserRef(buildReq->getUserRef());
  515.   rep->setConnectionPtr(buildReq->getConnectionPtr());
  516.   rep->setRequestType(buildReq->getRequestType());
  517.   rep->setTableId(buildReq->getTableId());
  518.   rep->setIndexType(buildReq->getIndexType());
  519.   rep->setIndexId(buildReq->getIndexId());
  520.   // conf
  521.   if (buildPtrP->m_errorCode == BuildIndxRef::NoError) {
  522.     ljam();
  523.     sendSignal(rep->getUserRef(), GSN_BUILDINDXCONF,
  524.         signal, BuildIndxConf::SignalLength, JBB);
  525.     return;
  526.   }
  527.   // ref
  528.   rep->setErrorCode(buildPtrP->m_errorCode);
  529.   sendSignal(rep->getUserRef(), GSN_BUILDINDXREF,
  530.       signal, BuildIndxRef::SignalLength, JBB);
  531. }