-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharray-bsearch.timl
46 lines (41 loc) · 1.39 KB
/
array-bsearch.timl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
(* Binary search on arrays *)
structure ArrayBSearch = struct
open Basic
open Nat
open Array
datatype order =
OrdLt
| OrdEq
| OrdGt
absidx T_array_bsearch_on_range : BigO (fn m n => $m * log2 $n) with
(* this function uses refinements as preconditions *)
fun array_bsearch_on_range ['a] {m : Nat} {len : Nat} {l : Nat} {n : Nat} {n > 0} {l + n <= len} (p : 'a -- $m --> order) (a : array 'a {len}, l : nat {l}, n : nat {n}) return nat_less_than {len} using T_array_bsearch_on_range m n =
(* #1 is the syntax for natural number 1 (of type [nat {1}]) *)
case le (n, #1) of
Le => NatLT l
| Gt =>
let
val half = floor_half n
val rest = nat_minus (n, half)
val m = nat_plus (l, half)
val ord = p (sub (a, m))
in
case ord of
OrdLt => array_bsearch_on_range p (a, l, half)
| _ => array_bsearch_on_range p (a, m, rest)
end
end
fun array_bsearch ['a] {m : Nat} {len : Nat} (p : 'a -- $m --> order) (a : array 'a {len}) return option (nat_less_than {len}) =
case le (length a, #0) of
Le => NONE
| Gt =>
let
val res = array_bsearch_on_range p (a, #0, length a)
in
case res of
NatLT n =>
case p (sub (a, n)) of
OrdEq => SOME res
| _ => NONE
end
end