mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-22 20:03:02 +01:00
Fixed #6900: IBatch add method fails silently when a memory limit is exceeded
This commit is contained in:
parent
864c9f08ad
commit
3ff54b7d77
File diff suppressed because it is too large
Load Diff
@ -324,6 +324,8 @@ int ResultSet::release()
|
||||
class Batch FB_FINAL : public RefCntIface<IBatchImpl<Batch, CheckStatusWrapper> >
|
||||
{
|
||||
public:
|
||||
static const ULONG DEFER_BATCH_LIMIT = 64;
|
||||
|
||||
Batch(Statement* s, IMessageMetadata* inFmt, unsigned parLength, const unsigned char* par);
|
||||
|
||||
// IResultSet implementation
|
||||
@ -346,6 +348,14 @@ private:
|
||||
void releaseStatement();
|
||||
void setBlobAlignment();
|
||||
|
||||
void cleanup()
|
||||
{
|
||||
if (blobPolicy != BLOB_NONE)
|
||||
blobStream = blobStreamBuffer;
|
||||
sizePointer = nullptr;
|
||||
messageStream = 0;
|
||||
}
|
||||
|
||||
void genBlobId(ISC_QUAD* blobId)
|
||||
{
|
||||
if (++genId.gds_quad_low == 0)
|
||||
@ -372,7 +382,7 @@ private:
|
||||
if (step == messageBufferSize)
|
||||
{
|
||||
// direct packet sent
|
||||
sendMessagePacket(step, ptr);
|
||||
sendMessagePacket(step, ptr, false);
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -381,7 +391,7 @@ private:
|
||||
messageStream += step;
|
||||
if (messageStream == messageBufferSize)
|
||||
{
|
||||
sendMessagePacket(messageBufferSize, messageStreamBuffer);
|
||||
sendMessagePacket(messageBufferSize, messageStreamBuffer, false);
|
||||
messageStream = 0;
|
||||
}
|
||||
}
|
||||
@ -401,7 +411,7 @@ private:
|
||||
ULONG space = blobBufferSize - (blobStream - blobStreamBuffer);
|
||||
if (space < Rsr::BatchStream::SIZEOF_BLOB_HEAD)
|
||||
{
|
||||
sendBlobPacket(blobStream - blobStreamBuffer, blobStreamBuffer);
|
||||
sendBlobPacket(blobStream - blobStreamBuffer, blobStreamBuffer, false);
|
||||
blobStream = blobStreamBuffer;
|
||||
}
|
||||
}
|
||||
@ -431,7 +441,7 @@ private:
|
||||
if (step == blobBufferSize)
|
||||
{
|
||||
// direct packet sent
|
||||
sendBlobPacket(blobBufferSize, ptr);
|
||||
sendBlobPacket(blobBufferSize, ptr, false);
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -440,7 +450,7 @@ private:
|
||||
blobStream += step;
|
||||
if (blobStream - blobStreamBuffer == blobBufferSize)
|
||||
{
|
||||
sendBlobPacket(blobBufferSize, blobStreamBuffer);
|
||||
sendBlobPacket(blobBufferSize, blobStreamBuffer, false);
|
||||
blobStream = blobStreamBuffer;
|
||||
sizePointer = NULL;
|
||||
}
|
||||
@ -497,22 +507,23 @@ private:
|
||||
ULONG size = blobStream - blobStreamBuffer;
|
||||
if (size)
|
||||
{
|
||||
sendBlobPacket(size, blobStreamBuffer);
|
||||
sendBlobPacket(size, blobStreamBuffer, messageStream == 0);
|
||||
blobStream = blobStreamBuffer;
|
||||
}
|
||||
}
|
||||
|
||||
if (messageStream)
|
||||
{
|
||||
sendMessagePacket(messageStream, messageStreamBuffer);
|
||||
sendMessagePacket(messageStream, messageStreamBuffer, true);
|
||||
messageStream = 0;
|
||||
}
|
||||
|
||||
batchActive = false;
|
||||
}
|
||||
|
||||
void sendBlobPacket(unsigned size, const UCHAR* ptr);
|
||||
void sendMessagePacket(unsigned size, const UCHAR* ptr);
|
||||
void sendBlobPacket(unsigned size, const UCHAR* ptr, bool flash);
|
||||
void sendMessagePacket(unsigned size, const UCHAR* ptr, bool flash);
|
||||
void sendDeferredPacket(rem_port* port, PACKET* packet, bool flash);
|
||||
|
||||
Firebird::AutoPtr<UCHAR, Firebird::ArrayDelete> messageStreamBuffer, blobStreamBuffer;
|
||||
ULONG messageStream;
|
||||
@ -2385,7 +2396,7 @@ void Batch::add(CheckStatusWrapper* status, unsigned count, const void* inBuffer
|
||||
}
|
||||
|
||||
|
||||
void Batch::sendMessagePacket(unsigned count, const UCHAR* ptr)
|
||||
void Batch::sendMessagePacket(unsigned count, const UCHAR* ptr, bool flash)
|
||||
{
|
||||
Rsr* statement = stmt->getStatement();
|
||||
CHECK_HANDLE(statement, isc_bad_req_handle);
|
||||
@ -2401,8 +2412,7 @@ void Batch::sendMessagePacket(unsigned count, const UCHAR* ptr)
|
||||
batch->p_batch_data.cstr_address = const_cast<UCHAR*>(ptr);
|
||||
statement->rsr_batch_size = alignedSize;
|
||||
|
||||
send_partial_packet(port, packet);
|
||||
defer_packet(port, packet, true);
|
||||
sendDeferredPacket(port, packet, flash);
|
||||
}
|
||||
|
||||
|
||||
@ -2529,7 +2539,7 @@ void Batch::addBlobStream(CheckStatusWrapper* status, unsigned length, const voi
|
||||
}
|
||||
|
||||
|
||||
void Batch::sendBlobPacket(unsigned size, const UCHAR* ptr)
|
||||
void Batch::sendBlobPacket(unsigned size, const UCHAR* ptr, bool flash)
|
||||
{
|
||||
Rsr* statement = stmt->getStatement();
|
||||
Rdb* rdb = statement->rsr_rdb;
|
||||
@ -2545,8 +2555,31 @@ void Batch::sendBlobPacket(unsigned size, const UCHAR* ptr)
|
||||
batch->p_batch_blob_data.cstr_address = const_cast<UCHAR*>(ptr);
|
||||
batch->p_batch_blob_data.cstr_length = size;
|
||||
|
||||
sendDeferredPacket(port, packet, flash);
|
||||
}
|
||||
|
||||
|
||||
void Batch::sendDeferredPacket(rem_port* port, PACKET* packet, bool flash)
|
||||
{
|
||||
send_partial_packet(port, packet);
|
||||
defer_packet(port, packet, true);
|
||||
|
||||
if ((port->port_protocol >= PROTOCOL_VERSION17) &&
|
||||
((port->port_deferred_packets->getCount() >= DEFER_BATCH_LIMIT) || flash))
|
||||
{
|
||||
packet->p_operation = op_batch_sync;
|
||||
send_packet(port, packet);
|
||||
receive_packet(port, packet);
|
||||
|
||||
LocalStatus warning;
|
||||
port->checkResponse(&warning, packet, false);
|
||||
Rsr* statement = stmt->getStatement();
|
||||
if (statement->haveException())
|
||||
{
|
||||
cleanup();
|
||||
statement->raiseException();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -2735,7 +2768,12 @@ IBatchCompletionState* Batch::execute(CheckStatusWrapper* status, ITransaction*
|
||||
statement->rsr_batch_cs = nullptr;
|
||||
|
||||
if (packet->p_operation == op_batch_cs)
|
||||
{
|
||||
// when working with 4.0.0 server we could not raise it in advance...
|
||||
statement->clearException();
|
||||
|
||||
return cs.release();
|
||||
}
|
||||
|
||||
REMOTE_check_response(status, rdb, packet);
|
||||
}
|
||||
@ -2765,10 +2803,7 @@ void Batch::cancel(CheckStatusWrapper* status)
|
||||
RefMutexGuard portGuard(*port->port_sync, FB_FUNCTION);
|
||||
|
||||
// Cleanup local data
|
||||
if (blobPolicy != BLOB_NONE)
|
||||
blobStream = blobStreamBuffer;
|
||||
sizePointer = nullptr;
|
||||
messageStream = 0;
|
||||
cleanup();
|
||||
batchActive = false;
|
||||
|
||||
// Prepare packet
|
||||
@ -7951,8 +7986,6 @@ static void receive_packet_noqueue(rem_port* port, PACKET* packet)
|
||||
|
||||
// Receive responses for all deferred packets that were already sent
|
||||
|
||||
Rdb* rdb = port->port_context;
|
||||
|
||||
if (port->port_deferred_packets)
|
||||
{
|
||||
while (port->port_deferred_packets->getCount())
|
||||
@ -7962,17 +7995,25 @@ static void receive_packet_noqueue(rem_port* port, PACKET* packet)
|
||||
break;
|
||||
|
||||
OBJCT stmt_id = 0;
|
||||
bool bCheckResponse = false, bFreeStmt = false;
|
||||
bool bCheckResponse = false, bFreeStmt = false, bAssign = false;
|
||||
|
||||
if (p->packet.p_operation == op_execute)
|
||||
switch (p->packet.p_operation)
|
||||
{
|
||||
case op_execute:
|
||||
stmt_id = p->packet.p_sqldata.p_sqldata_statement;
|
||||
bCheckResponse = true;
|
||||
}
|
||||
else if (p->packet.p_operation == op_free_statement)
|
||||
{
|
||||
bAssign = true;
|
||||
break;
|
||||
|
||||
case op_batch_msg:
|
||||
stmt_id = p->packet.p_batch_msg.p_batch_statement;
|
||||
bCheckResponse = true;
|
||||
break;
|
||||
|
||||
case op_free_statement:
|
||||
stmt_id = p->packet.p_sqlfree.p_sqlfree_statement;
|
||||
bFreeStmt = (p->packet.p_sqlfree.p_sqlfree_option == DSQL_drop);
|
||||
break;
|
||||
}
|
||||
|
||||
receive_packet_with_callback(port, &p->packet);
|
||||
@ -7983,9 +8024,9 @@ static void receive_packet_noqueue(rem_port* port, PACKET* packet)
|
||||
|
||||
if (bCheckResponse)
|
||||
{
|
||||
bool bAssign = true;
|
||||
try
|
||||
{
|
||||
Rdb* rdb = port->port_context;
|
||||
LocalStatus ls;
|
||||
CheckStatusWrapper status(&ls);
|
||||
REMOTE_check_response(&status, rdb, &p->packet);
|
||||
|
@ -715,7 +715,8 @@ rem_port* INET_analyze(ClntAuthBlock* cBlock,
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION13, ptype_lazy_send, 4),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION14, ptype_lazy_send, 5),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION15, ptype_lazy_send, 6),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_lazy_send, 7)
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_lazy_send, 7),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION17, ptype_lazy_send, 8)
|
||||
};
|
||||
fb_assert(FB_NELEM(protocols_to_try) <= FB_NELEM(cnct->p_cnct_versions));
|
||||
cnct->p_cnct_count = FB_NELEM(protocols_to_try);
|
||||
|
@ -173,7 +173,8 @@ rem_port* WNET_analyze(ClntAuthBlock* cBlock,
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION13, ptype_batch_send, 4),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION14, ptype_batch_send, 5),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION15, ptype_batch_send, 6),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_batch_send, 7)
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_batch_send, 7),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION17, ptype_batch_send, 8)
|
||||
};
|
||||
fb_assert(FB_NELEM(protocols_to_try) <= FB_NELEM(cnct->p_cnct_versions));
|
||||
cnct->p_cnct_count = FB_NELEM(protocols_to_try);
|
||||
|
@ -305,7 +305,8 @@ rem_port* XNET_analyze(ClntAuthBlock* cBlock,
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION13, ptype_batch_send, 4),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION14, ptype_batch_send, 5),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION15, ptype_batch_send, 6),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_batch_send, 7)
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION16, ptype_batch_send, 7),
|
||||
REMOTE_PROTOCOL(PROTOCOL_VERSION17, ptype_batch_send, 8)
|
||||
};
|
||||
fb_assert(FB_NELEM(protocols_to_try) <= FB_NELEM(cnct->p_cnct_versions));
|
||||
cnct->p_cnct_count = FB_NELEM(protocols_to_try);
|
||||
|
@ -1068,6 +1068,11 @@ bool_t xdr_protocol(RemoteXdr* xdrs, PACKET* p)
|
||||
return P_TRUE(xdrs, p);
|
||||
}
|
||||
|
||||
case op_batch_sync:
|
||||
{
|
||||
return P_TRUE(xdrs, p);
|
||||
}
|
||||
|
||||
case op_batch_set_bpb:
|
||||
{
|
||||
P_BATCH_SETBPB* b = &p->p_batch_setbpb;
|
||||
|
@ -93,6 +93,11 @@ const USHORT PROTOCOL_VERSION15 = (FB_PROTOCOL_FLAG | 15);
|
||||
const USHORT PROTOCOL_VERSION16 = (FB_PROTOCOL_FLAG | 16);
|
||||
const USHORT PROTOCOL_STMT_TOUT = PROTOCOL_VERSION16;
|
||||
|
||||
// Protocol 17:
|
||||
// - supports op_batch_sync
|
||||
|
||||
const USHORT PROTOCOL_VERSION17 = (FB_PROTOCOL_FLAG | 17);
|
||||
|
||||
// Architecture types
|
||||
|
||||
enum P_ARCH
|
||||
@ -289,6 +294,7 @@ enum P_OP
|
||||
op_repl_req = 108,
|
||||
|
||||
op_batch_cancel = 109,
|
||||
op_batch_sync = 110,
|
||||
|
||||
op_max
|
||||
};
|
||||
|
@ -1368,6 +1368,7 @@ public:
|
||||
void batch_exec(P_BATCH_EXEC*, PACKET*);
|
||||
void batch_rls(P_BATCH_FREE_CANCEL*, PACKET*);
|
||||
void batch_cancel(P_BATCH_FREE_CANCEL*, PACKET*);
|
||||
void batch_sync(PACKET*);
|
||||
void batch_bpb(P_BATCH_SETBPB*, PACKET*);
|
||||
void replicate(P_REPLICATE*, PACKET*);
|
||||
|
||||
|
@ -1918,7 +1918,7 @@ static bool accept_connection(rem_port* port, P_CNCT* connect, PACKET* send)
|
||||
{
|
||||
if ((protocol->p_cnct_version == PROTOCOL_VERSION10 ||
|
||||
(protocol->p_cnct_version >= PROTOCOL_VERSION11 &&
|
||||
protocol->p_cnct_version <= PROTOCOL_VERSION16)) &&
|
||||
protocol->p_cnct_version <= PROTOCOL_VERSION17)) &&
|
||||
(protocol->p_cnct_architecture == arch_generic ||
|
||||
protocol->p_cnct_architecture == ARCHITECTURE) &&
|
||||
protocol->p_cnct_weight >= weight)
|
||||
@ -3717,6 +3717,16 @@ void rem_port::batch_cancel(P_BATCH_FREE_CANCEL* batch, PACKET* sendL)
|
||||
}
|
||||
|
||||
|
||||
void rem_port::batch_sync(PACKET* sendL)
|
||||
{
|
||||
LocalStatus ls;
|
||||
CheckStatusWrapper status_vector(&ls);
|
||||
|
||||
// no need checking protocol version if client is using batch_sync, just return synced response
|
||||
this->send_response(sendL, 0, 0, &status_vector, false);
|
||||
}
|
||||
|
||||
|
||||
void rem_port::replicate(P_REPLICATE* repl, PACKET* sendL)
|
||||
{
|
||||
LocalStatus ls;
|
||||
@ -4992,6 +5002,10 @@ static bool process_packet(rem_port* port, PACKET* sendL, PACKET* receive, rem_p
|
||||
port->batch_cancel(&receive->p_batch_free_cancel, sendL);
|
||||
break;
|
||||
|
||||
case op_batch_sync:
|
||||
port->batch_sync(sendL);
|
||||
break;
|
||||
|
||||
case op_batch_blob_stream:
|
||||
port->batch_blob_stream(&receive->p_batch_blob, sendL);
|
||||
break;
|
||||
|
Loading…
Reference in New Issue
Block a user