mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-24 06:03:02 +01:00
Additional fixes for CORE-3981 (Sub-optimal predicate checking while selecting from a view) that also resolve the regression CORE-3986 (GSEC couldn't change user password after fix for CORE-3981).
This commit is contained in:
parent
18f91be9c6
commit
88bdcebaea
@ -148,7 +148,7 @@ static RecordSelExpr* pass1_rse(thread_db*, CompilerScratch*, RecordSelExpr*);
|
|||||||
static void pass1_source(thread_db*, CompilerScratch*, RecordSelExpr*, jrd_nod*, jrd_nod**, NodeStack&);
|
static void pass1_source(thread_db*, CompilerScratch*, RecordSelExpr*, jrd_nod*, jrd_nod**, NodeStack&);
|
||||||
static bool pass1_store(thread_db*, CompilerScratch*, jrd_nod*);
|
static bool pass1_store(thread_db*, CompilerScratch*, jrd_nod*);
|
||||||
static jrd_nod* pass1_update(thread_db*, CompilerScratch*, jrd_rel*, const trig_vec*, USHORT, USHORT,
|
static jrd_nod* pass1_update(thread_db*, CompilerScratch*, jrd_rel*, const trig_vec*, USHORT, USHORT,
|
||||||
SecurityClass::flags_t, jrd_rel*, USHORT);
|
SecurityClass::flags_t, jrd_rel*, USHORT, USHORT);
|
||||||
static void pass2_rse(thread_db*, CompilerScratch*, RecordSelExpr*);
|
static void pass2_rse(thread_db*, CompilerScratch*, RecordSelExpr*);
|
||||||
static jrd_nod* pass2_union(thread_db*, CompilerScratch*, jrd_nod*);
|
static jrd_nod* pass2_union(thread_db*, CompilerScratch*, jrd_nod*);
|
||||||
static jrd_nod* pass2_validation(thread_db*, CompilerScratch*, const Item&);
|
static jrd_nod* pass2_validation(thread_db*, CompilerScratch*, const Item&);
|
||||||
@ -3941,7 +3941,7 @@ jrd_nod* CMP_pass1(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
{
|
{
|
||||||
const CompilerScratch::csb_repeat* const sub_tail = &csb->csb_rpt[streams[i]];
|
const CompilerScratch::csb_repeat* const sub_tail = &csb->csb_rpt[streams[i]];
|
||||||
|
|
||||||
if (sub_tail->csb_view && sub_tail->csb_view_stream == csb->csb_view_stream)
|
if (sub_tail->csb_view && sub_tail->csb_view_stream == stream)
|
||||||
{
|
{
|
||||||
view_refs = true;
|
view_refs = true;
|
||||||
break;
|
break;
|
||||||
@ -4351,19 +4351,20 @@ static void pass1_erase(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
jrd_rel* parent = NULL;
|
jrd_rel* parent = NULL;
|
||||||
jrd_rel* view = NULL;
|
jrd_rel* view = NULL;
|
||||||
USHORT parent_stream = 0;
|
USHORT parent_stream;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
USHORT new_stream = (USHORT)(IPTR) node->nod_arg[e_erase_stream];
|
USHORT new_stream = (USHORT)(IPTR) node->nod_arg[e_erase_stream];
|
||||||
const USHORT stream = new_stream;
|
const USHORT stream = new_stream;
|
||||||
|
|
||||||
CompilerScratch::csb_repeat* tail = &csb->csb_rpt[stream];
|
CompilerScratch::csb_repeat* const tail = &csb->csb_rpt[stream];
|
||||||
tail->csb_flags |= csb_erase;
|
tail->csb_flags |= csb_erase;
|
||||||
|
|
||||||
jrd_rel* relation = csb->csb_rpt[stream].csb_relation;
|
jrd_rel* const relation = tail->csb_relation;
|
||||||
view = (relation->rel_view_rse) ? relation : view;
|
view = (relation->rel_view_rse) ? relation : view;
|
||||||
if (!parent) {
|
if (!parent) {
|
||||||
parent = tail->csb_view;
|
parent = tail->csb_view;
|
||||||
|
parent_stream = tail->csb_view_stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
post_trigger_access(csb, relation, ExternalAccess::exa_delete, view);
|
post_trigger_access(csb, relation, ExternalAccess::exa_delete, view);
|
||||||
@ -4394,9 +4395,8 @@ static void pass1_erase(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
// get the source relation, either a table or yet another view
|
// get the source relation, either a table or yet another view
|
||||||
|
|
||||||
jrd_nod* source =
|
jrd_nod* source = pass1_update(tdbb, csb, relation, trigger, stream, new_stream,
|
||||||
pass1_update(tdbb, csb, relation, trigger, stream, new_stream,
|
priv, parent, parent_stream, parent_stream);
|
||||||
priv, parent, parent_stream);
|
|
||||||
|
|
||||||
if (!source) {
|
if (!source) {
|
||||||
|
|
||||||
@ -4410,7 +4410,7 @@ static void pass1_erase(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
// remap the source stream
|
// remap the source stream
|
||||||
|
|
||||||
UCHAR* map = csb->csb_rpt[stream].csb_map;
|
UCHAR* map = tail->csb_map;
|
||||||
|
|
||||||
if (trigger) {
|
if (trigger) {
|
||||||
|
|
||||||
@ -4530,7 +4530,7 @@ static void pass1_modify(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
jrd_rel* parent = NULL;
|
jrd_rel* parent = NULL;
|
||||||
jrd_rel* view = NULL;
|
jrd_rel* view = NULL;
|
||||||
USHORT parent_stream = 0;
|
USHORT parent_stream, parent_new_stream;
|
||||||
|
|
||||||
// to support nested views, loop until we hit a table or
|
// to support nested views, loop until we hit a table or
|
||||||
// a view with user-defined triggers (which means no update)
|
// a view with user-defined triggers (which means no update)
|
||||||
@ -4539,13 +4539,17 @@ static void pass1_modify(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
USHORT stream = (USHORT)(IPTR) node->nod_arg[e_mod_org_stream];
|
USHORT stream = (USHORT)(IPTR) node->nod_arg[e_mod_org_stream];
|
||||||
USHORT new_stream = (USHORT)(IPTR) node->nod_arg[e_mod_new_stream];
|
USHORT new_stream = (USHORT)(IPTR) node->nod_arg[e_mod_new_stream];
|
||||||
|
|
||||||
CompilerScratch::csb_repeat* tail = &csb->csb_rpt[new_stream];
|
CompilerScratch::csb_repeat* const tail = &csb->csb_rpt[stream];
|
||||||
tail->csb_flags |= csb_modify;
|
CompilerScratch::csb_repeat* const new_tail = &csb->csb_rpt[new_stream];
|
||||||
|
new_tail->csb_flags |= csb_modify;
|
||||||
|
|
||||||
jrd_rel* relation = csb->csb_rpt[stream].csb_relation;
|
jrd_rel* const relation = tail->csb_relation;
|
||||||
view = (relation->rel_view_rse) ? relation : view;
|
view = (relation->rel_view_rse) ? relation : view;
|
||||||
if (!parent) {
|
if (!parent) {
|
||||||
parent = tail->csb_view;
|
fb_assert(tail->csb_view == new_tail->csb_view);
|
||||||
|
parent = new_tail->csb_view;
|
||||||
|
parent_stream = tail->csb_view_stream;
|
||||||
|
parent_new_stream = new_tail->csb_view_stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
post_trigger_access(csb, relation, ExternalAccess::exa_update, view);
|
post_trigger_access(csb, relation, ExternalAccess::exa_update, view);
|
||||||
@ -4572,8 +4576,8 @@ static void pass1_modify(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
// get the source relation, either a table or yet another view
|
// get the source relation, either a table or yet another view
|
||||||
|
|
||||||
jrd_nod* source = pass1_update(tdbb, csb, relation, trigger, stream,
|
jrd_nod* source = pass1_update(tdbb, csb, relation, trigger, stream, new_stream, priv,
|
||||||
new_stream, priv, parent, parent_stream);
|
parent, parent_stream, parent_new_stream);
|
||||||
|
|
||||||
if (!source) {
|
if (!source) {
|
||||||
|
|
||||||
@ -4594,10 +4598,11 @@ static void pass1_modify(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
parent = relation;
|
parent = relation;
|
||||||
parent_stream = stream;
|
parent_stream = stream;
|
||||||
|
parent_new_stream = new_stream;
|
||||||
|
|
||||||
// remap the source stream
|
// remap the source stream
|
||||||
|
|
||||||
UCHAR* map = csb->csb_rpt[stream].csb_map;
|
UCHAR* map = tail->csb_map;
|
||||||
|
|
||||||
stream = (USHORT)(IPTR) source->nod_arg[e_rel_stream];
|
stream = (USHORT)(IPTR) source->nod_arg[e_rel_stream];
|
||||||
stream = map[stream];
|
stream = map[stream];
|
||||||
@ -4866,6 +4871,16 @@ static void pass1_source(thread_db* tdbb,
|
|||||||
|
|
||||||
stack.push(source);
|
stack.push(source);
|
||||||
|
|
||||||
|
jrd_rel* const parent_view = csb->csb_view;
|
||||||
|
const USHORT view_stream = csb->csb_view_stream;
|
||||||
|
|
||||||
|
const USHORT stream = (USHORT)(IPTR) source->nod_arg[STREAM_INDEX(source)];
|
||||||
|
fb_assert(stream <= MAX_STREAMS);
|
||||||
|
CompilerScratch::csb_repeat* const element = CMP_csb_element(csb, stream);
|
||||||
|
element->csb_view = parent_view;
|
||||||
|
fb_assert(view_stream <= MAX_STREAMS);
|
||||||
|
element->csb_view_stream = (UCHAR) view_stream;
|
||||||
|
|
||||||
// special case: procedure
|
// special case: procedure
|
||||||
|
|
||||||
if (source->nod_type == nod_procedure) {
|
if (source->nod_type == nod_procedure) {
|
||||||
@ -4874,17 +4889,8 @@ static void pass1_source(thread_db* tdbb,
|
|||||||
MET_lookup_procedure_id(tdbb, (SSHORT)(IPTR) source->nod_arg[e_prc_procedure], false, false, 0);
|
MET_lookup_procedure_id(tdbb, (SSHORT)(IPTR) source->nod_arg[e_prc_procedure], false, false, 0);
|
||||||
post_procedure_access(tdbb, csb, procedure);
|
post_procedure_access(tdbb, csb, procedure);
|
||||||
CMP_post_resource(&csb->csb_resources, procedure, Resource::rsc_procedure, procedure->prc_id);
|
CMP_post_resource(&csb->csb_resources, procedure, Resource::rsc_procedure, procedure->prc_id);
|
||||||
|
|
||||||
jrd_rel* const parent_view = csb->csb_view;
|
|
||||||
const USHORT view_stream = csb->csb_view_stream;
|
|
||||||
source->nod_arg[e_prc_view] = (jrd_nod*) parent_view;
|
source->nod_arg[e_prc_view] = (jrd_nod*) parent_view;
|
||||||
|
|
||||||
const USHORT stream = (USHORT)(IPTR) source->nod_arg[e_prc_stream];
|
|
||||||
CompilerScratch::csb_repeat* const element = CMP_csb_element(csb, stream);
|
|
||||||
element->csb_view = parent_view;
|
|
||||||
fb_assert(view_stream <= MAX_STREAMS);
|
|
||||||
element->csb_view_stream = (UCHAR) view_stream;
|
|
||||||
|
|
||||||
if (parent_view)
|
if (parent_view)
|
||||||
{
|
{
|
||||||
ViewContexts& ctx = parent_view->rel_view_contexts;
|
ViewContexts& ctx = parent_view->rel_view_contexts;
|
||||||
@ -4895,6 +4901,7 @@ static void pass1_source(thread_db* tdbb,
|
|||||||
Firebird::string(csb->csb_pool, ctx[pos]->vcx_context_name);
|
Firebird::string(csb->csb_pool, ctx[pos]->vcx_context_name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4908,7 +4915,6 @@ static void pass1_source(thread_db* tdbb,
|
|||||||
// special case: group-by/global aggregates
|
// special case: group-by/global aggregates
|
||||||
|
|
||||||
if (source->nod_type == nod_aggregate) {
|
if (source->nod_type == nod_aggregate) {
|
||||||
fb_assert((int) (IPTR) source->nod_arg[e_agg_stream] <= MAX_STREAMS);
|
|
||||||
CMP_pass1(tdbb, csb, source);
|
CMP_pass1(tdbb, csb, source);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -4917,19 +4923,10 @@ static void pass1_source(thread_db* tdbb,
|
|||||||
// prepare to check protection of relation when a field in the stream of the
|
// prepare to check protection of relation when a field in the stream of the
|
||||||
// relation is accessed.
|
// relation is accessed.
|
||||||
|
|
||||||
jrd_rel* const parent_view = csb->csb_view;
|
|
||||||
const USHORT view_stream = csb->csb_view_stream;
|
|
||||||
|
|
||||||
jrd_rel* view = (jrd_rel*) source->nod_arg[e_rel_relation];
|
jrd_rel* view = (jrd_rel*) source->nod_arg[e_rel_relation];
|
||||||
CMP_post_resource(&csb->csb_resources, view, Resource::rsc_relation, view->rel_id);
|
CMP_post_resource(&csb->csb_resources, view, Resource::rsc_relation, view->rel_id);
|
||||||
source->nod_arg[e_rel_view] = (jrd_nod*) parent_view;
|
source->nod_arg[e_rel_view] = (jrd_nod*) parent_view;
|
||||||
|
|
||||||
const USHORT stream = (USHORT)(IPTR) source->nod_arg[e_rel_stream];
|
|
||||||
CompilerScratch::csb_repeat* const element = CMP_csb_element(csb, stream);
|
|
||||||
element->csb_view = parent_view;
|
|
||||||
fb_assert(view_stream <= MAX_STREAMS);
|
|
||||||
element->csb_view_stream = (UCHAR) view_stream;
|
|
||||||
|
|
||||||
// in the case where there is a parent view, find the context name
|
// in the case where there is a parent view, find the context name
|
||||||
|
|
||||||
if (parent_view) {
|
if (parent_view) {
|
||||||
@ -5065,7 +5062,7 @@ static bool pass1_store(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
jrd_rel* parent = NULL;
|
jrd_rel* parent = NULL;
|
||||||
jrd_rel* view = NULL;
|
jrd_rel* view = NULL;
|
||||||
USHORT parent_stream = 0;
|
USHORT parent_stream;
|
||||||
|
|
||||||
// to support nested views, loop until we hit a table or
|
// to support nested views, loop until we hit a table or
|
||||||
// a view with user-defined triggers (which means no update)
|
// a view with user-defined triggers (which means no update)
|
||||||
@ -5073,13 +5070,14 @@ static bool pass1_store(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
for (;;) {
|
for (;;) {
|
||||||
const USHORT stream = (USHORT)(IPTR) node->nod_arg[e_sto_relation]->nod_arg[e_rel_stream];
|
const USHORT stream = (USHORT)(IPTR) node->nod_arg[e_sto_relation]->nod_arg[e_rel_stream];
|
||||||
|
|
||||||
CompilerScratch::csb_repeat* tail = &csb->csb_rpt[stream];
|
CompilerScratch::csb_repeat* const tail = &csb->csb_rpt[stream];
|
||||||
tail->csb_flags |= csb_store;
|
tail->csb_flags |= csb_store;
|
||||||
|
|
||||||
jrd_rel* relation = csb->csb_rpt[stream].csb_relation;
|
jrd_rel* const relation = tail->csb_relation;
|
||||||
view = (relation->rel_view_rse) ? relation : view;
|
view = (relation->rel_view_rse) ? relation : view;
|
||||||
if (!parent) {
|
if (!parent) {
|
||||||
parent = tail->csb_view;
|
parent = tail->csb_view;
|
||||||
|
parent_stream = tail->csb_view_stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
post_trigger_access(csb, relation, ExternalAccess::exa_insert, view);
|
post_trigger_access(csb, relation, ExternalAccess::exa_insert, view);
|
||||||
@ -5098,8 +5096,8 @@ static bool pass1_store(thread_db* tdbb, CompilerScratch* csb, jrd_nod* node)
|
|||||||
|
|
||||||
// get the source relation, either a table or yet another view
|
// get the source relation, either a table or yet another view
|
||||||
|
|
||||||
jrd_nod* source =
|
jrd_nod* source = pass1_update(tdbb, csb, relation, trigger, stream, stream,
|
||||||
pass1_update(tdbb, csb, relation, trigger, stream, stream, priv, parent, parent_stream);
|
priv, parent, parent_stream, parent_stream);
|
||||||
|
|
||||||
if (!source) {
|
if (!source) {
|
||||||
|
|
||||||
@ -5174,7 +5172,8 @@ static jrd_nod* pass1_update(thread_db* tdbb,
|
|||||||
USHORT update_stream,
|
USHORT update_stream,
|
||||||
SecurityClass::flags_t priv,
|
SecurityClass::flags_t priv,
|
||||||
jrd_rel* view,
|
jrd_rel* view,
|
||||||
USHORT view_stream)
|
USHORT view_stream,
|
||||||
|
USHORT view_update_stream)
|
||||||
{
|
{
|
||||||
/**************************************
|
/**************************************
|
||||||
*
|
*
|
||||||
@ -5206,8 +5205,13 @@ static jrd_nod* pass1_update(thread_db* tdbb,
|
|||||||
fb_assert(view_stream <= MAX_STREAMS);
|
fb_assert(view_stream <= MAX_STREAMS);
|
||||||
CMP_csb_element(csb, stream)->csb_view = view;
|
CMP_csb_element(csb, stream)->csb_view = view;
|
||||||
CMP_csb_element(csb, stream)->csb_view_stream = (UCHAR) view_stream;
|
CMP_csb_element(csb, stream)->csb_view_stream = (UCHAR) view_stream;
|
||||||
CMP_csb_element(csb, update_stream)->csb_view = view;
|
|
||||||
CMP_csb_element(csb, update_stream)->csb_view_stream = (UCHAR) view_stream;
|
if (stream != update_stream)
|
||||||
|
{
|
||||||
|
fb_assert(view_update_stream <= MAX_STREAMS);
|
||||||
|
CMP_csb_element(csb, update_stream)->csb_view = view;
|
||||||
|
CMP_csb_element(csb, update_stream)->csb_view_stream = (UCHAR) view_update_stream;
|
||||||
|
}
|
||||||
|
|
||||||
// if we're not a view, everything's cool
|
// if we're not a view, everything's cool
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user