mirror of
https://github.com/MariaDB/server.git
synced 2025-01-20 05:52:27 +01:00
ndb - bug#27005
Handle API failure during resend API failure could cause release of table object, which will make resend crash when dereferencing table object Solution, use table_id+hash+schemaversion instead of *raw* pointer in resend storage/ndb/src/kernel/blocks/suma/Suma.cpp: Handle API failure during resend API failure could cause release of table object, which will make resend crash when dereferencing table object Solution, use table_id+hash+schemaversion instead of *raw* pointer in resend storage/ndb/test/tools/listen.cpp: add new events
This commit is contained in:
parent
7e4252809c
commit
ee05abd661
2 changed files with 56 additions and 30 deletions
|
@ -756,6 +756,17 @@ Suma::execNODE_FAILREP(Signal* signal){
|
||||||
Restart.resetRestart(signal);
|
Restart.resetRestart(signal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ERROR_INSERTED(13032))
|
||||||
|
{
|
||||||
|
Uint32 node = c_subscriber_nodes.find(0);
|
||||||
|
if (node != NodeBitmask::NotFound)
|
||||||
|
{
|
||||||
|
ndbout_c("Inserting API_FAILREQ node: %u", node);
|
||||||
|
signal->theData[0] = node;
|
||||||
|
EXECUTE_DIRECT(QMGR, GSN_API_FAILREQ, signal, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
signal->theData[0] = SumaContinueB::RESEND_BUCKET;
|
signal->theData[0] = SumaContinueB::RESEND_BUCKET;
|
||||||
|
|
||||||
NdbNodeBitmask tmp;
|
NdbNodeBitmask tmp;
|
||||||
|
@ -3461,6 +3472,9 @@ Suma::execFIRE_TRIG_ORD(Signal* signal)
|
||||||
f_bufferLock = 0;
|
f_bufferLock = 0;
|
||||||
b_bufferLock = 0;
|
b_bufferLock = 0;
|
||||||
|
|
||||||
|
ndbrequire((tabPtr.p = c_tablePool.getPtr(tabPtr.i)) != 0);
|
||||||
|
Uint32 tableId = tabPtr.p->m_tableId;
|
||||||
|
|
||||||
Uint32 bucket= hashValue % c_no_of_buckets;
|
Uint32 bucket= hashValue % c_no_of_buckets;
|
||||||
m_max_seen_gci = (gci > m_max_seen_gci ? gci : m_max_seen_gci);
|
m_max_seen_gci = (gci > m_max_seen_gci ? gci : m_max_seen_gci);
|
||||||
if(m_active_buckets.get(bucket) ||
|
if(m_active_buckets.get(bucket) ||
|
||||||
|
@ -3483,7 +3497,7 @@ Suma::execFIRE_TRIG_ORD(Signal* signal)
|
||||||
|
|
||||||
SubTableData * data = (SubTableData*)signal->getDataPtrSend();//trg;
|
SubTableData * data = (SubTableData*)signal->getDataPtrSend();//trg;
|
||||||
data->gci = gci;
|
data->gci = gci;
|
||||||
data->tableId = tabPtr.p->m_tableId;
|
data->tableId = tableId;
|
||||||
data->requestInfo = 0;
|
data->requestInfo = 0;
|
||||||
SubTableData::setOperation(data->requestInfo, event);
|
SubTableData::setOperation(data->requestInfo, event);
|
||||||
data->logType = 0;
|
data->logType = 0;
|
||||||
|
@ -3506,10 +3520,11 @@ Suma::execFIRE_TRIG_ORD(Signal* signal)
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Uint32* dst;
|
Uint32* dst;
|
||||||
Uint32 sz = f_trigBufferSize + b_trigBufferSize + 2;
|
Uint32 sz = f_trigBufferSize + b_trigBufferSize + 3;
|
||||||
if((dst = get_buffer_ptr(signal, bucket, gci, sz)))
|
if((dst = get_buffer_ptr(signal, bucket, gci, sz)))
|
||||||
{
|
{
|
||||||
* dst++ = tabPtr.i;
|
* dst++ = tableId;
|
||||||
|
* dst++ = tabPtr.p->m_schemaVersion;
|
||||||
* dst++ = (event << 16) | f_trigBufferSize;
|
* dst++ = (event << 16) | f_trigBufferSize;
|
||||||
memcpy(dst, f_buffer, f_trigBufferSize << 2);
|
memcpy(dst, f_buffer, f_trigBufferSize << 2);
|
||||||
dst += f_trigBufferSize;
|
dst += f_trigBufferSize;
|
||||||
|
@ -3641,7 +3656,7 @@ Suma::execSUB_GCP_COMPLETE_REP(Signal* signal)
|
||||||
|
|
||||||
if(c_buckets[i].m_buffer_tail != RNIL)
|
if(c_buckets[i].m_buffer_tail != RNIL)
|
||||||
{
|
{
|
||||||
Uint32* dst;
|
//Uint32* dst;
|
||||||
get_buffer_ptr(signal, i, gci, 0);
|
get_buffer_ptr(signal, i, gci, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3986,8 +4001,8 @@ void
|
||||||
Suma::completeSubRemove(SubscriptionPtr subPtr)
|
Suma::completeSubRemove(SubscriptionPtr subPtr)
|
||||||
{
|
{
|
||||||
DBUG_ENTER("Suma::completeSubRemove");
|
DBUG_ENTER("Suma::completeSubRemove");
|
||||||
Uint32 subscriptionId = subPtr.p->m_subscriptionId;
|
//Uint32 subscriptionId = subPtr.p->m_subscriptionId;
|
||||||
Uint32 subscriptionKey = subPtr.p->m_subscriptionKey;
|
//Uint32 subscriptionKey = subPtr.p->m_subscriptionKey;
|
||||||
|
|
||||||
c_subscriptions.release(subPtr);
|
c_subscriptions.release(subPtr);
|
||||||
DBUG_PRINT("info",("c_subscriptionPool size: %d free: %d",
|
DBUG_PRINT("info",("c_subscriptionPool size: %d free: %d",
|
||||||
|
@ -5003,27 +5018,30 @@ Suma::resend_bucket(Signal* signal, Uint32 buck, Uint32 min_gci,
|
||||||
{
|
{
|
||||||
g_cnt++;
|
g_cnt++;
|
||||||
Uint32 table = * src++ ;
|
Uint32 table = * src++ ;
|
||||||
|
Uint32 schemaVersion = * src++;
|
||||||
Uint32 event = * src >> 16;
|
Uint32 event = * src >> 16;
|
||||||
Uint32 sz_1 = (* src ++) & 0xFFFF;
|
Uint32 sz_1 = (* src ++) & 0xFFFF;
|
||||||
|
|
||||||
ndbassert(sz - 2 >= sz_1);
|
ndbassert(sz - 3 >= sz_1);
|
||||||
|
|
||||||
LinearSectionPtr ptr[3];
|
LinearSectionPtr ptr[3];
|
||||||
const Uint32 nptr= reformat(signal, ptr,
|
const Uint32 nptr= reformat(signal, ptr,
|
||||||
src, sz_1,
|
src, sz_1,
|
||||||
src + sz_1, sz - 2 - sz_1);
|
src + sz_1, sz - 3 - sz_1);
|
||||||
Uint32 ptrLen= 0;
|
Uint32 ptrLen= 0;
|
||||||
for(Uint32 i =0; i < nptr; i++)
|
for(Uint32 i =0; i < nptr; i++)
|
||||||
ptrLen+= ptr[i].sz;
|
ptrLen+= ptr[i].sz;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Signal to subscriber(s)
|
* Signal to subscriber(s)
|
||||||
*/
|
*/
|
||||||
Ptr<Table> tabPtr;
|
Ptr<Table> tabPtr;
|
||||||
ndbrequire((tabPtr.p = c_tablePool.getPtr(table)) != 0);
|
if (c_tables.find(tabPtr, table) &&
|
||||||
|
tabPtr.p->m_schemaVersion == schemaVersion)
|
||||||
|
{
|
||||||
SubTableData * data = (SubTableData*)signal->getDataPtrSend();//trg;
|
SubTableData * data = (SubTableData*)signal->getDataPtrSend();//trg;
|
||||||
data->gci = last_gci;
|
data->gci = last_gci;
|
||||||
data->tableId = tabPtr.p->m_tableId;
|
data->tableId = table;
|
||||||
data->requestInfo = 0;
|
data->requestInfo = 0;
|
||||||
SubTableData::setOperation(data->requestInfo, event);
|
SubTableData::setOperation(data->requestInfo, event);
|
||||||
data->logType = 0;
|
data->logType = 0;
|
||||||
|
@ -5031,7 +5049,8 @@ Suma::resend_bucket(Signal* signal, Uint32 buck, Uint32 min_gci,
|
||||||
data->totalLen = ptrLen;
|
data->totalLen = ptrLen;
|
||||||
|
|
||||||
{
|
{
|
||||||
LocalDLList<Subscriber> list(c_subscriberPool,tabPtr.p->c_subscribers);
|
LocalDLList<Subscriber>
|
||||||
|
list(c_subscriberPool,tabPtr.p->c_subscribers);
|
||||||
SubscriberPtr subbPtr;
|
SubscriberPtr subbPtr;
|
||||||
for(list.first(subbPtr); !subbPtr.isNull(); list.next(subbPtr))
|
for(list.first(subbPtr); !subbPtr.isNull(); list.next(subbPtr))
|
||||||
{
|
{
|
||||||
|
@ -5043,6 +5062,7 @@ Suma::resend_bucket(Signal* signal, Uint32 buck, Uint32 min_gci,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -168,9 +168,15 @@ main(int argc, const char** argv){
|
||||||
break;
|
break;
|
||||||
case NdbDictionary::Event::TE_DROP:
|
case NdbDictionary::Event::TE_DROP:
|
||||||
break;
|
break;
|
||||||
|
case NdbDictionary::Event::TE_NODE_FAILURE:
|
||||||
|
break;
|
||||||
|
case NdbDictionary::Event::TE_SUBSCRIBE:
|
||||||
|
case NdbDictionary::Event::TE_UNSUBSCRIBE:
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
/* We should REALLY never get here. */
|
/* We should REALLY never get here. */
|
||||||
ndbout_c("Error: unknown event type");
|
ndbout_c("Error: unknown event type: %u",
|
||||||
|
(Uint32)pOp->getEventType());
|
||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
} while ((pOp= MyNdb.nextEvent()) && gci == pOp->getGCI());
|
} while ((pOp= MyNdb.nextEvent()) && gci == pOp->getGCI());
|
||||||
|
|
Loading…
Reference in a new issue