-
Notifications
You must be signed in to change notification settings - Fork 54
/
Copy path04_binary_search.rs
52 lines (46 loc) · 1.36 KB
/
04_binary_search.rs
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
47
48
49
50
51
52
extern crate creusot_contracts;
use creusot_contracts::{
logic::{Int, Seq},
*,
};
#[predicate]
fn sorted_range(s: Seq<u32>, l: Int, u: Int) -> bool {
pearlite! {
forall<i : Int, j : Int> l <= i && i < j && j < u ==> s[i] <= s[j]
}
}
#[predicate]
fn sorted(s: Seq<u32>) -> bool {
sorted_range(s, 0, s.len())
}
#[requires([email protected]() <= usize::MAX@)]
#[requires(sorted(arr@))]
#[ensures(forall<x:usize> result == Ok(x) ==> arr[x@] == elem)]
#[ensures(forall<x:usize> result == Err(x) ==>
forall<i:usize> i < x ==> arr[i@] <= elem)]
#[ensures(forall<x:usize> result == Err(x) ==>
forall<i:usize> x < i && i@ < [email protected]() ==> elem < arr[i@])]
pub fn binary_search(arr: &Vec<u32>, elem: u32) -> Result<usize, usize> {
if arr.len() == 0 {
return Err(0);
}
let mut size = arr.len();
let mut base = 0;
#[invariant(0 < size@ && size@ + base@ <= [email protected]())]
#[invariant(forall<i : usize> i < base ==> arr[i@] <= elem)]
#[invariant(forall<i : usize> base@ + size@ < i@ && i@ < [email protected]() ==> elem < arr[i@])]
while size > 1 {
let half = size / 2;
let mid = base + half;
base = if arr[mid] > elem { base } else { mid };
size -= half;
}
let cmp = arr[base];
if cmp == elem {
Ok(base)
} else if cmp < elem {
Err(base + 1)
} else {
Err(base)
}
}