/* * The contents of this file are subject to the Interbase 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.Inprise.com/IPL.html * * Software distributed under the License is distributed on an * "AS IS" basis, 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 Inprise Corporation * and its predecessors. Portions created by Inprise Corporation are * Copyright (C) Inprise Corporation. * * All Rights Reserved. * Contributor(s): ______________________________________. */ #include "firebird.h" #include "../jrd/jrd.h" #include "../jrd/req.h" #include "../jrd/cmp_proto.h" #include "../jrd/evl_proto.h" #include "../jrd/met_proto.h" #include "../jrd/vio_proto.h" #include "RecordSource.h" using namespace Firebird; using namespace Jrd; // ------------------------------ // Data access: nested loops join // ------------------------------ NestedLoopJoin::NestedLoopJoin(CompilerScratch* csb, size_t count, RecordSource* const* args) : m_outerJoin(false), m_semiJoin(false), m_antiJoin(false), m_args(csb->csb_pool), m_boolean(NULL) { m_impure = CMP_impure(csb, sizeof(Impure)); m_args.resize(count); for (size_t i = 0; i < count; i++) { m_args[i] = args[i]; } } NestedLoopJoin::NestedLoopJoin(CompilerScratch* csb, RecordSource* outer, RecordSource* inner, BoolExprNode* boolean, bool semiJoin, bool antiJoin) : m_outerJoin(true), m_semiJoin(semiJoin), m_antiJoin(antiJoin), m_args(csb->csb_pool), m_boolean(boolean) { fb_assert(outer && inner); m_impure = CMP_impure(csb, sizeof(Impure)); m_args.add(outer); m_args.add(inner); } void NestedLoopJoin::open(thread_db* tdbb) const { jrd_req* const request = tdbb->getRequest(); Impure* const impure = request->getImpure(m_impure); impure->irsb_flags = irsb_open | irsb_first | irsb_mustread; } void NestedLoopJoin::close(thread_db* tdbb) const { jrd_req* const request = tdbb->getRequest(); invalidateRecords(request); Impure* const impure = request->getImpure(m_impure); if (impure->irsb_flags & irsb_open) { impure->irsb_flags &= ~irsb_open; for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->close(tdbb); } } } bool NestedLoopJoin::getRecord(thread_db* tdbb) const { jrd_req* const request = tdbb->getRequest(); Impure* const impure = request->getImpure(m_impure); if (!(impure->irsb_flags & irsb_open)) { return false; } if (m_outerJoin) { fb_assert(m_args.getCount() == 2); const RecordSource* const outer = m_args[0]; const RecordSource* const inner = m_args[1]; if (impure->irsb_flags & irsb_first) { outer->open(tdbb); impure->irsb_flags &= ~irsb_first; } while (true) { if (impure->irsb_flags & irsb_mustread) { if (!outer->getRecord(tdbb)) { return false; } if (m_boolean && !m_boolean->execute(tdbb, request)) { // The boolean pertaining to the left sub-stream is false // so just join sub-stream to a null valued right sub-stream inner->nullRecords(tdbb); return true; } impure->irsb_flags &= ~(irsb_mustread | irsb_joined); inner->open(tdbb); } if (m_semiJoin) { if (inner->getRecord(tdbb)) { impure->irsb_flags &= ~irsb_joined; } else { impure->irsb_flags |= irsb_joined; } } else if (m_antiJoin) { if (inner->getRecord(tdbb)) { impure->irsb_flags |= irsb_joined; } else { impure->irsb_flags &= ~irsb_joined; } } else { if (inner->getRecord(tdbb)) { impure->irsb_flags |= irsb_joined; return true; } } inner->close(tdbb); impure->irsb_flags |= irsb_mustread; if (!(impure->irsb_flags & irsb_joined)) { // The current left sub-stream record has not been joined to anything. // Join it to a null valued right sub-stream. inner->nullRecords(tdbb); return true; } } } else { if (impure->irsb_flags & irsb_first) { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->open(tdbb); if (!fetchRecord(tdbb, i)) { return false; } } impure->irsb_flags &= ~irsb_first; } // hvlad: self referenced members are removed from recursive SELECT's // in recursive CTE (it is done in dsql\pass1.cpp). If there are no other // members in such SELECT then rsb_count will be zero. Handle it. else if (m_args.isEmpty()) { return false; } else if (!fetchRecord(tdbb, m_args.getCount() - 1)) { return false; } } return true; } bool NestedLoopJoin::refetchRecord(thread_db* /*tdbb*/) const { return true; } bool NestedLoopJoin::lockRecord(thread_db* /*tdbb*/) const { status_exception::raise(Arg::Gds(isc_record_lock_not_supp)); return false; // compiler silencer } void NestedLoopJoin::print(thread_db* tdbb, string& plan, bool detailed, unsigned level) const { if (detailed) { plan += printIndent(++level) + " Nested Loop Join "; plan += m_semiJoin ? "(semi)" : m_antiJoin ? "(anti)" : m_outerJoin ? "(outer)" : "(inner)"; for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->print(tdbb, plan, true, level); } } else { level++; plan += "JOIN ("; for (size_t i = 0; i < m_args.getCount(); i++) { if (i) { plan += ", "; } m_args[i]->print(tdbb, plan, false, level); } plan += ")"; } } void NestedLoopJoin::markRecursive() { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->markRecursive(); } } void NestedLoopJoin::findUsedStreams(StreamList& streams, bool expandAll) const { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->findUsedStreams(streams, expandAll); } } void NestedLoopJoin::invalidateRecords(jrd_req* request) const { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->invalidateRecords(request); } } void NestedLoopJoin::nullRecords(thread_db* tdbb) const { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->nullRecords(tdbb); } } void NestedLoopJoin::saveRecords(thread_db* tdbb) const { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->saveRecords(tdbb); } } void NestedLoopJoin::restoreRecords(thread_db* tdbb) const { for (size_t i = 0; i < m_args.getCount(); i++) { m_args[i]->restoreRecords(tdbb); } } bool NestedLoopJoin::fetchRecord(thread_db* tdbb, size_t n) const { const RecordSource* const arg = m_args[n]; if (arg->getRecord(tdbb)) { return true; } // We have exhausted this stream, so close it; if there is // another candidate record from the n-1 streams to the left, // then reopen the stream and start again from the beginning. while (true) { arg->close(tdbb); if (n == 0 || !fetchRecord(tdbb, n - 1)) { return false; } arg->open(tdbb); if (arg->getRecord(tdbb)) { return true; } } }