Skip to content

Commit c22dbaa

Browse files
committed
types: drop BoundMutex and instead use references into the type context slab
This completes the transition to using type contexts to keep track of (and allocate/mass-deallocate) type bounds :). There are three major improvements in this changeset: * We no longer leak memory when infinite type bounds are constructed. * It is no longer possible to create distinct programs where the variables are mixed up. (Ok, you can do this still, but you have to explicitly use the same type context for both programs, which is an obvious bug.) * Unification and binding happen atomically, so if you are doing type inference across multiple threads, crosstalk won't happen between them.
1 parent 2e24c49 commit c22dbaa

File tree

2 files changed

+39
-67
lines changed

2 files changed

+39
-67
lines changed

src/types/context.rs

+39-28
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,6 @@ use std::sync::{Arc, Mutex, MutexGuard};
1919

2020
use crate::dag::{Dag, DagLike};
2121

22-
use super::bound_mutex::BoundMutex;
2322
use super::{Bound, CompleteBound, Error, Final, Type};
2423

2524
/// Type inference context, or handle to a context.
@@ -60,9 +59,13 @@ impl Context {
6059

6160
/// Helper function to allocate a bound and return a reference to it.
6261
fn alloc_bound(&self, bound: Bound) -> BoundRef {
62+
let mut lock = self.lock();
63+
lock.slab.push(bound);
64+
let index = lock.slab.len() - 1;
65+
6366
BoundRef {
6467
context: Arc::as_ptr(&self.slab),
65-
index: Arc::new(BoundMutex::new(bound)),
68+
index,
6669
}
6770
}
6871

@@ -132,7 +135,8 @@ impl Context {
132135
/// Panics if passed a `BoundRef` that was not allocated by this context.
133136
pub fn get(&self, bound: &BoundRef) -> Bound {
134137
bound.assert_matches_context(self);
135-
bound.index.get().shallow_clone()
138+
let lock = self.lock();
139+
lock.slab[bound.index].shallow_clone()
136140
}
137141

138142
/// Reassigns a bound to a different bound.
@@ -146,8 +150,8 @@ impl Context {
146150
///
147151
/// Also panics if passed a `BoundRef` that was not allocated by this context.
148152
pub fn reassign_non_complete(&self, bound: BoundRef, new: Bound) {
149-
bound.assert_matches_context(self);
150-
bound.index.set(new)
153+
let mut lock = self.lock();
154+
lock.reassign_non_complete(bound, new);
151155
}
152156

153157
/// Binds the type to a given bound. If this fails, attach the provided
@@ -156,15 +160,15 @@ impl Context {
156160
/// Fails if the type has an existing incompatible bound.
157161
pub fn bind(&self, existing: &Type, new: Bound, hint: &'static str) -> Result<(), Error> {
158162
let existing_root = existing.bound.root();
159-
let lock = self.lock();
163+
let mut lock = self.lock();
160164
lock.bind(existing_root, new, hint)
161165
}
162166

163167
/// Unify the type with another one.
164168
///
165169
/// Fails if the bounds on the two types are incompatible
166170
pub fn unify(&self, ty1: &Type, ty2: &Type, hint: &'static str) -> Result<(), Error> {
167-
let lock = self.lock();
171+
let mut lock = self.lock();
168172
lock.unify(ty1, ty2, hint)
169173
}
170174

@@ -179,9 +183,7 @@ impl Context {
179183
#[derive(Debug, Clone)]
180184
pub struct BoundRef {
181185
context: *const Mutex<Vec<Bound>>,
182-
// Will become an index into the context in a latter commit, but for
183-
// now we set it to an Arc<BoundMutex> to preserve semantics.
184-
index: Arc<BoundMutex>,
186+
index: usize,
185187
}
186188

187189
impl BoundRef {
@@ -199,7 +201,7 @@ impl BoundRef {
199201
pub fn occurs_check_id(&self) -> OccursCheckId {
200202
OccursCheckId {
201203
context: self.context,
202-
index: Arc::as_ptr(&self.index),
204+
index: self.index,
203205
}
204206
}
205207
}
@@ -210,13 +212,13 @@ impl super::PointerLike for BoundRef {
210212
self.context, other.context,
211213
"tried to compare two bounds from different inference contexts"
212214
);
213-
Arc::ptr_eq(&self.index, &other.index)
215+
self.index == other.index
214216
}
215217

216218
fn shallow_clone(&self) -> Self {
217219
BoundRef {
218220
context: self.context,
219-
index: Arc::clone(&self.index),
221+
index: self.index,
220222
}
221223
}
222224
}
@@ -240,9 +242,7 @@ impl<'ctx> DagLike for (&'ctx Context, BoundRef) {
240242
#[derive(Copy, Clone, PartialEq, Eq, Debug, Hash)]
241243
pub struct OccursCheckId {
242244
context: *const Mutex<Vec<Bound>>,
243-
// Will become an index into the context in a latter commit, but for
244-
// now we set it to an Arc<BoundMutex> to preserve semantics.
245-
index: *const BoundMutex,
245+
index: usize,
246246
}
247247

248248
/// Structure representing an inference context with its slab allocator mutex locked.
@@ -254,17 +254,25 @@ struct LockedContext<'ctx> {
254254
}
255255

256256
impl<'ctx> LockedContext<'ctx> {
257+
fn reassign_non_complete(&mut self, bound: BoundRef, new: Bound) {
258+
assert!(
259+
!matches!(self.slab[bound.index], Bound::Complete(..)),
260+
"tried to modify finalized type",
261+
);
262+
self.slab[bound.index] = new;
263+
}
264+
257265
/// Unify the type with another one.
258266
///
259267
/// Fails if the bounds on the two types are incompatible
260-
fn unify(&self, existing: &Type, other: &Type, hint: &'static str) -> Result<(), Error> {
268+
fn unify(&mut self, existing: &Type, other: &Type, hint: &'static str) -> Result<(), Error> {
261269
existing.bound.unify(&other.bound, |x_bound, y_bound| {
262-
self.bind(x_bound, y_bound.index.get(), hint)
270+
self.bind(x_bound, self.slab[y_bound.index].shallow_clone(), hint)
263271
})
264272
}
265273

266-
fn bind(&self, existing: BoundRef, new: Bound, hint: &'static str) -> Result<(), Error> {
267-
let existing_bound = existing.index.get();
274+
fn bind(&mut self, existing: BoundRef, new: Bound, hint: &'static str) -> Result<(), Error> {
275+
let existing_bound = self.slab[existing.index].shallow_clone();
268276
let bind_error = || Error::Bind {
269277
existing_bound: existing_bound.shallow_clone(),
270278
new_bound: new.shallow_clone(),
@@ -277,7 +285,7 @@ impl<'ctx> LockedContext<'ctx> {
277285
// Free types are simply dropped and replaced by the new bound
278286
(Bound::Free(_), _) => {
279287
// Free means non-finalized, so set() is ok.
280-
existing.index.set(new);
288+
self.reassign_non_complete(existing, new);
281289
Ok(())
282290
}
283291
// Binding complete->complete shouldn't ever happen, but if so, we just
@@ -319,14 +327,17 @@ impl<'ctx> LockedContext<'ctx> {
319327
//
320328
// It also gives the user access to more information about the type,
321329
// prior to finalization.
322-
if let (Some(data1), Some(data2)) = (y1.final_data(), y2.final_data()) {
323-
existing
324-
.index
325-
.set(Bound::Complete(if let Bound::Sum(..) = existing_bound {
326-
Final::sum(data1, data2)
330+
let y1_bound = &self.slab[y1.bound.root().index];
331+
let y2_bound = &self.slab[y2.bound.root().index];
332+
if let (Bound::Complete(data1), Bound::Complete(data2)) = (y1_bound, y2_bound) {
333+
self.reassign_non_complete(
334+
existing,
335+
Bound::Complete(if let Bound::Sum(..) = existing_bound {
336+
Final::sum(Arc::clone(data1), Arc::clone(data2))
327337
} else {
328-
Final::product(data1, data2)
329-
}));
338+
Final::product(Arc::clone(data1), Arc::clone(data2))
339+
}),
340+
);
330341
}
331342
Ok(())
332343
}

src/types/mod.rs

-39
Original file line numberDiff line numberDiff line change
@@ -148,45 +148,6 @@ impl fmt::Display for Error {
148148

149149
impl std::error::Error for Error {}
150150

151-
mod bound_mutex {
152-
use super::Bound;
153-
use std::fmt;
154-
use std::sync::Mutex;
155-
156-
/// Source or target type of a Simplicity expression
157-
pub struct BoundMutex {
158-
/// The type's status according to the union-bound algorithm.
159-
inner: Mutex<Bound>,
160-
}
161-
162-
impl fmt::Debug for BoundMutex {
163-
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
164-
self.get().fmt(f)
165-
}
166-
}
167-
168-
impl BoundMutex {
169-
pub fn new(bound: Bound) -> Self {
170-
BoundMutex {
171-
inner: Mutex::new(bound),
172-
}
173-
}
174-
175-
pub fn get(&self) -> Bound {
176-
self.inner.lock().unwrap().shallow_clone()
177-
}
178-
179-
pub fn set(&self, new: Bound) {
180-
let mut lock = self.inner.lock().unwrap();
181-
assert!(
182-
!matches!(*lock, Bound::Complete(..)),
183-
"tried to modify finalized type",
184-
);
185-
*lock = new;
186-
}
187-
}
188-
}
189-
190151
/// The state of a [`Type`] based on all constraints currently imposed on it.
191152
#[derive(Clone, Debug)]
192153
pub enum Bound {

0 commit comments

Comments
 (0)