Skip to content

Commit

Permalink
Merge pull request viperproject#752 from viperproject/meilers_fix_751
Browse files Browse the repository at this point in the history
  • Loading branch information
marcoeilers authored Sep 26, 2023
2 parents 2e306af + 8527848 commit 4cdc6b9
Show file tree
Hide file tree
Showing 2 changed files with 36 additions and 15 deletions.
44 changes: 32 additions & 12 deletions src/main/scala/decider/TermToZ3APIConverter.scala
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ class TermToZ3APIConverter

val sortCache = mutable.HashMap[Sort, Z3Sort]()
val funcDeclCache = mutable.HashMap[(String, Seq[Sort], Sort), Z3FuncDecl]()
val smtFuncDeclCache = mutable.HashMap[(String, Seq[Sort]), (Z3FuncDecl, Seq[Z3Expr])]()
val termCache = mutable.HashMap[Term, Z3Expr]()

def convert(s: Sort): Z3Sort = convertSort(s)
Expand Down Expand Up @@ -471,19 +472,37 @@ class TermToZ3APIConverter
// workaround: since we cannot create a function application with just the name, we let Z3 parse
// a string that uses the function, take the AST, and get the func decl from there, so that we can
// programmatically create a func app.
val decls = args.zipWithIndex.map{case (a, i) => s"(declare-const workaround${i} ${smtlibConverter.convert(a.sort)})"}.mkString(" ")
val funcAppString = s"(${functionName} ${(0 until args.length).map(i => "workaround" + i).mkString(" ")})"
val assertion = decls + s" (assert (= ${funcAppString} ${funcAppString}))"
val workaround = ctx.parseSMTLIB2String(assertion, null, null, null, null)
val app = workaround(0).getArgs()(0)
val decl = app.getFuncDecl
val actualArgs = if (decl.getArity > args.length){
// the function name we got wasn't just a function name but also contained a first argument.
// this happens with float operations where functionName contains a rounding mode.
app.getArgs.toSeq.slice(0, decl.getArity - args.length) ++ args.map(convertTerm(_))
}else {
args.map(convertTerm(_))

val cacheKey = (functionName, args.map(_.sort))
val (decl, additionalArgs: Seq[Z3Expr]) = if (smtFuncDeclCache.contains(cacheKey)) {
smtFuncDeclCache(cacheKey)
} else {
val declPreamble = "(define-sort $Perm () Real) " // ME: Re-declare the Perm sort.
// ME: The parsing happens in a fresh context that doesn't know any of our current declarations.
// In principle, it might be necessary to re-declare all sorts we're using anywhere. However, I don't see how there
// could be any Z3 internal functions that exist for those custom sorts. For the Real (i.e., Perm) sort, however,
// such functions exist. So we re-declare *only* this sort.
val decls = declPreamble + args.zipWithIndex.map { case (a, i) => s"(declare-const workaround${i} ${smtlibConverter.convert(a.sort)})" }.mkString(" ")
val funcAppString = if (args.nonEmpty)
s"(${functionName} ${(0 until args.length).map(i => "workaround" + i).mkString(" ")})"
else
functionName
val assertion = decls + s" (assert (= ${funcAppString} ${funcAppString}))"
val workaround = ctx.parseSMTLIB2String(assertion, null, null, null, null)
val app = workaround(0).getArgs()(0)
val decl = app.getFuncDecl
val additionalArgs = if (decl.getArity > args.length) {
// the function name we got wasn't just a function name but also contained a first argument.
// this happens with float operations where functionName contains a rounding mode.
app.getArgs.toSeq.slice(0, decl.getArity - args.length)
} else {
Seq()
}
smtFuncDeclCache.put(cacheKey, (decl, additionalArgs))
(decl, additionalArgs)
}

val actualArgs = additionalArgs ++ args.map(convertTerm(_))
ctx.mkApp(decl, actualArgs.toArray : _*)
}

Expand Down Expand Up @@ -522,6 +541,7 @@ class TermToZ3APIConverter
sanitizedNamesCache.clear()
macros.clear()
funcDeclCache.clear()
smtFuncDeclCache.clear()
sortCache.clear()
termCache.clear()
unitConstructor = null
Expand Down
7 changes: 4 additions & 3 deletions src/main/scala/decider/Z3ProverAPI.scala
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import com.typesafe.scalalogging.LazyLogging
import viper.silicon.common.config.Version
import viper.silicon.interfaces.decider.{Prover, Result, Sat, Unknown, Unsat}
import viper.silicon.state.IdentifierFactory
import viper.silicon.state.terms.{App, Decl, Fun, FunctionDecl, Implies, MacroDecl, Not, Quantification, Sort, SortDecl, SortWrapperDecl, Term, TriggerGenerator, sorts}
import viper.silicon.state.terms.{App, Decl, Fun, FunctionDecl, Implies, MacroDecl, Not, Quantification, Sort, SortDecl, SortWrapperDecl, Term, TriggerGenerator, Var, sorts}
import viper.silicon.{Config, Map}
import viper.silicon.verifier.Verifier
import viper.silver.reporter.{InternalWarningMessage, Reporter}
Expand Down Expand Up @@ -258,11 +258,12 @@ class Z3ProverAPI(uniqueId: String,
triggerGenerator.setCustomIsForbiddenInTrigger(triggerGenerator.advancedIsForbiddenInTrigger)
val cleanTerm = term.transform {
case q@Quantification(_, _, _, triggers, _, _, _) if triggers.nonEmpty =>
val goodTriggers = triggers.filterNot(trig => trig.p.exists(ptrn => ptrn.shallowCollect {
val goodTriggers = triggers.filterNot(trig => trig.p.exists(ptrn =>
ptrn.isInstanceOf[Var] || ptrn.shallowCollect {
case t => triggerGenerator.isForbiddenInTrigger(t)
}.nonEmpty))
q.copy(triggers = goodTriggers)
}()
}(_ => true)
cleanTerm
}

Expand Down

0 comments on commit 4cdc6b9

Please sign in to comment.