1use vstd::prelude::*;
2
3use vstd::raw_ptr::MemContents;
4use vstd::simple_pptr::{self, PPtr};
5use vstd::std_specs::convert::{FromSpec, FromSpecImpl};
6
7use core::marker::PhantomData;
8
9verus! {
10
11pub trait Repr<R: Sized>: Sized {
12 type Perm;
14
15 spec fn wf(r: R, perm: Self::Perm) -> bool;
16
17 spec fn to_repr_spec(self, perm: Self::Perm) -> (R, Self::Perm);
18
19 fn to_repr(self, Tracked(perm): Tracked<&mut Self::Perm>) -> (res: R)
20 ensures
21 res == self.to_repr_spec(*old(perm)).0,
22 *final(perm) == self.to_repr_spec(*old(perm)).1,
23 ;
24
25 spec fn from_repr_spec(r: R, perm: Self::Perm) -> Self;
26
27 fn from_repr(r: R, Tracked(perm): Tracked<&Self::Perm>) -> (res: Self)
28 requires
29 Self::wf(r, *perm),
30 ensures
31 res == Self::from_repr_spec(r, *perm),
32 ;
33
34 fn from_borrowed<'a>(r: &'a R, Tracked(perm): Tracked<&'a Self::Perm>) -> (res: &'a Self)
35 requires
36 Self::wf(*r, *perm),
37 ensures
38 *res == Self::from_repr_spec(*r, *perm),
39 ;
40
41 proof fn from_to_repr(self, perm: Self::Perm)
42 ensures
43 Self::from_repr_spec(self.to_repr_spec(perm).0, self.to_repr_spec(perm).1) == self,
44 ;
45
46 proof fn to_from_repr(r: R, perm: Self::Perm)
47 requires
48 Self::wf(r, perm),
49 ensures
50 Self::from_repr_spec(r, perm).to_repr_spec(perm) == (r, perm),
51 ;
52
53 proof fn to_repr_wf(self, perm: Self::Perm)
54 ensures
55 Self::wf(self.to_repr_spec(perm).0, self.to_repr_spec(perm).1),
56 ;
57}
58
59pub struct ReprPtr<R, T: Repr<R>> {
62 pub ptr: PPtr<R>,
63 pub _T: PhantomData<T>,
64}
65
66impl<R, T: Repr<R>> Clone for ReprPtr<R, T> {
67 fn clone(&self) -> Self {
68 Self { ptr: self.ptr, _T: PhantomData }
69 }
70}
71
72impl<R, T: Repr<R>> Copy for ReprPtr<R, T> {
73
74}
75
76impl<R, T: Repr<R>> FromSpecImpl<PPtr<R>> for ReprPtr<R, T> {
77 open spec fn obeys_from_spec() -> bool {
78 true
79 }
80
81 open spec fn from_spec(ptr: PPtr<R>) -> Self {
82 Self { ptr: ptr, _T: PhantomData }
83 }
84}
85
86impl<R, T: Repr<R>> From<PPtr<R>> for ReprPtr<R, T> {
87 fn from(ptr: PPtr<R>) -> Self {
88 Self { ptr: ptr, _T: PhantomData }
89 }
90}
91
92impl<R, T: Repr<R>> FromSpecImpl<ReprPtr<R, T>> for PPtr<R> {
93 open spec fn obeys_from_spec() -> bool {
94 true
95 }
96
97 open spec fn from_spec(ptr: ReprPtr<R, T>) -> Self {
98 ptr.ptr
99 }
100}
101
102impl<R, T: Repr<R>> From<ReprPtr<R, T>> for PPtr<R> {
103 fn from(ptr: ReprPtr<R, T>) -> Self {
104 ptr.ptr
105 }
106}
107
108impl<R, T: Repr<R>> ReprPtr<R, T> {
109 pub open spec fn new_spec(ptr: PPtr<R>) -> Self {
110 Self { ptr: ptr, _T: PhantomData }
111 }
112
113 #[verifier::external_body]
114 pub fn new_borrowed<'a>(ptr: &'a PPtr<R>) -> (res: &'a Self)
115 ensures
116 *res == Self::new_spec(*ptr),
117 {
118 unimplemented!()
119 }
120
121 pub fn from_pptr(ptr: PPtr<R>) -> (res: Self)
122 ensures
123 res == Self::new_spec(ptr),
124 res.addr() == ptr.addr(),
125 res.ptr == ptr,
126 {
127 Self { ptr: ptr, _T: PhantomData }
128 }
129
130 pub open spec fn to_pptr(self) -> PPtr<R> {
131 self.ptr
132 }
133
134 pub open spec fn addr_spec(self) -> usize {
135 self.ptr.addr()
136 }
137
138 #[verifier::when_used_as_spec(addr_spec)]
139 pub fn addr(self) -> (u: usize)
140 returns
141 self.addr_spec(),
142 {
143 self.ptr.addr()
144 }
145
146 pub exec fn take(self, Tracked(perm): Tracked<&mut PointsTo<R, T>>) -> (v: T)
147 requires
148 old(perm).pptr() == self,
149 old(perm).is_init(),
150 old(perm).wf(&old(perm).inner_perms),
151 ensures
152 final(perm).pptr() == old(perm).pptr(),
153 final(perm).mem_contents() == MemContents::Uninit::<T>,
154 v == old(perm).value(),
155 {
156 proof {
157 T::from_to_repr(perm.value(), perm.inner_perms);
158 }
159 T::from_repr(self.ptr.take(Tracked(&mut perm.points_to)), Tracked(&perm.inner_perms))
160 }
161
162 pub exec fn put(self, Tracked(perm): Tracked<&mut PointsTo<R, T>>, v: T)
163 requires
164 old(perm).pptr() == self,
165 old(perm).mem_contents() == MemContents::Uninit::<T>,
166 ensures
167 final(perm).pptr() == old(perm).pptr(),
168 final(perm).mem_contents() == MemContents::Init(v),
169 final(perm).wf(&final(perm).inner_perms),
170 {
171 proof {
172 v.from_to_repr(perm.inner_perms);
173 v.to_repr_wf(perm.inner_perms);
174 }
175 self.ptr.put(Tracked(&mut perm.points_to), v.to_repr(Tracked(&mut perm.inner_perms)))
176 }
177
178 pub exec fn borrow<'a>(self, Tracked(perm): Tracked<&'a PointsTo<R, T>>) -> (v: &'a T)
179 requires
180 perm.pptr() == self,
181 perm.is_init(),
182 perm.wf(&perm.inner_perms),
183 ensures
184 *v === perm.value(),
185 {
186 T::from_borrowed(self.ptr.borrow(Tracked(&perm.points_to)), Tracked(&perm.inner_perms))
187 }
188
189 #[verifier::external_body]
197 pub exec fn borrow_mut<'a>(self, Tracked(perm): Tracked<&'a mut PointsTo<R, T>>) -> (v:
198 &'a mut T)
199 requires
200 old(perm).pptr() == self,
201 old(perm).is_init(),
202 old(perm).wf(&old(perm).inner_perms),
203 ensures
204 *v == old(perm).value(),
205 final(perm).pptr() == old(perm).pptr(),
206 final(perm).is_init(),
207 final(perm).wf(&final(perm).inner_perms),
208 {
209 unsafe { &mut *(self.ptr.addr() as *mut T) }
212 }
213}
214
215#[verifier::accept_recursive_types(T)]
216pub tracked struct PointsTo<R, T: Repr<R>> {
217 pub tracked points_to: simple_pptr::PointsTo<R>,
218 pub tracked inner_perms: T::Perm,
219 pub _T: PhantomData<T>,
220}
221
222impl<R, T: Repr<R>> PointsTo<R, T> {
223 pub open spec fn new_spec(points_to: simple_pptr::PointsTo<R>, inner_perms: T::Perm) -> Self {
224 Self { points_to, inner_perms, _T: PhantomData }
225 }
226
227 pub proof fn new(
228 tracked points_to: simple_pptr::PointsTo<R>,
229 tracked inner_perms: T::Perm,
230 ) -> tracked Self {
231 Self { points_to: points_to, inner_perms, _T: PhantomData }
232 }
233
234 pub open spec fn wf(self, perm: &T::Perm) -> bool {
235 &&& T::wf(self.points_to.value(), *perm)
236 }
237
238 pub open spec fn addr(self) -> usize {
239 self.points_to.addr()
240 }
241
242 pub open spec fn mem_contents(self) -> MemContents<T> {
243 match self.points_to.mem_contents() {
244 MemContents::<R>::Uninit => MemContents::<T>::Uninit,
245 MemContents::<R>::Init(r) => MemContents::<T>::Init(
246 T::from_repr_spec(r, self.inner_perms),
247 ),
248 }
249 }
250
251 pub open spec fn is_init(self) -> bool {
252 self.mem_contents().is_init()
253 }
254
255 pub open spec fn is_uninit(self) -> bool {
256 self.mem_contents().is_uninit()
257 }
258
259 pub open spec fn value(self) -> T
260 recommends
261 self.is_init(),
262 {
263 self.mem_contents().value()
264 }
265
266 pub open spec fn pptr(self) -> ReprPtr<R, T> {
267 ReprPtr { ptr: self.points_to.pptr(), _T: PhantomData }
268 }
269
270 pub broadcast proof fn pptr_implies_addr(&self)
271 ensures
272 self.addr() == #[trigger] self.pptr().addr(),
273 {
274 }
275}
276
277impl<R, T: Repr<R>> From<PointsTo<R, T>> for vstd::simple_pptr::PointsTo<R> {
278 #[verifier::external_body]
279 fn from(ptr: PointsTo<R, T>) -> Self {
280 ptr.points_to
281 }
282}
283
284}