drnim.rst 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. .. default-role:: code
  2. ===================================
  3. DrNim User Guide
  4. ===================================
  5. :Author: Andreas Rumpf
  6. :Version: |nimversion|
  7. .. contents::
  8. Introduction
  9. ============
  10. This document describes the usage of the *DrNim* tool. DrNim combines
  11. the Nim frontend with the `Z3 <https://github.com/Z3Prover/z3>`_ proof
  12. engine, in order to allow verify/validate software written in Nim.
  13. DrNim's command-line options are the same as the Nim compiler's.
  14. DrNim currently only checks the sections of your code that are marked
  15. via `staticBoundChecks: on`:
  16. .. code-block:: nim
  17. {.push staticBoundChecks: on.}
  18. # <--- code section here ---->
  19. {.pop.}
  20. DrNim currently only tries to prove array indexing or subrange checks,
  21. overflow errors are *not* prevented. Overflows will be checked for in
  22. the future.
  23. Later versions of the **Nim compiler** will **assume** that the checks inside
  24. the `staticBoundChecks: on` environment have been proven correct and so
  25. it will **omit** the runtime checks. If you do not want this behavior, use
  26. instead `{.push staticBoundChecks: defined(nimDrNim).}`. This way the
  27. Nim compiler remains unaware of the performed proofs but DrNim will prove
  28. your code.
  29. Installation
  30. ============
  31. Run `koch drnim`, the executable will afterwards be in `$nim/bin/drnim`.
  32. Motivating Example
  33. ==================
  34. The follow example highlights what DrNim can easily do, even
  35. without additional annotations:
  36. .. code-block:: nim
  37. {.push staticBoundChecks: on.}
  38. proc sum(a: openArray[int]): int =
  39. for i in 0..a.len:
  40. result += a[i]
  41. {.pop.}
  42. echo sum([1, 2, 3])
  43. This program contains a famous "index out of bounds" bug. DrNim
  44. detects it and produces the following error message::
  45. cannot prove: i <= len(a) + -1; counter example: i -> 0 a.len -> 0 [IndexCheck]
  46. In other words for `i == 0` and `a.len == 0` (for example!) there would be
  47. an index out of bounds error.
  48. Pre-, postconditions and invariants
  49. ===================================
  50. DrNim adds 4 additional annotations (pragmas) to Nim:
  51. - `requires`:idx:
  52. - `ensures`:idx:
  53. - `invariant`:idx:
  54. - `assume`:idx:
  55. These pragmas are ignored by the Nim compiler so that they don't have to
  56. be disabled via `when defined(nimDrNim)`.
  57. Invariant
  58. ---------
  59. An `invariant` is a proposition that must be true after every loop
  60. iteration, it's tied to the loop body it's part of.
  61. Requires
  62. --------
  63. A `requires` annotation describes what the function expects to be true
  64. before it's called so that it can perform its operation. A `requires`
  65. annotation is also called a `precondition`:idx:.
  66. Ensures
  67. -------
  68. An `ensures` annotation describes what will be true after the function
  69. call. An `ensures` annotation is also called a `postcondition`:idx:.
  70. Assume
  71. ------
  72. An `assume` annotation describes what DrNim should **assume** to be true
  73. in this section of the program. It is an unsafe escape mechanism comparable
  74. to Nim's `cast` statement. Use it only when you really know better
  75. than DrNim. You should add a comment to a paper that proves the proposition
  76. you assume.
  77. Example: insertionSort
  78. ======================
  79. **Note**: This example does not yet work with DrNim.
  80. .. code-block:: nim
  81. import std / logic
  82. proc insertionSort(a: var openArray[int]) {.
  83. ensures: forall(i in 1..<a.len, a[i-1] <= a[i]).} =
  84. for k in 1 ..< a.len:
  85. {.invariant: 1 <= k and k <= a.len.}
  86. {.invariant: forall(j in 1..<k, i in 0..<j, a[i] <= a[j]).}
  87. var t = k
  88. while t > 0 and a[t-1] > a[t]:
  89. {.invariant: k < a.len.}
  90. {.invariant: 0 <= t and t <= k.}
  91. {.invariant: forall(j in 1..k, i in 0..<j, j == t or a[i] <= a[j]).}
  92. swap a[t], a[t-1]
  93. dec t
  94. Unfortunately, the invariants required to prove that this code is correct take more
  95. code than the imperative instructions. However, this effort can be compensated
  96. by the fact that the result needs very little testing. Be aware though that
  97. DrNim only proves that after `insertionSort` this condition holds::
  98. forall(i in 1..<a.len, a[i-1] <= a[i])
  99. This is required, but not sufficient to describe that a `sort` operation
  100. was performed. For example, the same postcondition is true for this proc
  101. which doesn't sort at all:
  102. .. code-block:: nim
  103. import std / logic
  104. proc insertionSort(a: var openArray[int]) {.
  105. ensures: forall(i in 1..<a.len, a[i-1] <= a[i]).} =
  106. # does not sort, overwrites `a`'s contents!
  107. for i in 0..<a.len: a[i] = i
  108. Syntax of propositions
  109. ======================
  110. The basic syntax is `ensures|requires|invariant: <prop>`.
  111. A `prop` is either a comparison or a compound::
  112. prop = nim_bool_expression
  113. | prop 'and' prop
  114. | prop 'or' prop
  115. | prop '->' prop # implication
  116. | prop '<->' prop
  117. | 'not' prop
  118. | '(' prop ')' # you can group props via ()
  119. | forallProp
  120. | existsProp
  121. forallProp = 'forall' '(' quantifierList ',' prop ')'
  122. existsProp = 'exists' '(' quantifierList ',' prop ')'
  123. quantifierList = quantifier (',' quantifier)*
  124. quantifier = <new identifier> 'in' nim_iteration_expression
  125. `nim_iteration_expression` here is an ordinary expression of Nim code
  126. that describes an iteration space, for example `1..4` or `1..<a.len`.
  127. `nim_bool_expression` here is an ordinary expression of Nim code of
  128. type `bool` like `a == 3` or `23 > a.len`.
  129. The supported subset of Nim code that can be used in these expressions
  130. is currently underspecified but `let` variables, function parameters
  131. and `result` (which represents the function's final result) are amenable
  132. for verification. The expressions must not have any side-effects and must
  133. terminate.
  134. The operators `forall`, `exists`, `->`, `<->` have to imported
  135. from `std / logic`.