8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-23 10:43:03 +01:00

Attempted to fix CORE-2440, CORE-5118 and CORE-5900 together (expression indices contain NULL keys after restore).

This commit is contained in:
Dmitry Yemanov 2019-01-17 14:06:54 +03:00
parent d8ab4a69af
commit fea7c61d97
5 changed files with 52 additions and 23 deletions

View File

@ -344,6 +344,20 @@ int RESTORE_restore (const TEXT* file_name, const TEXT* database_name)
} }
END_ERROR; END_ERROR;
if (tdgbl->global_trans)
{
BURP_verbose (68);
// msg 68 committing meta data
EXEC SQL COMMIT TRANSACTION tdgbl->global_trans;
if (gds_status->hasData())
general_on_error ();
// Check to see if there is a warning
if (gds_status->getState() & Firebird::IStatus::STATE_WARNINGS)
{
BURP_print_warning(gds_status);
}
}
// Activate the indices for foreign keys and do another commit // Activate the indices for foreign keys and do another commit
if (!tdgbl->gbl_sw_deactivate_indexes) if (!tdgbl->gbl_sw_deactivate_indexes)
{ {
@ -467,20 +481,6 @@ int RESTORE_restore (const TEXT* file_name, const TEXT* database_name)
END_ERROR; END_ERROR;
} }
if (tdgbl->global_trans)
{
BURP_verbose (68);
// msg 68 committing meta data
EXEC SQL COMMIT TRANSACTION tdgbl->global_trans;
if (gds_status->hasData())
general_on_error ();
// Check to see if there is a warning
if (gds_status->getState() & Firebird::IStatus::STATE_WARNINGS)
{
BURP_print_warning(gds_status);
}
}
EXEC SQL SET TRANSACTION ISOLATION LEVEL READ COMMITTED NO_AUTO_UNDO; EXEC SQL SET TRANSACTION ISOLATION LEVEL READ COMMITTED NO_AUTO_UNDO;
if (gds_status->hasData()) if (gds_status->hasData())
EXEC SQL SET TRANSACTION; EXEC SQL SET TRANSACTION;
@ -7037,9 +7037,10 @@ bool get_index(BurpGlobals* tdgbl, const burp_rel* relation)
* *
**************************************/ **************************************/
BASED_ON RDB$INDICES.RDB$INDEX_NAME index_name; BASED_ON RDB$INDICES.RDB$INDEX_NAME index_name;
att_type attribute; att_type attribute;
bool foreign_index = false; bool foreign_index = false;
scan_attr_t scan_next_attr; bool expr_index = false;
scan_attr_t scan_next_attr;
SSHORT count = 0, segments = 0; SSHORT count = 0, segments = 0;
@ -7089,7 +7090,7 @@ bool get_index(BurpGlobals* tdgbl, const burp_rel* relation)
} }
else else
{ {
if (!X.RDB$INDEX_INACTIVE && foreign_index) if (!X.RDB$INDEX_INACTIVE && (foreign_index || expr_index))
X.RDB$INDEX_INACTIVE = DEFERRED_ACTIVE; X.RDB$INDEX_INACTIVE = DEFERRED_ACTIVE;
} }
if (tdgbl->gbl_sw_deactivate_indexes) if (tdgbl->gbl_sw_deactivate_indexes)
@ -7129,6 +7130,12 @@ bool get_index(BurpGlobals* tdgbl, const burp_rel* relation)
break; break;
case att_index_expression_blr: case att_index_expression_blr:
expr_index = true;
// Defer expression index activation
if (!X.RDB$INDEX_INACTIVE)
X.RDB$INDEX_INACTIVE = DEFERRED_ACTIVE;
if (tdgbl->gbl_sw_deactivate_indexes)
X.RDB$INDEX_INACTIVE = TRUE;
X.RDB$EXPRESSION_BLR.NULL = FALSE; X.RDB$EXPRESSION_BLR.NULL = FALSE;
get_blr_blob (tdgbl, X.RDB$EXPRESSION_BLR, false); get_blr_blob (tdgbl, X.RDB$EXPRESSION_BLR, false);
break; break;
@ -11317,6 +11324,7 @@ void update_global_field(BurpGlobals* tdgbl)
ON_ERROR ON_ERROR
general_on_error (); general_on_error ();
END_ERROR; END_ERROR;
gfld* n_gfield = gfield->gfld_next; gfld* n_gfield = gfield->gfld_next;
BURP_free (gfield); BURP_free (gfield);
gfield = n_gfield; gfield = n_gfield;

View File

@ -2496,6 +2496,8 @@ SLONG MET_lookup_index_name(thread_db* tdbb,
{ {
if (X.RDB$INDEX_INACTIVE == 0) if (X.RDB$INDEX_INACTIVE == 0)
*status = MET_object_active; *status = MET_object_active;
else if (X.RDB$INDEX_INACTIVE == 3)
*status = MET_object_deferred_active;
else else
*status = MET_object_inactive; *status = MET_object_inactive;

View File

@ -51,6 +51,7 @@ namespace Jrd
enum IndexStatus enum IndexStatus
{ {
MET_object_active, MET_object_active,
MET_object_deferred_active,
MET_object_inactive, MET_object_inactive,
MET_object_unknown MET_object_unknown
}; };

View File

@ -972,7 +972,7 @@ static void check_indices(const CompilerScratch::csb_repeat* csb_tail)
// if there were no indices fetched at all but the // if there were no indices fetched at all but the
// user specified some, error out using the first index specified // user specified some, error out using the first index specified
if (!csb_tail->csb_indices && plan->accessType) if (!csb_tail->csb_indices && plan->accessType && !tdbb->getAttachment()->isGbak())
{ {
// index %s cannot be used in the specified plan // index %s cannot be used in the specified plan
ERR_post(Arg::Gds(isc_index_unused) << plan->accessType->items[0].indexName); ERR_post(Arg::Gds(isc_index_unused) << plan->accessType->items[0].indexName);

View File

@ -994,6 +994,8 @@ static PlanNode* par_plan(thread_db* tdbb, CompilerScratch* csb)
node_type = (USHORT) csb->csb_blr_reader.getByte(); node_type = (USHORT) csb->csb_blr_reader.getByte();
const bool isGbak = tdbb->getAttachment()->isGbak();
switch (node_type) switch (node_type)
{ {
case blr_navigational: case blr_navigational:
@ -1012,7 +1014,7 @@ static PlanNode* par_plan(thread_db* tdbb, CompilerScratch* csb)
if (idx_status == MET_object_unknown || idx_status == MET_object_inactive) if (idx_status == MET_object_unknown || idx_status == MET_object_inactive)
{ {
if (tdbb->getAttachment()->isGbak()) if (isGbak)
{ {
PAR_warning(Arg::Warning(isc_indexname) << Arg::Str(name) << PAR_warning(Arg::Warning(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name)); Arg::Str(relation->rel_name));
@ -1020,7 +1022,15 @@ static PlanNode* par_plan(thread_db* tdbb, CompilerScratch* csb)
else else
{ {
PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) << PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name)); Arg::Str(relation->rel_name));
}
}
else if (idx_status == MET_object_deferred_active)
{
if (!isGbak)
{
PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name));
} }
} }
@ -1070,7 +1080,7 @@ static PlanNode* par_plan(thread_db* tdbb, CompilerScratch* csb)
if (idx_status == MET_object_unknown || idx_status == MET_object_inactive) if (idx_status == MET_object_unknown || idx_status == MET_object_inactive)
{ {
if (tdbb->getAttachment()->isGbak()) if (isGbak)
{ {
PAR_warning(Arg::Warning(isc_indexname) << Arg::Str(name) << PAR_warning(Arg::Warning(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name)); Arg::Str(relation->rel_name));
@ -1078,7 +1088,15 @@ static PlanNode* par_plan(thread_db* tdbb, CompilerScratch* csb)
else else
{ {
PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) << PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name)); Arg::Str(relation->rel_name));
}
}
else if (idx_status == MET_object_deferred_active)
{
if (!isGbak)
{
PAR_error(csb, Arg::Gds(isc_indexname) << Arg::Str(name) <<
Arg::Str(relation->rel_name));
} }
} }