/src/vgo/verification/plugins/nusmv/smv.go

https://bitbucket.org/teythoon/vgo · Go · 188 lines · 146 code · 29 blank · 13 comment · 13 complexity · 9d9d441e9640f0e13acb924a58d064ac MD5 · raw file

  1. // Copyright (c) 2013 Justus Winter <4winter@informatik.uni-hamburg.de>
  2. //
  3. // Permission to use, copy, modify, and distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  8. // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  9. // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  10. // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  11. // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  12. // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  13. // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  14. package nusmv
  15. import (
  16. "fmt"
  17. "io"
  18. "regexp"
  19. "strings"
  20. "text/template"
  21. "vgo/human"
  22. "vgo/logic"
  23. "vgo/verification"
  24. )
  25. func writeSMV(w io.Writer, t *template.Template, m verification.Model, f []logic.Formula) (*smvContext, error) {
  26. c := &smvContext{
  27. Model: m,
  28. formulas: f,
  29. atomMap: make(human.StringIds),
  30. }
  31. return c, t.Execute(w, c)
  32. }
  33. func WriteSMV(m verification.Model, w io.Writer, f []logic.Formula) (err error) {
  34. _, err = writeSMV(w, smvTemplate, m, f)
  35. return
  36. }
  37. func WriteSMVMin(w io.Writer, m verification.Model, f []logic.Formula) (err error) {
  38. _, err = writeSMV(w, smvTemplateMin, m, f)
  39. return
  40. }
  41. type smvContext struct {
  42. verification.Model
  43. formulas []logic.Formula
  44. atomMap human.StringIds
  45. }
  46. func (s *smvContext) States() int {
  47. return len(s.Model.Nodes())
  48. }
  49. func (s *smvContext) Transitions() string {
  50. r := make([]string, 0, len(s.Model.Edges()))
  51. for _, e := range s.Model.Edges() {
  52. r = append(r, fmt.Sprintf("(_=%v&next(_)=%v)", e.From, e.To))
  53. }
  54. return strings.Join(r, "|")
  55. }
  56. func (s *smvContext) Vars() (r []atom) {
  57. for _, a := range s.Model.Atoms() {
  58. r = append(r, s.atom(a))
  59. }
  60. return
  61. }
  62. func (s *smvContext) Formulas() (r []string) {
  63. for _, f := range s.formulas {
  64. r = append(r, s.smvFormula(f))
  65. }
  66. return
  67. }
  68. type atom struct {
  69. *smvContext
  70. *logic.AtomicProposition
  71. }
  72. func (a atom) String() string {
  73. return fmt.Sprintf("a%v", a.smvContext.atomMap.Map(a.Symbol()))
  74. }
  75. const caseDefault = "TRUE:FALSE;"
  76. func (a atom) Assignment() string {
  77. r := make([]string, 0, a.Model.Len())
  78. for n, _ := range a.Model.Nodes() {
  79. if a.Model.EvaluateAtom(n, a.AtomicProposition) {
  80. r = append(r, fmt.Sprintf("_=%v", n))
  81. }
  82. }
  83. if len(r) == 0 {
  84. return caseDefault
  85. }
  86. return fmt.Sprintf("%v:TRUE;"+caseDefault, strings.Join(r, "|"))
  87. }
  88. func (s *smvContext) atom(a *logic.AtomicProposition) atom {
  89. return atom{s, a}
  90. }
  91. func (s *smvContext) smvFormula(f logic.Formula) string {
  92. symbol := f.Symbol()
  93. switch n := f.(type) {
  94. case *logic.AtomicProposition:
  95. return " " + s.atom(n).String()
  96. case *logic.Constant:
  97. if n == logic.Top {
  98. return "TRUE"
  99. }
  100. return "FALSE"
  101. case *logic.Not:
  102. symbol = "!"
  103. case *logic.Or:
  104. symbol = "|"
  105. case *logic.And:
  106. symbol = "&"
  107. case *logic.Until:
  108. return fmt.Sprintf("[%v U %v]", s.smvFormula(f.Operands()[0]), s.smvFormula(f.Operands()[1]))
  109. case *logic.ForAll:
  110. if o, ok := f.Operands()[0].(*logic.Release); ok {
  111. return fmt.Sprintf("!E [!%v U !%v]", s.smvFormula(o.Operands()[0]), s.smvFormula(o.Operands()[1]))
  112. }
  113. return " A" + s.smvFormula(f.Operands()[0])
  114. case *logic.Exists:
  115. if o, ok := f.Operands()[0].(*logic.Release); ok {
  116. return fmt.Sprintf("!A [!%v U !%v]", s.smvFormula(o.Operands()[0]), s.smvFormula(o.Operands()[1]))
  117. }
  118. return " E" + s.smvFormula(f.Operands()[0])
  119. }
  120. o := f.Operands()
  121. switch len(o) {
  122. case 1:
  123. return fmt.Sprintf("%s%s", symbol, s.smvFormula(o[0]))
  124. case 2:
  125. return fmt.Sprintf("(%s %s %s)", s.smvFormula(o[0]), symbol, s.smvFormula(o[1]))
  126. }
  127. panic("Not reached")
  128. }
  129. var smvTemplate, smvTemplateMin *template.Template
  130. func init() {
  131. m := "MODULE main\n"
  132. smvTemplate = template.Must(template.New("smv").Parse(m + smvTemplateString))
  133. norm := regexp.MustCompile(`\s+`)
  134. min := regexp.MustCompile(` ?([:;=]) ?`)
  135. t := norm.ReplaceAllString(smvTemplateString, " ")
  136. t = min.ReplaceAllString(t, "$1")
  137. t = strings.TrimSpace(t)
  138. smvTemplateMin = template.Must(template.New("smvmin").Parse(m + t))
  139. }
  140. const smvTemplateString = `
  141. VAR
  142. _: 0..{{.States}};
  143. {{range .Vars}} {{.}}: boolean;
  144. {{end}}
  145. TRANS
  146. {{.Transitions}};
  147. ASSIGN
  148. init(_) := 0;
  149. {{range .Vars}} {{.}} := case
  150. {{.Assignment}}
  151. esac;
  152. {{end}}
  153. {{range .Formulas}}
  154. CTLSPEC {{.}};
  155. {{end}}
  156. `