8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-22 21:23:04 +01:00
firebird-mirror/src/dsql/DsqlCursor.cpp

271 lines
6.1 KiB
C++
Raw Normal View History

/*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.0 (the "License");
* you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Dmitry Yemanov
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2015 Dmitry Yemanov <dimitrf@firebirdsql.org>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#include "firebird.h"
#include "../jrd/tra_proto.h"
#include "../jrd/trace/TraceManager.h"
#include "../jrd/trace/TraceDSQLHelpers.h"
#include "../dsql/DsqlCursor.h"
using namespace Firebird;
using namespace Jrd;
const char* const SCRATCH = "fb_cursor_";
const ULONG PREFETCH_SIZE = 65536; // 64 KB
DsqlCursor::DsqlCursor(dsql_req* req, ULONG flags)
: m_request(req), m_flags(flags),
m_space(req->getPool(), SCRATCH),
m_state(BOS), m_eof(false), m_position(0), m_cachedCount(0),
m_messageSize(req->getStatement()->getReceiveMsg()->msg_length)
{
TRA_link_cursor(m_request->req_transaction, this);
}
jrd_tra* DsqlCursor::getTransaction() const
{
return m_request->req_transaction;
}
Attachment* DsqlCursor::getAttachment() const
{
return m_request->req_dbb->dbb_attachment;
}
void DsqlCursor::close(thread_db* tdbb, DsqlCursor* cursor)
{
if (!cursor)
return;
Jrd::Attachment* const attachment = cursor->getAttachment();
dsql_req* const request = cursor->m_request;
if (request->req_request)
{
ThreadStatusGuard status_vector(tdbb);
try
{
// Report some remaining fetches if any
if (request->req_fetch_baseline)
{
TraceDSQLFetch trace(attachment, request);
trace.fetch(true, ITracePlugin::TRACE_RESULT_SUCCESS);
}
if (request->req_traced && TraceManager::need_dsql_free(attachment))
{
TraceSQLStatementImpl stmt(request, NULL);
TraceManager::event_dsql_free(attachment, &stmt, DSQL_close);
}
JRD_unwind_request(tdbb, request->req_request);
}
catch (Firebird::Exception&)
{} // no-op
}
request->req_cursor = NULL;
TRA_unlink_cursor(request->req_transaction, cursor);
delete cursor;
}
int DsqlCursor::fetchNext(thread_db* tdbb, UCHAR* buffer)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
{
m_eof = !m_request->fetch(tdbb, buffer);
if (m_eof)
{
m_state = EOS;
return 1;
}
m_state = POSITIONED;
return 0;
}
if (m_state == EOS)
return 1;
const FB_UINT64 position = (m_state == BOS) ? 0 : m_position + 1;
return fetchFromCache(tdbb, buffer, position);
}
int DsqlCursor::fetchPrior(thread_db* tdbb, UCHAR* buffer)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
(Arg::Gds(isc_invalid_fetch_option) << Arg::Str("PRIOR")).raise();
if (m_state == BOS)
return -1;
if (!m_position)
{
m_state = BOS;
return -1;
}
const FB_UINT64 position = ((m_state == EOS) ? m_cachedCount : m_position) - 1;
return fetchFromCache(tdbb, buffer, position);
}
int DsqlCursor::fetchFirst(thread_db* tdbb, UCHAR* buffer)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
(Arg::Gds(isc_invalid_fetch_option) << Arg::Str("FIRST")).raise();
return fetchAbsolute(tdbb, buffer, 1);
}
int DsqlCursor::fetchLast(thread_db* tdbb, UCHAR* buffer)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
(Arg::Gds(isc_invalid_fetch_option) << Arg::Str("LAST")).raise();
return fetchAbsolute(tdbb, buffer, -1);
}
int DsqlCursor::fetchAbsolute(thread_db* tdbb, UCHAR* buffer, SLONG position)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
(Arg::Gds(isc_invalid_fetch_option) << Arg::Str("ABSOLUTE")).raise();
if (!position)
{
m_state = BOS;
return -1;
}
SINT64 offset = -1;
if (position < 0)
{
if (!m_eof)
{
cacheInput(tdbb);
fb_assert(m_eof);
}
offset = m_cachedCount;
}
return fetchFromCache(tdbb, buffer, position + offset);
}
int DsqlCursor::fetchRelative(thread_db* tdbb, UCHAR* buffer, SLONG offset)
{
if (!(m_flags & IStatement::CURSOR_TYPE_SCROLLABLE))
(Arg::Gds(isc_invalid_fetch_option) << Arg::Str("RELATIVE")).raise();
if (m_state == BOS)
{
if (offset <= 0)
return -1;
return fetchFromCache(tdbb, buffer, offset - 1);
}
else if (m_state == EOS)
{
if (offset >= 0)
return 1;
fb_assert(m_eof);
if ((SINT64) m_cachedCount + offset < 0)
return -1;
return fetchFromCache(tdbb, buffer, m_cachedCount + offset);
}
if ((SINT64) m_position + offset < 0)
{
m_state = BOS;
return -1;
}
return fetchFromCache(tdbb, buffer, m_position + offset);
}
int DsqlCursor::fetchFromCache(thread_db* tdbb, UCHAR* buffer, FB_UINT64 position)
{
if (position >= m_cachedCount)
{
if (m_eof || !cacheInput(tdbb, position))
{
m_state = EOS;
return 1;
}
}
fb_assert(position < m_cachedCount);
const FB_UINT64 offset = position * m_messageSize;
const FB_UINT64 readBytes = m_space.read(offset, buffer, m_messageSize);
fb_assert(readBytes == m_messageSize);
m_position = position;
m_state = POSITIONED;
return 0;
}
bool DsqlCursor::cacheInput(thread_db* tdbb, FB_UINT64 position)
{
fb_assert(!m_eof);
const ULONG prefetchCount = MAX(PREFETCH_SIZE / m_messageSize, 1);
const ULONG prefetchSize = prefetchCount * m_messageSize;
UCharBuffer messageBuffer;
UCHAR* const buffer = messageBuffer.getBuffer(prefetchSize);
while (position >= m_cachedCount)
{
ULONG count = 0;
for (; count < prefetchCount; count++)
{
UCHAR* const ptr = buffer + count * m_messageSize;
if (!m_request->fetch(tdbb, ptr))
{
m_eof = true;
break;
}
}
if (count)
{
const FB_UINT64 offset = m_cachedCount * m_messageSize;
const ULONG fetchedSize = count * m_messageSize;
const FB_UINT64 writtenBytes = m_space.write(offset, buffer, fetchedSize);
fb_assert(writtenBytes == fetchedSize);
m_cachedCount += count;
}
if (m_eof)
break;
}
return (position < m_cachedCount);
}