ProgTutorial/Recipes/external_solver.ML
author griff
Tue, 21 Jul 2009 13:22:17 +0200
changeset 274 f9f3ecc949c5
parent 189 069d525f8f1d
permissions -rw-r--r--
still merging

signature IFF_SOLVER =
sig
  val decide : string -> bool
end


structure IffSolver : IFF_SOLVER =
struct

exception FAIL

datatype formula = Atom of string | Iff of formula * formula

fun scan s =
  let

    fun push yss = [] :: yss 

    fun add x (ys :: yss) = (x :: ys) :: yss
      | add _ _ = raise FAIL

    fun pop ([a, b] :: yss) = add (Iff (b, a)) yss
      | pop _ = raise FAIL

    fun formula [] acc = acc
      | formula ("(" :: xs) acc = formula xs (push acc)
      | formula (")" :: xs) acc = formula xs (pop acc)
      | formula ("<=>" :: xs) acc = formula xs acc
      | formula (x :: xs) acc = formula xs (add (Atom x) acc)

    val tokens = String.tokens (fn c => c = #" ") s
  in
    (case formula tokens [[]] of
      [[f]] => f
    | _ => raise FAIL)
  end


fun decide s =
  let
    fun fold_atoms f (Atom n) = f s
      | fold_atoms f (Iff (p, q)) = fold_atoms f p #> fold_atoms f q

    fun collect s tab =
      (case Symtab.lookup tab s of
        NONE => Symtab.update (s, false) tab
      | SOME v => Symtab.update (s, not v) tab)

    fun check tab = Symtab.fold (fn (_, v) => fn b => b andalso v) tab true
  in 
    (case try scan s of
      NONE => false
    | SOME f => check (fold_atoms collect f Symtab.empty))
  end

end