Skip to content

Commit

Permalink
Merge pull request #2856 from informalsystems/gabriela/fix-quint-type…
Browse files Browse the repository at this point in the history
…-conversion

Use type given by quint to build lambda bodies
  • Loading branch information
bugarela authored Mar 8, 2024
2 parents 811b480 + 3dac602 commit dd1fef9
Show file tree
Hide file tree
Showing 4 changed files with 32 additions and 8 deletions.
3 changes: 3 additions & 0 deletions .unreleased/bug-fixes/quint-lambda-types-conversion.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
When converting Quint lambdas, derive the return type from the Quint type inferred for
the lambda, rather the type inferred for the body expression, avoiding mismatches with
Apalache type variables. (#2856)
16 changes: 8 additions & 8 deletions tla-io/src/main/scala/at/forsyte/apalache/io/quint/Quint.scala
Original file line number Diff line number Diff line change
Expand Up @@ -82,18 +82,18 @@ class Quint(quintOutput: QuintOutput) {
// operators that take parameters, but these require different constructs
// in Apalache's IR. Thus, we need to decompose the parts of a QuintLambda
// for two different purposes.
private val lambdaBodyAndParams: QuintLambda => NullaryOpReader[(TBuilderInstruction, Seq[(OperParam, TlaType1)])] = {
private val lambdaBodyAndParams: QuintLambda => NullaryOpReader[(TBuilderInstruction, Seq[(OperParam, TlaType1)], TlaType1)] = {
case ex @ QuintLambda(id, paramNames, _, body) =>
val quintParamTypes = types(id).typ match {
case QuintOperT(types, _) => types
case invalidType => throw new QuintIRParseError(s"lambda ${ex} has invalid type ${invalidType}")
val (quintParamTypes, quintResType) = types(id).typ match {
case QuintOperT(types, resType) => (types, resType)
case invalidType => throw new QuintIRParseError(s"lambda ${ex} has invalid type ${invalidType}")
}
val operParams = paramNames.zip(quintParamTypes).map(operParam)
val paramTypes = quintParamTypes.map(typeConv.convert(_))
val typedParams = operParams.zip(paramTypes)
for {
tlaBody <- tlaExpression(body)
} yield (tlaBody, typedParams)
} yield (tlaBody, typedParams, typeConv.convert(quintResType))
}

private def typeTagOfId(id: BigInt): TypeTag = {
Expand Down Expand Up @@ -668,7 +668,7 @@ class Quint(quintOutput: QuintOutput) {
(expr match {
// Parameterized operators are defined in Quint using Lambdas
case lam: QuintLambda =>
lambdaBodyAndParams(lam)
lambdaBodyAndParams(lam).map { case (body, params, _) => (body, params) }
// Otherwise it's an operator with no params
case other => tlaExpression(other).map(b => (b, List()))
}).map {
Expand Down Expand Up @@ -716,8 +716,8 @@ class Quint(quintOutput: QuintOutput) {
.map(tlaExpr => tla.letIn(tlaExpr, tlaOpDef))
}
case lam: QuintLambda =>
lambdaBodyAndParams(lam).map { case (body, typedParams) =>
tla.lambda(nameGen.uniqueLambdaName(), body, typedParams: _*)
lambdaBodyAndParams(lam).map { case (body, typedParams, resultType) =>
tla.lambda(nameGen.uniqueLambdaName(), body, resultType, typedParams: _*)
}
case app: QuintApp => tlaApplication(app)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -637,6 +637,12 @@ class TestQuintEx extends AnyFunSuite {
assert(convert(quintMatch) == expected)
}

test("prefers quint types over inferred types") {
// Regression on https://github.com/informalsystems/quint/issues/1393
val expr = Q.lam(Seq("x" -> QuintBoolT()), Q.nam("x", QuintVarT("t0")), QuintBoolT())
assert(translate(expr).typeTag == Typed(OperT1(Seq(BoolT1), BoolT1)))
}

test("can convert builtin assert operator") {
assert(convert(Q.app("assert", Q.nIsGreaterThan0)(QuintBoolT())) == "n > 0")
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -299,6 +299,21 @@ class ScopedBuilder(val strict: Boolean = true)
} yield ex
}

/** Alternative to lambda, which accepts an explicit result type */
def lambda(
uniqueName: String,
body: TBuilderInstruction,
resultType: TlaType1,
params: TypedParam*): TBuilderInstruction = {
params.foreach(validateParamType)
for {
bodyEx <- body
paramTypes = params.map(_._2)
operType = OperT1(paramTypes, resultType) // use given type instead of type tag
ex <- letIn(name(uniqueName, operType), decl(uniqueName, body, params: _*))
} yield ex
}

/** {{{LET decl(...) = ... IN body}}} */
def letIn(body: TBuilderInstruction, decl: TBuilderOperDeclInstruction): TBuilderInstruction = for {
usedBeforeDecl <- getAllUsed // decl name may not appear in decl body
Expand Down

0 comments on commit dd1fef9

Please sign in to comment.