2015-01-27 11:29:30 +01:00
|
|
|
/*
|
|
|
|
* 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;
|
|
|
|
|
2015-02-14 19:55:00 +01:00
|
|
|
static const char* const SCRATCH = "fb_cursor_";
|
|
|
|
static const ULONG PREFETCH_SIZE = 65536; // 64 KB
|
2015-01-27 11:29:30 +01:00
|
|
|
|
|
|
|
DsqlCursor::DsqlCursor(dsql_req* req, ULONG flags)
|
2015-02-17 12:42:50 +01:00
|
|
|
: m_request(req), m_resultSet(NULL), m_flags(flags),
|
2015-01-27 11:29:30 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2015-02-17 12:42:50 +01:00
|
|
|
DsqlCursor::~DsqlCursor()
|
|
|
|
{
|
|
|
|
if (m_resultSet)
|
|
|
|
m_resultSet->resetHandle();
|
|
|
|
}
|
|
|
|
|
2015-01-27 11:29:30 +01:00
|
|
|
jrd_tra* DsqlCursor::getTransaction() const
|
|
|
|
{
|
|
|
|
return m_request->req_transaction;
|
|
|
|
}
|
|
|
|
|
|
|
|
Attachment* DsqlCursor::getAttachment() const
|
|
|
|
{
|
|
|
|
return m_request->req_dbb->dbb_attachment;
|
|
|
|
}
|
|
|
|
|
2015-02-17 12:42:50 +01:00
|
|
|
void DsqlCursor::setInterfacePtr(JResultSet* interfacePtr) throw()
|
|
|
|
{
|
|
|
|
fb_assert(!m_resultSet);
|
|
|
|
m_resultSet = interfacePtr;
|
|
|
|
}
|
|
|
|
|
2015-01-27 11:29:30 +01:00
|
|
|
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);
|
2015-02-18 16:01:17 +01:00
|
|
|
trace.fetch(true, ITracePlugin::RESULT_SUCCESS);
|
2015-01-27 11:29:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|