src/HOL/Tools/prop_logic.ML
author webertj
Sun, 13 Jun 2004 17:57:35 +0200
changeset 14939 29fe4a9a7cb5
parent 14753 f40b45db8cf0
child 14964 2c1456d705e9
permissions -rw-r--r--
faster defcnf conversion
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     1
(*  Title:      HOL/Tools/prop_logic.ML
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     2
    ID:         $Id$
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     3
    Author:     Tjark Weber
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     4
    Copyright   2004
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     5
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     6
Formulas of propositional logic.
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     7
*)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     8
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
     9
signature PROP_LOGIC =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    10
sig
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    11
	datatype prop_formula =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    12
		  True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    13
		| False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    14
		| BoolVar of int  (* NOTE: only use indices >= 1 *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    15
		| Not of prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    16
		| Or of prop_formula * prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    17
		| And of prop_formula * prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    18
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    19
	val SNot : prop_formula -> prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    20
	val SOr  : prop_formula * prop_formula -> prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    21
	val SAnd : prop_formula * prop_formula -> prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    22
14681
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
    23
	val indices : prop_formula -> int list  (* set of all variable indices *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    24
	val maxidx  : prop_formula -> int  (* maximal variable index *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    25
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    26
	val nnf    : prop_formula -> prop_formula  (* negation normal form *)
14681
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
    27
	val cnf    : prop_formula -> prop_formula  (* conjunctive normal form *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    28
	val defcnf : prop_formula -> prop_formula  (* definitional cnf *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    29
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    30
	val exists      : prop_formula list -> prop_formula  (* finite disjunction *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    31
	val all         : prop_formula list -> prop_formula  (* finite conjunction *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    32
	val dot_product : prop_formula list * prop_formula list -> prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    33
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    34
	val eval : (int -> bool) -> prop_formula -> bool  (* semantics *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    35
end;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    36
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    37
structure PropLogic : PROP_LOGIC =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    38
struct
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    39
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    40
(* ------------------------------------------------------------------------- *)
14753
f40b45db8cf0 Comments fixed
webertj
parents: 14681
diff changeset
    41
(* prop_formula: formulas of propositional logic, built from Boolean         *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    42
(*               variables (referred to by index) and True/False using       *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    43
(*               not/or/and                                                  *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    44
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    45
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    46
	datatype prop_formula =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    47
		  True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    48
		| False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    49
		| BoolVar of int  (* NOTE: only use indices >= 1 *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    50
		| Not of prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    51
		| Or of prop_formula * prop_formula
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    52
		| And of prop_formula * prop_formula;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    53
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    54
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    55
(* The following constructor functions make sure that True and False do not  *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    56
(* occur within any of the other connectives (i.e. Not, Or, And), and        *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    57
(* perform double-negation elimination.                                      *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    58
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    59
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    60
	(* prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    61
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    62
	fun SNot True     = False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    63
	  | SNot False    = True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    64
	  | SNot (Not fm) = fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    65
	  | SNot fm       = Not fm;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    66
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    67
	(* prop_formula * prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    68
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    69
	fun SOr (True, _)   = True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    70
	  | SOr (_, True)   = True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    71
	  | SOr (False, fm) = fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    72
	  | SOr (fm, False) = fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    73
	  | SOr (fm1, fm2)  = Or (fm1, fm2);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    74
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    75
	(* prop_formula * prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    76
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    77
	fun SAnd (True, fm) = fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    78
	  | SAnd (fm, True) = fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    79
	  | SAnd (False, _) = False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    80
	  | SAnd (_, False) = False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    81
	  | SAnd (fm1, fm2) = And (fm1, fm2);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    82
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    83
(* ------------------------------------------------------------------------- *)
14753
f40b45db8cf0 Comments fixed
webertj
parents: 14681
diff changeset
    84
(* indices: collects all indices of Boolean variables that occur in a        *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    85
(*      propositional formula 'fm'; no duplicates                            *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    86
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    87
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    88
	(* prop_formula -> int list *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    89
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    90
	fun indices True            = []
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    91
	  | indices False           = []
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    92
	  | indices (BoolVar i)     = [i]
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    93
	  | indices (Not fm)        = indices fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    94
	  | indices (Or (fm1,fm2))  = (indices fm1) union_int (indices fm2)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    95
	  | indices (And (fm1,fm2)) = (indices fm1) union_int (indices fm2);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    96
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    97
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    98
(* maxidx: computes the maximal variable index occuring in a formula of      *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
    99
(*      propositional logic 'fm'; 0 if 'fm' contains no variable             *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   100
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   101
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   102
	(* prop_formula -> int *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   103
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   104
	fun maxidx True            = 0
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   105
	  | maxidx False           = 0
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   106
	  | maxidx (BoolVar i)     = i
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   107
	  | maxidx (Not fm)        = maxidx fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   108
	  | maxidx (Or (fm1,fm2))  = Int.max (maxidx fm1, maxidx fm2)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   109
	  | maxidx (And (fm1,fm2)) = Int.max (maxidx fm1, maxidx fm2);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   110
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   111
(* ------------------------------------------------------------------------- *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   112
(* exception SAME: raised to indicate that the return value of a function is *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   113
(*                 identical to its argument (optimization to allow sharing, *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   114
(*                 rather than copying)                                      *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   115
(* ------------------------------------------------------------------------- *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   116
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   117
	exception SAME;
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   118
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   119
(* ------------------------------------------------------------------------- *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   120
(* nnf: computes the negation normal form of a formula 'fm' of propositional *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   121
(*      logic (i.e. only variables may be negated, but not subformulas)      *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   122
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   123
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   124
	(* prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   125
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   126
	fun
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   127
	(* constants *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   128
	    nnf True                  = True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   129
	  | nnf False                 = False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   130
	(* variables *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   131
	  | nnf (BoolVar i)           = (BoolVar i)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   132
	(* 'or' and 'and' as outermost connectives are left untouched *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   133
	  | nnf (Or  (fm1,fm2))       = SOr  (nnf fm1, nnf fm2)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   134
	  | nnf (And (fm1,fm2))       = SAnd (nnf fm1, nnf fm2)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   135
	(* 'not' + constant *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   136
	  | nnf (Not True)            = False
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   137
	  | nnf (Not False)           = True
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   138
	(* 'not' + variable *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   139
	  | nnf (Not (BoolVar i))     = Not (BoolVar i)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   140
	(* pushing 'not' inside of 'or'/'and' using de Morgan's laws *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   141
	  | nnf (Not (Or  (fm1,fm2))) = SAnd (nnf (SNot fm1), nnf (SNot fm2))
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   142
	  | nnf (Not (And (fm1,fm2))) = SOr  (nnf (SNot fm1), nnf (SNot fm2))
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   143
	(* double-negation elimination *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   144
	  | nnf (Not (Not fm))        = nnf fm;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   145
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   146
(* ------------------------------------------------------------------------- *)
14681
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   147
(* cnf: computes the conjunctive normal form (i.e. a conjunction of          *)
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   148
(*      disjunctions) of a formula 'fm' of propositional logic.  The result  *)
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   149
(*      formula may be exponentially longer than 'fm'.                       *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   150
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   151
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   152
	(* prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   153
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   154
	fun cnf fm =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   155
	let
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   156
		fun
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   157
		(* constants *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   158
		    cnf_from_nnf True             = True
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   159
		  | cnf_from_nnf False            = False
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   160
		(* literals *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   161
		  | cnf_from_nnf (BoolVar i)      = BoolVar i
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   162
		  | cnf_from_nnf (Not fm1)        = Not fm1  (* 'fm1' must be a variable since the formula is in NNF *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   163
		(* pushing 'or' inside of 'and' using distributive laws *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   164
		  | cnf_from_nnf (Or (fm1, fm2))  =
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   165
			let
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   166
				fun cnf_or (And (fm11, fm12), fm2) =
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   167
					And (cnf_or (fm11, fm2), cnf_or (fm12, fm2))
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   168
				  | cnf_or (fm1, And (fm21, fm22)) =
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   169
					And (cnf_or (fm1, fm21), cnf_or (fm1, fm22))
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   170
				(* neither subformula contains 'and' *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   171
				  | cnf_or (fm1, fm2) =
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   172
					Or (fm1, fm2)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   173
			in
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   174
				cnf_or (cnf_from_nnf fm1, cnf_from_nnf fm2)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   175
			end
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   176
		(* 'and' as outermost connective is left untouched *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   177
		  | cnf_from_nnf (And (fm1, fm2)) = And (cnf_from_nnf fm1, cnf_from_nnf fm2)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   178
	in
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   179
		(cnf_from_nnf o nnf) fm
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   180
	end;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   181
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   182
(* ------------------------------------------------------------------------- *)
14681
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   183
(* defcnf: computes the definitional conjunctive normal form of a formula    *)
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   184
(*      'fm' of propositional logic, introducing auxiliary variables if      *)
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   185
(*      necessary to avoid an exponential blowup of the formula.  The result *)
16fcef3a3174 comments modified
webertj
parents: 14452
diff changeset
   186
(*      formula is satisfiable if and only if 'fm' is satisfiable.           *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   187
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   188
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   189
	(* prop_formula -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   190
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   191
	fun defcnf fm =
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   192
	let
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   193
		(* prop_formula * int -> prop_formula * int *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   194
		(* 'new' specifies the next index that is available to introduce an auxiliary variable *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   195
		fun
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   196
		(* constants *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   197
		    defcnf_from_nnf (True, new)            = (True, new)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   198
		  | defcnf_from_nnf (False, new)           = (False, new)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   199
		(* literals *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   200
		  | defcnf_from_nnf (BoolVar i, new)       = (BoolVar i, new)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   201
		  | defcnf_from_nnf (Not fm1, new)         = (Not fm1, new)  (* 'fm1' must be a variable since the formula is in NNF *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   202
		(* pushing 'or' inside of 'and' using auxiliary variables *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   203
		  | defcnf_from_nnf (Or (fm1, fm2), new)   =
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   204
			let
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   205
				val (fm1', new')  = defcnf_from_nnf (fm1, new)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   206
				val (fm2', new'') = defcnf_from_nnf (fm2, new')
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   207
				(* prop_formula * prop_formula -> int -> prop_formula * int *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   208
				fun defcnf_or (And (fm11, fm12), fm2) new =
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   209
					let
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   210
						val aux            = BoolVar new
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   211
						val (fm_a, new')   = defcnf_or (fm11, aux)     (new+1)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   212
						val (fm_b, new'')  = defcnf_or (fm12, aux)     new'
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   213
						val (fm_c, new''') = defcnf_or (fm2,  Not aux) new''
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   214
					in
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   215
						(And (And (fm_a, fm_b), fm_c), new''')
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   216
					end
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   217
				  | defcnf_or (fm1, And (fm21, fm22)) new =
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   218
					let
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   219
						val aux            = BoolVar new
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   220
						val (fm_a, new')   = defcnf_or (fm1,  Not aux) (new+1)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   221
						val (fm_b, new'')  = defcnf_or (fm21, aux)     new'
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   222
						val (fm_c, new''') = defcnf_or (fm22, aux)     new''
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   223
					in
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   224
						(And (fm_a, And (fm_b, fm_c)), new''')
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   225
					end
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   226
				(* neither subformula contains 'and' *)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   227
				  | defcnf_or (fm1, fm2) new =
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   228
					(Or (fm1, fm2), new)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   229
			in
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   230
				defcnf_or (fm1', fm2') new''
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   231
			end
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   232
		(* 'and' as outermost connective is left untouched *)
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   233
		  | defcnf_from_nnf (And (fm1, fm2), new)   =
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   234
			let
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   235
				val (fm1', new')  = defcnf_from_nnf (fm1, new)
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   236
				val (fm2', new'') = defcnf_from_nnf (fm2, new')
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   237
			in
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   238
				(And (fm1', fm2'), new'')
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   239
			end
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   240
		val fm'    = nnf fm
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   241
	in
14939
29fe4a9a7cb5 faster defcnf conversion
webertj
parents: 14753
diff changeset
   242
		(fst o defcnf_from_nnf) (fm', (maxidx fm')+1)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   243
	end;
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   244
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   245
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   246
(* exists: computes the disjunction over a list 'xs' of propositional        *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   247
(*      formulas                                                             *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   248
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   249
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   250
	(* prop_formula list -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   251
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   252
	fun exists xs = foldl SOr (False, xs);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   253
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   254
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   255
(* all: computes the conjunction over a list 'xs' of propositional formulas  *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   256
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   257
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   258
	(* prop_formula list -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   259
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   260
	fun all xs = foldl SAnd (True, xs);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   261
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   262
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   263
(* dot_product: ([x1,...,xn], [y1,...,yn]) -> x1*y1+...+xn*yn                *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   264
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   265
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   266
	(* prop_formula list * prop_formula list -> prop_formula *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   267
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   268
	fun dot_product (xs,ys) = exists (map SAnd (xs~~ys));
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   269
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   270
(* ------------------------------------------------------------------------- *)
14753
f40b45db8cf0 Comments fixed
webertj
parents: 14681
diff changeset
   271
(* eval: given an assignment 'a' of Boolean values to variable indices, the  *)
14452
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   272
(*      truth value of a propositional formula 'fm' is computed              *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   273
(* ------------------------------------------------------------------------- *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   274
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   275
	(* (int -> bool) -> prop_formula -> bool *)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   276
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   277
	fun eval a True            = true
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   278
	  | eval a False           = false
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   279
	  | eval a (BoolVar i)     = (a i)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   280
	  | eval a (Not fm)        = not (eval a fm)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   281
	  | eval a (Or (fm1,fm2))  = (eval a fm1) orelse (eval a fm2)
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   282
	  | eval a (And (fm1,fm2)) = (eval a fm1) andalso (eval a fm2);
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   283
c24d90dbf0c9 Formulas of propositional logic
webertj
parents:
diff changeset
   284
end;