Commit 31a5dd0a authored by joreland@mysql.com's avatar joreland@mysql.com

Implemented general purpose psuedo columns: row_count & fragment

Implemented new interpreter instruction: exit_ok_last

This two new features combined can be used to make fast select count
 
parent 7a3d0c83
...@@ -30,6 +30,12 @@ class AttributeHeader { ...@@ -30,6 +30,12 @@ class AttributeHeader {
friend class Suma; friend class Suma;
public: public:
/**
* Psuedo columns
*/
STATIC_CONST( FRAGMENT = 0xFFFE );
STATIC_CONST( ROW_COUNT = 0xFFFD );
/** Initialize AttributeHeader at location aHeaderPtr */ /** Initialize AttributeHeader at location aHeaderPtr */
static AttributeHeader& init(void* aHeaderPtr, Uint32 anAttributeId, static AttributeHeader& init(void* aHeaderPtr, Uint32 anAttributeId,
Uint32 aDataSize); Uint32 aDataSize);
......
...@@ -23,14 +23,8 @@ ...@@ -23,14 +23,8 @@
* *
* When adding a new signal, remember to update MAX_GSN and SignalNames.cpp * When adding a new signal, remember to update MAX_GSN and SignalNames.cpp
*/ */
const GlobalSignalNumber MAX_GSN = 712; const GlobalSignalNumber MAX_GSN = 712;
struct GsnName { struct GsnName {
GlobalSignalNumber gsn; GlobalSignalNumber gsn;
const char * name; const char * name;
...@@ -898,7 +892,6 @@ extern const GlobalSignalNumber NO_OF_SIGNAL_NAMES; ...@@ -898,7 +892,6 @@ extern const GlobalSignalNumber NO_OF_SIGNAL_NAMES;
#define GSN_TUX_MAINT_REF 679 #define GSN_TUX_MAINT_REF 679
// not used 680 // not used 680
// not used 712
// not used 681 // not used 681
/** /**
...@@ -951,6 +944,6 @@ extern const GlobalSignalNumber NO_OF_SIGNAL_NAMES; ...@@ -951,6 +944,6 @@ extern const GlobalSignalNumber NO_OF_SIGNAL_NAMES;
#define GSN_TUX_BOUND_INFO 710 #define GSN_TUX_BOUND_INFO 710
#define GSN_ACC_LOCKREQ 711 #define GSN_ACC_LOCKREQ 711
#define GSN_READ_ROWCOUNT_REQ 712
#endif #endif
...@@ -80,7 +80,7 @@ class TupKeyConf { ...@@ -80,7 +80,7 @@ class TupKeyConf {
friend bool printTUPKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo); friend bool printTUPKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
public: public:
STATIC_CONST( SignalLength = 6 ); STATIC_CONST( SignalLength = 5 );
private: private:
...@@ -88,11 +88,10 @@ private: ...@@ -88,11 +88,10 @@ private:
* DATA VARIABLES * DATA VARIABLES
*/ */
Uint32 userPtr; Uint32 userPtr;
Uint32 pageId;
Uint32 pageIndex;
Uint32 readLength; Uint32 readLength;
Uint32 writeLength; Uint32 writeLength;
Uint32 noFiredTriggers; Uint32 noFiredTriggers;
Uint32 lastRow;
}; };
class TupKeyRef { class TupKeyRef {
......
...@@ -609,6 +609,20 @@ public: ...@@ -609,6 +609,20 @@ public:
int interpret_exit_nok(Uint32 ErrorCode); int interpret_exit_nok(Uint32 ErrorCode);
int interpret_exit_nok(); int interpret_exit_nok();
/**
* Interpreted program instruction:
*
* For scanning transactions,
* return this row, but no more from this fragment
*
* For non-scanning transactions,
* abort the whole transaction.
*
* @return -1 if unsuccessful.
*/
int interpret_exit_last_row();
/** /**
* Interpreted program instruction: * Interpreted program instruction:
* Define a subroutine in an interpreted operation. * Define a subroutine in an interpreted operation.
......
...@@ -926,6 +926,7 @@ private: ...@@ -926,6 +926,7 @@ private:
void execACC_OVER_REC(Signal* signal); void execACC_OVER_REC(Signal* signal);
void execACC_SAVE_PAGES(Signal* signal); void execACC_SAVE_PAGES(Signal* signal);
void execNEXTOPERATION(Signal* signal); void execNEXTOPERATION(Signal* signal);
void execREAD_ROWCOUNTREQ(Signal* signal);
// Received signals // Received signals
void execSTTOR(Signal* signal); void execSTTOR(Signal* signal);
......
...@@ -148,6 +148,7 @@ Dbacc::Dbacc(const class Configuration & conf): ...@@ -148,6 +148,7 @@ Dbacc::Dbacc(const class Configuration & conf):
addRecSignal(GSN_ACC_OVER_REC, &Dbacc::execACC_OVER_REC); addRecSignal(GSN_ACC_OVER_REC, &Dbacc::execACC_OVER_REC);
addRecSignal(GSN_ACC_SAVE_PAGES, &Dbacc::execACC_SAVE_PAGES); addRecSignal(GSN_ACC_SAVE_PAGES, &Dbacc::execACC_SAVE_PAGES);
addRecSignal(GSN_NEXTOPERATION, &Dbacc::execNEXTOPERATION); addRecSignal(GSN_NEXTOPERATION, &Dbacc::execNEXTOPERATION);
addRecSignal(GSN_READ_ROWCOUNT_REQ, &Dbacc::execREAD_ROWCOUNTREQ);
// Received signals // Received signals
addRecSignal(GSN_STTOR, &Dbacc::execSTTOR); addRecSignal(GSN_STTOR, &Dbacc::execSTTOR);
......
...@@ -13384,3 +13384,17 @@ void Dbacc::execSET_VAR_REQ(Signal* signal) ...@@ -13384,3 +13384,17 @@ void Dbacc::execSET_VAR_REQ(Signal* signal)
#endif #endif
}//execSET_VAR_REQ() }//execSET_VAR_REQ()
void
Dbacc::execREAD_ROWCOUNTREQ(Signal* signal){
jamEntry();
fragrecptr.i = signal->theData[0];
ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
rootfragrecptr.i = fragrecptr.p->myroot;
ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
Uint64 tmp = rootfragrecptr.p->noOfElements;
Uint32 * src = (Uint32*)&tmp;
signal->theData[0] = src[0];
signal->theData[1] = src[1];
}
...@@ -575,6 +575,7 @@ public: ...@@ -575,6 +575,7 @@ public:
Uint8 scanReleaseCounter; Uint8 scanReleaseCounter;
Uint8 scanTcWaiting; Uint8 scanTcWaiting;
Uint8 scanKeyinfoFlag; Uint8 scanKeyinfoFlag;
Uint8 m_last_row;
}; // Size 272 bytes }; // Size 272 bytes
typedef Ptr<ScanRecord> ScanRecordPtr; typedef Ptr<ScanRecord> ScanRecordPtr;
...@@ -2097,6 +2098,7 @@ private: ...@@ -2097,6 +2098,7 @@ private:
void execSTART_EXEC_SR(Signal* signal); void execSTART_EXEC_SR(Signal* signal);
void execEXEC_SRREQ(Signal* signal); void execEXEC_SRREQ(Signal* signal);
void execEXEC_SRCONF(Signal* signal); void execEXEC_SRCONF(Signal* signal);
void execREAD_ROWCOUNTREQ(Signal* signal);
void execDUMP_STATE_ORD(Signal* signal); void execDUMP_STATE_ORD(Signal* signal);
void execACC_COM_BLOCK(Signal* signal); void execACC_COM_BLOCK(Signal* signal);
......
...@@ -323,6 +323,8 @@ Dblqh::Dblqh(const class Configuration & conf): ...@@ -323,6 +323,8 @@ Dblqh::Dblqh(const class Configuration & conf):
addRecSignal(GSN_TUX_ADD_ATTRCONF, &Dblqh::execTUX_ADD_ATTRCONF); addRecSignal(GSN_TUX_ADD_ATTRCONF, &Dblqh::execTUX_ADD_ATTRCONF);
addRecSignal(GSN_TUX_ADD_ATTRREF, &Dblqh::execTUX_ADD_ATTRREF); addRecSignal(GSN_TUX_ADD_ATTRREF, &Dblqh::execTUX_ADD_ATTRREF);
addRecSignal(GSN_READ_ROWCOUNT_REQ, &Dblqh::execREAD_ROWCOUNTREQ);
initData(); initData();
#ifdef VM_TRACE #ifdef VM_TRACE
......
...@@ -2511,6 +2511,21 @@ Dblqh::updatePackedList(Signal* signal, HostRecord * ahostptr, Uint16 hostId) ...@@ -2511,6 +2511,21 @@ Dblqh::updatePackedList(Signal* signal, HostRecord * ahostptr, Uint16 hostId)
}//if }//if
}//Dblqh::updatePackedList() }//Dblqh::updatePackedList()
void
Dblqh::execREAD_ROWCOUNTREQ(Signal* signal){
jamEntry();
TcConnectionrecPtr regTcPtr;
regTcPtr.i = signal->theData[0];
ptrCheckGuard(regTcPtr, ctcConnectrecFileSize, tcConnectionrec);
FragrecordPtr regFragptr;
regFragptr.i = regTcPtr.p->fragmentptr;
ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
signal->theData[0] = regFragptr.p->accFragptr[regTcPtr.p->localFragptr];
EXECUTE_DIRECT(DBACC, GSN_READ_ROWCOUNT_REQ, signal, 1);
}
/* ************>> */ /* ************>> */
/* TUPKEYCONF > */ /* TUPKEYCONF > */
/* ************>> */ /* ************>> */
...@@ -7014,6 +7029,14 @@ void Dblqh::continueScanNextReqLab(Signal* signal) ...@@ -7014,6 +7029,14 @@ void Dblqh::continueScanNextReqLab(Signal* signal)
return; return;
}//if }//if
if(scanptr.p->m_last_row){
jam();
scanptr.p->scanCompletedStatus = ZTRUE;
scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
sendScanFragConf(signal, ZFALSE);
return;
}
// Update timer on tcConnectRecord // Update timer on tcConnectRecord
tcConnectptr.p->tcTimer = cLqhTimeOutCount; tcConnectptr.p->tcTimer = cLqhTimeOutCount;
...@@ -7959,13 +7982,10 @@ bool Dblqh::keyinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length) ...@@ -7959,13 +7982,10 @@ bool Dblqh::keyinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length)
* ------------------------------------------------------------------------- */ * ------------------------------------------------------------------------- */
void Dblqh::scanTupkeyConfLab(Signal* signal) void Dblqh::scanTupkeyConfLab(Signal* signal)
{ {
UintR tdata3; const TupKeyConf * conf = (TupKeyConf *)signal->getDataPtr();
UintR tdata4; UintR tdata4 = conf->readLength;
UintR tdata5; UintR tdata5 = conf->lastRow;
tdata3 = signal->theData[2];
tdata4 = signal->theData[3];
tdata5 = signal->theData[4];
tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED; tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
scanptr.i = tcConnectptr.p->tcScanRec; scanptr.i = tcConnectptr.p->tcScanRec;
releaseActiveFrag(signal); releaseActiveFrag(signal);
...@@ -7996,15 +8016,15 @@ void Dblqh::scanTupkeyConfLab(Signal* signal) ...@@ -7996,15 +8016,15 @@ void Dblqh::scanTupkeyConfLab(Signal* signal)
ndbrequire(scanptr.p->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN); ndbrequire(scanptr.p->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN);
scanptr.p->scanOpLength[scanptr.p->scanCompletedOperations] = tdata4; scanptr.p->scanOpLength[scanptr.p->scanCompletedOperations] = tdata4;
scanptr.p->scanCompletedOperations++; scanptr.p->scanCompletedOperations++;
if ((scanptr.p->scanCompletedOperations == scanptr.p->m_last_row = conf->lastRow;
scanptr.p->scanConcurrentOperations) &&
(scanptr.p->scanLockHold == ZTRUE)) { const bool done = (scanptr.p->scanCompletedOperations == scanptr.p->scanConcurrentOperations) | conf->lastRow;
if (done && (scanptr.p->scanLockHold == ZTRUE)) {
jam(); jam();
scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ; scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
sendScanFragConf(signal, ZFALSE); sendScanFragConf(signal, ZFALSE);
return; return;
} else if (scanptr.p->scanCompletedOperations == } else if (done){
scanptr.p->scanConcurrentOperations) {
jam(); jam();
scanptr.p->scanReleaseCounter = scanptr.p->scanCompletedOperations; scanptr.p->scanReleaseCounter = scanptr.p->scanCompletedOperations;
scanReleaseLocksLab(signal); scanReleaseLocksLab(signal);
...@@ -8310,6 +8330,7 @@ Uint32 Dblqh::initScanrec(const ScanFragReq* scanFragReq) ...@@ -8310,6 +8330,7 @@ Uint32 Dblqh::initScanrec(const ScanFragReq* scanFragReq)
scanptr.p->scanLocalFragid = 0; scanptr.p->scanLocalFragid = 0;
scanptr.p->scanTcWaiting = ZTRUE; scanptr.p->scanTcWaiting = ZTRUE;
scanptr.p->scanNumber = ~0; scanptr.p->scanNumber = ~0;
scanptr.p->m_last_row = 0;
for (Uint32 i = 0; i < scanConcurrentOperations; i++) { for (Uint32 i = 0; i < scanConcurrentOperations; i++) {
jam(); jam();
......
...@@ -622,7 +622,10 @@ struct Operationrec { ...@@ -622,7 +622,10 @@ struct Operationrec {
Uint32 tcOpIndex; Uint32 tcOpIndex;
Uint32 gci; Uint32 gci;
Uint32 noFiredTriggers; Uint32 noFiredTriggers;
union {
Uint32 hashValue; // only used in TUP_COMMITREQ Uint32 hashValue; // only used in TUP_COMMITREQ
Uint32 lastRow;
};
Bitmask<MAXNROFATTRIBUTESINWORDS> changeMask; Bitmask<MAXNROFATTRIBUTESINWORDS> changeMask;
}; };
typedef Ptr<Operationrec> OperationrecPtr; typedef Ptr<Operationrec> OperationrecPtr;
...@@ -1623,6 +1626,7 @@ private: ...@@ -1623,6 +1626,7 @@ private:
//------------------------------------------------------------------ //------------------------------------------------------------------
//------------------------------------------------------------------ //------------------------------------------------------------------
bool nullFlagCheck(Uint32 attrDes2); bool nullFlagCheck(Uint32 attrDes2);
bool readRowcount(Uint32 userPtr, Uint32* outBuffer);
//------------------------------------------------------------------ //------------------------------------------------------------------
//------------------------------------------------------------------ //------------------------------------------------------------------
......
...@@ -844,20 +844,18 @@ void Dbtup::sendTUPKEYCONF(Signal* signal, ...@@ -844,20 +844,18 @@ void Dbtup::sendTUPKEYCONF(Signal* signal,
TupKeyConf * const tupKeyConf = (TupKeyConf *)signal->getDataPtrSend(); TupKeyConf * const tupKeyConf = (TupKeyConf *)signal->getDataPtrSend();
Uint32 RuserPointer = regOperPtr->userpointer; Uint32 RuserPointer = regOperPtr->userpointer;
Uint32 RfragPageId = regOperPtr->fragPageId;
Uint32 RpageIndex = regOperPtr->pageIndex;
Uint32 RattroutbufLen = regOperPtr->attroutbufLen; Uint32 RattroutbufLen = regOperPtr->attroutbufLen;
Uint32 RnoFiredTriggers = regOperPtr->noFiredTriggers; Uint32 RnoFiredTriggers = regOperPtr->noFiredTriggers;
BlockReference Ruserblockref = regOperPtr->userblockref; BlockReference Ruserblockref = regOperPtr->userblockref;
Uint32 lastRow = regOperPtr->lastRow;
regOperPtr->transstate = STARTED; regOperPtr->transstate = STARTED;
regOperPtr->tupleState = NO_OTHER_OP; regOperPtr->tupleState = NO_OTHER_OP;
tupKeyConf->userPtr = RuserPointer; tupKeyConf->userPtr = RuserPointer;
tupKeyConf->pageId = RfragPageId;
tupKeyConf->pageIndex = RpageIndex;
tupKeyConf->readLength = RattroutbufLen; tupKeyConf->readLength = RattroutbufLen;
tupKeyConf->writeLength = TlogSize; tupKeyConf->writeLength = TlogSize;
tupKeyConf->noFiredTriggers = RnoFiredTriggers; tupKeyConf->noFiredTriggers = RnoFiredTriggers;
tupKeyConf->lastRow = lastRow;
EXECUTE_DIRECT(refToBlock(Ruserblockref), GSN_TUPKEYCONF, signal, EXECUTE_DIRECT(refToBlock(Ruserblockref), GSN_TUPKEYCONF, signal,
TupKeyConf::SignalLength); TupKeyConf::SignalLength);
...@@ -920,6 +918,7 @@ int Dbtup::handleReadReq(Signal* signal, ...@@ -920,6 +918,7 @@ int Dbtup::handleReadReq(Signal* signal,
return -1; return -1;
} else { } else {
jam(); jam();
regOperPtr->lastRow = 0;
if (interpreterStartLab(signal, pagePtr, Ttupheadoffset) != -1) { if (interpreterStartLab(signal, pagePtr, Ttupheadoffset) != -1) {
return 0; return 0;
}//if }//if
...@@ -1978,13 +1977,20 @@ int Dbtup::interpreterNextLab(Signal* signal, ...@@ -1978,13 +1977,20 @@ int Dbtup::interpreterNextLab(Signal* signal,
} }
case Interpreter::EXIT_OK: case Interpreter::EXIT_OK:
case Interpreter::EXIT_OK_LAST:
jam(); jam();
#ifdef TRACE_INTERPRETER #ifdef TRACE_INTERPRETER
ndbout_c(" - exit_ok"); ndbout_c(" - exit_ok");
#endif #endif
return TdataWritten; return TdataWritten;
case Interpreter::EXIT_OK_LAST:
jam();
#if 1
ndbout_c(" - exit_ok_last");
#endif
operPtr.p->lastRow = 1;
return TdataWritten;
case Interpreter::EXIT_REFUSE: case Interpreter::EXIT_REFUSE:
jam(); jam();
#ifdef TRACE_INTERPRETER #ifdef TRACE_INTERPRETER
......
...@@ -187,6 +187,14 @@ int Dbtup::readAttributes(Page* const pagePtr, ...@@ -187,6 +187,14 @@ int Dbtup::readAttributes(Page* const pagePtr,
} else { } else {
return (Uint32)-1; return (Uint32)-1;
}//if }//if
} else if(attributeId == AttributeHeader::FRAGMENT){
AttributeHeader::init(&outBuffer[tmpAttrBufIndex], attributeId, 1);
outBuffer[tmpAttrBufIndex+1] = fragptr.p->fragmentId;
tOutBufIndex = tmpAttrBufIndex + 2;
} else if(attributeId == AttributeHeader::ROW_COUNT){
AttributeHeader::init(&outBuffer[tmpAttrBufIndex], attributeId, 2);
readRowcount(operPtr.p->userpointer, outBuffer+tmpAttrBufIndex+1);
tOutBufIndex = tmpAttrBufIndex + 3;
} else { } else {
terrorCode = ZATTRIBUTE_ID_ERROR; terrorCode = ZATTRIBUTE_ID_ERROR;
return (Uint32)-1; return (Uint32)-1;
...@@ -195,6 +203,7 @@ int Dbtup::readAttributes(Page* const pagePtr, ...@@ -195,6 +203,7 @@ int Dbtup::readAttributes(Page* const pagePtr,
return tOutBufIndex; return tOutBufIndex;
}//Dbtup::readAttributes() }//Dbtup::readAttributes()
#if 0
int Dbtup::readAttributesWithoutHeader(Page* const pagePtr, int Dbtup::readAttributesWithoutHeader(Page* const pagePtr,
Uint32 tupHeadOffset, Uint32 tupHeadOffset,
Uint32* inBuffer, Uint32* inBuffer,
...@@ -247,6 +256,7 @@ int Dbtup::readAttributesWithoutHeader(Page* const pagePtr, ...@@ -247,6 +256,7 @@ int Dbtup::readAttributesWithoutHeader(Page* const pagePtr,
ndbrequire(attrBufIndex == inBufLen); ndbrequire(attrBufIndex == inBufLen);
return tOutBufIndex; return tOutBufIndex;
}//Dbtup::readAttributes() }//Dbtup::readAttributes()
#endif
bool bool
Dbtup::readFixedSizeTHOneWordNotNULL(Uint32* outBuffer, Dbtup::readFixedSizeTHOneWordNotNULL(Uint32* outBuffer,
...@@ -893,4 +903,13 @@ Dbtup::updateDynSmallVarSize(Uint32* inBuffer, ...@@ -893,4 +903,13 @@ Dbtup::updateDynSmallVarSize(Uint32* inBuffer,
return false; return false;
}//Dbtup::updateDynSmallVarSize() }//Dbtup::updateDynSmallVarSize()
bool
Dbtup::readRowcount(Uint32 userPtr, Uint32* outBuffer){
Uint32 tmp[sizeof(SignalHeader)+25];
Signal * signal = (Signal*)&tmp;
signal->theData[0] = userPtr;
EXECUTE_DIRECT(DBLQH, GSN_READ_ROWCOUNT_REQ, signal, 1);
outBuffer[0] = signal->theData[0];
outBuffer[1] = signal->theData[1];
}
...@@ -325,7 +325,7 @@ NdbOperation::getValue_impl(const NdbColumnImpl* tAttrInfo, char* aValue) ...@@ -325,7 +325,7 @@ NdbOperation::getValue_impl(const NdbColumnImpl* tAttrInfo, char* aValue)
if (theStatus == FinalGetValue) { if (theStatus == FinalGetValue) {
; // Simply continue with getValue ; // Simply continue with getValue
} else if (theStatus == ExecInterpretedValue) { } else if (theStatus == ExecInterpretedValue) {
if (insertATTRINFO(Interpreter::EXIT_OK_LAST) == -1) if (insertATTRINFO(Interpreter::EXIT_OK) == -1)
return NULL; return NULL;
theInterpretedSize = theTotalCurrAI_Len - theInterpretedSize = theTotalCurrAI_Len -
(theInitialReadSize + 5); (theInitialReadSize + 5);
...@@ -415,7 +415,7 @@ NdbOperation::setValue( const NdbColumnImpl* tAttrInfo, ...@@ -415,7 +415,7 @@ NdbOperation::setValue( const NdbColumnImpl* tAttrInfo,
// We insert an exit from interpretation since we are now starting // We insert an exit from interpretation since we are now starting
// to set values in the tuple by setValue. // to set values in the tuple by setValue.
//-------------------------------------------------------------------- //--------------------------------------------------------------------
if (insertATTRINFO(Interpreter::EXIT_OK_LAST) == -1){ if (insertATTRINFO(Interpreter::EXIT_OK) == -1){
return -1; return -1;
} }
theInterpretedSize = theTotalCurrAI_Len - theInterpretedSize = theTotalCurrAI_Len -
......
...@@ -354,7 +354,7 @@ NdbOperation::prepareSendInterpreted() ...@@ -354,7 +354,7 @@ NdbOperation::prepareSendInterpreted()
Uint32 tTotalCurrAI_Len = theTotalCurrAI_Len; Uint32 tTotalCurrAI_Len = theTotalCurrAI_Len;
Uint32 tInitReadSize = theInitialReadSize; Uint32 tInitReadSize = theInitialReadSize;
if (theStatus == ExecInterpretedValue) { if (theStatus == ExecInterpretedValue) {
if (insertATTRINFO(Interpreter::EXIT_OK_LAST) != -1) { if (insertATTRINFO(Interpreter::EXIT_OK) != -1) {
//------------------------------------------------------------------------- //-------------------------------------------------------------------------
// Since we read the total length before inserting the last entry in the // Since we read the total length before inserting the last entry in the
// signals we need to add one to the total length. // signals we need to add one to the total length.
......
...@@ -888,6 +888,18 @@ NdbOperation::interpret_exit_ok() ...@@ -888,6 +888,18 @@ NdbOperation::interpret_exit_ok()
return 0; return 0;
} }
int
NdbOperation::interpret_exit_last_row()
{
INT_DEBUG(("interpret_exit_last_row"));
if (initial_interpreterCheck() == -1)
return -1;
if (insertATTRINFO(Interpreter::EXIT_OK_LAST) == -1)
return -1;
theErrorLine++;
return 0;
}
/************************************************************************************************ /************************************************************************************************
int NdbOperation::interpret_exit_nok(Uint32 ErrorCode) int NdbOperation::interpret_exit_nok(Uint32 ErrorCode)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment