From 4049dd0ba194a7c2c9f4ad36973dc1cc74e84516 Mon Sep 17 00:00:00 2001 From: "Jip J. Dekker" Date: Thu, 14 Nov 2024 15:10:44 +1100 Subject: [PATCH] Fix crash when type instance concatenation is performed on invalid types Resolves #867 --- changes.rst | 2 + lib/typecheck.cpp | 121 +++++++++++----------- tests/spec/unit/regression/github_867.mzn | 8 ++ 3 files changed, 70 insertions(+), 61 deletions(-) create mode 100644 tests/spec/unit/regression/github_867.mzn diff --git a/changes.rst b/changes.rst index 4656de237..271950ce2 100644 --- a/changes.rst +++ b/changes.rst @@ -32,6 +32,8 @@ Bug fixes: - Fix assertion failures when using arrays as argument to bin packing constraints (:bugref:`865`). - Fix operator precedences for ``intersect`` and unary plus/minus. +- Fix crash when type instance concatenation is performed on invalid types + (:bugref:`867`). .. _v2.8.7: diff --git a/lib/typecheck.cpp b/lib/typecheck.cpp index 407942909..f66394dec 100644 --- a/lib/typecheck.cpp +++ b/lib/typecheck.cpp @@ -2588,7 +2588,66 @@ class Typer { std::vector args(2); args[0] = bop->lhs(); args[1] = bop->rhs(); - if (FunctionI* fi = _model->matchFn(_env, bop->opToString(), args, true)) { + if (bop->op() == BOT_PLUSPLUS && (Expression::isa(bop->lhs()) || + (Expression::isa(bop->lhs()) && + Expression::cast(bop->lhs())->decl()->isTypeAlias()))) { + // Special case: concatenating type expressions + + // Check whether the rhs is also a type expression + if (!(Expression::isa(bop->rhs()) || + (Expression::isa(bop->rhs()) && + Expression::cast(bop->rhs())->decl()->isTypeAlias()))) { + std::ostringstream ss; + ss << "operator application for `" << bop->opToString() + << "' cannot combine type expression`" << bop->lhs() << "' with expression `" + << bop->rhs() << "'."; + throw TypeError(_env, Expression::loc(bop), ss.str()); + } + + // Resolve potential type aliases + if (auto* alias = Expression::dynamicCast(bop->lhs())) { + bop->lhs(Expression::cast(alias->decl()->e())); + } + if (auto* alias = Expression::dynamicCast(bop->rhs())) { + bop->rhs(Expression::cast(alias->decl()->e())); + } + + Type lhsT = Expression::type(bop->lhs()); + Type rhsT = Expression::type(bop->rhs()); + // Check whether type expressions can be correctly combined + if (!lhsT.structBT()) { + std::ostringstream ss; + ss << "operator application for `" << bop->opToString() << "' is not allowed on the `" + << lhsT.toString(_env) << "' type instance."; + throw TypeError(_env, Expression::loc(bop), ss.str()); + } + if (!rhsT.structBT()) { + std::ostringstream ss; + ss << "operator application for `" << bop->opToString() << "' is not allowed on the `" + << rhsT.toString(_env) << "' type instance."; + throw TypeError(_env, Expression::loc(bop), ss.str()); + } + if (lhsT.bt() != rhsT.bt()) { + std::ostringstream ss; + ss << "operator application for `" << bop->opToString() << "' cannot combine type `" + << lhsT.toString(_env) << "' with typ `" << rhsT.toString(_env) << "'."; + throw TypeError(_env, Expression::loc(bop), ss.str()); + } + + // Note: BinOp is resolved during typechecking of TypeInst + } else if (bop->op() == BOT_PLUSPLUS && Expression::type(bop->lhs()).structBT() && + Expression::type(bop->lhs()).bt() == Expression::type(bop->rhs()).bt() && + Expression::type(bop->lhs()).dim() == 0 && Expression::type(bop->rhs()).dim() == 0) { + // Special case: concatenating tuples or records + Type lhsT = Expression::type(bop->lhs()); + Type rhsT = Expression::type(bop->rhs()); + if (lhsT.isrecord()) { + bop->type(_env.mergeRecord(lhsT, rhsT, Expression::loc(bop))); + } else { + assert(lhsT.istuple()); + bop->type(_env.concatTuple(lhsT, rhsT)); + } + } else if (FunctionI* fi = _model->matchFn(_env, bop->opToString(), args, true)) { bop->lhs(add_coercion(_env, _model, bop->lhs(), fi->argtype(_env, args, 0))()); bop->rhs(add_coercion(_env, _model, bop->rhs(), fi->argtype(_env, args, 1))()); args[0] = bop->lhs(); @@ -2730,66 +2789,6 @@ class Typer { } } } - } else if (bop->op() == BOT_PLUSPLUS && - (Expression::isa(bop->lhs()) || - (Expression::isa(bop->lhs()) && - Expression::cast(bop->lhs())->decl()->isTypeAlias()))) { - // Special case: concatenating type expressions - - // Check whether the rhs is also a type expression - if (!(Expression::isa(bop->rhs()) || - (Expression::isa(bop->rhs()) && - Expression::cast(bop->rhs())->decl()->isTypeAlias()))) { - std::ostringstream ss; - ss << "operator application for `" << bop->opToString() - << "' cannot combine type expression`" << bop->lhs() << "' with expression `" - << bop->rhs() << "'."; - throw TypeError(_env, Expression::loc(bop), ss.str()); - } - - // Resolve potential type aliases - if (auto* alias = Expression::dynamicCast(bop->lhs())) { - bop->lhs(Expression::cast(alias->decl()->e())); - } - if (auto* alias = Expression::dynamicCast(bop->rhs())) { - bop->rhs(Expression::cast(alias->decl()->e())); - } - - Type lhsT = Expression::type(bop->lhs()); - Type rhsT = Expression::type(bop->rhs()); - // Check whether type expressions can be correctly combined - if (!lhsT.structBT()) { - std::ostringstream ss; - ss << "operator application for `" << bop->opToString() << "' is not allowed on the `" - << lhsT.toString(_env) << "' type."; - throw TypeError(_env, Expression::loc(bop), ss.str()); - } - if (!rhsT.structBT()) { - std::ostringstream ss; - ss << "operator application for `" << bop->opToString() << "' is not allowed on the `" - << rhsT.toString(_env) << "' type."; - throw TypeError(_env, Expression::loc(bop), ss.str()); - } - if (lhsT.bt() != rhsT.bt()) { - std::ostringstream ss; - ss << "operator application for `" << bop->opToString() << "' cannot combine type `" - << lhsT.toString(_env) << "' with typ `" << rhsT.toString(_env) << "'."; - throw TypeError(_env, Expression::loc(bop), ss.str()); - } - - // Note: BinOp is resolved during typechecking of TypeInst - } else if (bop->op() == BOT_PLUSPLUS && Expression::type(bop->lhs()).structBT() && - Expression::type(bop->lhs()).bt() == Expression::type(bop->rhs()).bt() && - Expression::type(bop->lhs()).dim() == 0 && Expression::type(bop->rhs()).dim() == 0) { - // Special case: concatenating tuples or records - Type lhsT = Expression::type(bop->lhs()); - Type rhsT = Expression::type(bop->rhs()); - if (lhsT.isrecord()) { - bop->type(_env.mergeRecord(lhsT, rhsT, Expression::loc(bop))); - } else { - assert(lhsT.istuple()); - bop->type(_env.concatTuple(lhsT, rhsT)); - } } else { std::ostringstream ss; ss << "type error in operator application for `" << bop->opToString() diff --git a/tests/spec/unit/regression/github_867.mzn b/tests/spec/unit/regression/github_867.mzn new file mode 100644 index 000000000..2787d5d2b --- /dev/null +++ b/tests/spec/unit/regression/github_867.mzn @@ -0,0 +1,8 @@ +/*** +!Test +solvers: [gecode] +expected: !Error + type: TypeError +***/ + +set of 1..2 ++ set of 2..3: no; \ No newline at end of file