Skip to content

Commit 46c8720

Browse files
committed
f make Bound private, use Type as non-recursive public type
After the previous commit, we still had a memory leak in the case of cyclic types, but for a very dumb reason: the Type structure was carrying a Context Arc, and Type was recursive. Instead, introduce a new private TypeInner type which is recursive and does not have direct access to the Context. Make Bound private as well, since it is mutually recursive with Context. I am not thrilled with this commit and may revisit it, but it fixes the leak and lets me get on with fuzzing before I go to bed.
1 parent 0fd9250 commit 46c8720

File tree

3 files changed

+166
-139
lines changed

3 files changed

+166
-139
lines changed

src/types/arrow.rs

+15-13
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ use crate::node::{
1818
CoreConstructible, DisconnectConstructible, JetConstructible, NoDisconnect,
1919
WitnessConstructible,
2020
};
21-
use crate::types::{Bound, Context, Error, Final, Type};
21+
use crate::types::{Context, Error, Final, Type};
2222
use crate::{jet::Jet, Value};
2323

2424
use super::variable::new_name;
@@ -123,17 +123,19 @@ impl Arrow {
123123

124124
let target = Type::free(&ctx, String::new());
125125
if let Some(lchild_arrow) = lchild_arrow {
126-
ctx.bind(
126+
ctx.bind_product(
127127
&lchild_arrow.source,
128-
Bound::Product(a, c.shallow_clone()),
128+
&a,
129+
&c,
129130
"case combinator: left source = A × C",
130131
)?;
131132
ctx.unify(&target, &lchild_arrow.target, "").unwrap();
132133
}
133134
if let Some(rchild_arrow) = rchild_arrow {
134-
ctx.bind(
135+
ctx.bind_product(
135136
&rchild_arrow.source,
136-
Bound::Product(b, c),
137+
&b,
138+
&c,
137139
"case combinator: left source = B × C",
138140
)?;
139141
ctx.unify(
@@ -162,21 +164,21 @@ impl Arrow {
162164
let c = rchild_arrow.source.shallow_clone();
163165
let d = rchild_arrow.target.shallow_clone();
164166

165-
let prod_256_a = Bound::Product(Type::two_two_n(ctx, 8), a.shallow_clone());
166-
let prod_b_c = Bound::Product(b.shallow_clone(), c);
167-
let prod_b_d = Type::product(ctx, b, d);
168-
169-
ctx.bind(
167+
ctx.bind_product(
170168
&lchild_arrow.source,
171-
prod_256_a,
169+
&Type::two_two_n(ctx, 8),
170+
&a,
172171
"disconnect combinator: left source = 2^256 × A",
173172
)?;
174-
ctx.bind(
173+
ctx.bind_product(
175174
&lchild_arrow.target,
176-
prod_b_c,
175+
&b,
176+
&c,
177177
"disconnect combinator: left target = B × C",
178178
)?;
179179

180+
let prod_b_d = Type::product(ctx, b, d);
181+
180182
Ok(Arrow {
181183
source: a,
182184
target: prod_b_d,

src/types/context.rs

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

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

22-
use super::{Bound, CompleteBound, Error, Final, Type};
22+
use super::{Bound, CompleteBound, Error, Final, Type, TypeInner};
2323

2424
/// Type inference context, or handle to a context.
2525
///
@@ -60,13 +60,7 @@ impl Context {
6060
/// Helper function to allocate a bound and return a reference to it.
6161
fn alloc_bound(&self, bound: Bound) -> BoundRef {
6262
let mut lock = self.lock();
63-
lock.slab.push(bound);
64-
let index = lock.slab.len() - 1;
65-
66-
BoundRef {
67-
context: Arc::as_ptr(&self.slab),
68-
index,
69-
}
63+
lock.alloc_bound(Arc::as_ptr(&self.slab), bound)
7064
}
7165

7266
/// Allocate a new free type bound, and return a reference to it.
@@ -90,10 +84,24 @@ impl Context {
9084
///
9185
/// Panics if either of the child types are from a different inference context.
9286
pub fn alloc_sum(&self, left: Type, right: Type) -> BoundRef {
93-
left.bound.root().assert_matches_context(self);
94-
right.bound.root().assert_matches_context(self);
87+
assert_eq!(
88+
left.ctx, *self,
89+
"left type did not match inference context of sum"
90+
);
91+
assert_eq!(
92+
right.ctx, *self,
93+
"right type did not match inference context of sum"
94+
);
9595

96-
self.alloc_bound(Bound::sum(left, right))
96+
let mut lock = self.lock();
97+
if let Some((data1, data2)) = lock.complete_pair_data(&left.inner, &right.inner) {
98+
lock.alloc_bound(
99+
Arc::as_ptr(&self.slab),
100+
Bound::Complete(Final::sum(data1, data2)),
101+
)
102+
} else {
103+
lock.alloc_bound(Arc::as_ptr(&self.slab), Bound::Sum(left.inner, right.inner))
104+
}
97105
}
98106

99107
/// Allocate a new product-type bound, and return a reference to it.
@@ -102,10 +110,27 @@ impl Context {
102110
///
103111
/// Panics if either of the child types are from a different inference context.
104112
pub fn alloc_product(&self, left: Type, right: Type) -> BoundRef {
105-
left.bound.root().assert_matches_context(self);
106-
right.bound.root().assert_matches_context(self);
113+
assert_eq!(
114+
left.ctx, *self,
115+
"left type did not match inference context of product"
116+
);
117+
assert_eq!(
118+
right.ctx, *self,
119+
"right type did not match inference context of product"
120+
);
107121

108-
self.alloc_bound(Bound::product(left, right))
122+
let mut lock = self.lock();
123+
if let Some((data1, data2)) = lock.complete_pair_data(&left.inner, &right.inner) {
124+
lock.alloc_bound(
125+
Arc::as_ptr(&self.slab),
126+
Bound::Complete(Final::product(data1, data2)),
127+
)
128+
} else {
129+
lock.alloc_bound(
130+
Arc::as_ptr(&self.slab),
131+
Bound::Product(left.inner, right.inner),
132+
)
133+
}
109134
}
110135

111136
/// Creates a new handle to the context.
@@ -133,7 +158,7 @@ impl Context {
133158
/// # Panics
134159
///
135160
/// Panics if passed a `BoundRef` that was not allocated by this context.
136-
pub fn get(&self, bound: &BoundRef) -> Bound {
161+
pub(super) fn get(&self, bound: &BoundRef) -> Bound {
137162
bound.assert_matches_context(self);
138163
let lock = self.lock();
139164
lock.slab[bound.index].shallow_clone()
@@ -150,34 +175,54 @@ impl Context {
150175
/// probably a bug.
151176
///
152177
/// Also panics if passed a `BoundRef` that was not allocated by this context.
153-
pub fn reassign_non_complete(&self, bound: BoundRef, new: Bound) {
178+
pub(super) fn reassign_non_complete(&self, bound: BoundRef, new: Bound) {
154179
let mut lock = self.lock();
155180
lock.reassign_non_complete(bound, new);
156181
}
157182

158-
/// Binds the type to a given bound. If this fails, attach the provided
159-
/// hint to the error.
183+
/// Binds the type to a product bound formed by the two inner types. If this
184+
/// fails, attach the provided hint to the error.
160185
///
161186
/// Fails if the type has an existing incompatible bound.
162-
pub fn bind(&self, existing: &Type, new: Bound, hint: &'static str) -> Result<(), Error> {
163-
let existing_root = existing.bound.root();
187+
pub fn bind_product(
188+
&self,
189+
existing: &Type,
190+
prod_l: &Type,
191+
prod_r: &Type,
192+
hint: &'static str,
193+
) -> Result<(), Error> {
194+
assert_eq!(existing.ctx, *self);
195+
assert_eq!(prod_l.ctx, *self);
196+
assert_eq!(prod_r.ctx, *self);
197+
198+
let existing_root = existing.inner.bound.root();
199+
let new_bound = Bound::Product(prod_l.inner.shallow_clone(), prod_r.inner.shallow_clone());
200+
164201
let mut lock = self.lock();
165-
lock.bind(existing_root, new).map_err(|e| Error::Bind {
166-
existing_bound: e.existing,
167-
new_bound: e.new,
168-
hint,
202+
lock.bind(existing_root, new_bound).map_err(|e| {
203+
let new_bound = lock.alloc_bound(Arc::as_ptr(&self.slab), e.new);
204+
Error::Bind {
205+
existing_bound: Type::wrap_bound(self, e.existing),
206+
new_bound: Type::wrap_bound(self, new_bound),
207+
hint,
208+
}
169209
})
170210
}
171211

172212
/// Unify the type with another one.
173213
///
174214
/// Fails if the bounds on the two types are incompatible
175215
pub fn unify(&self, ty1: &Type, ty2: &Type, hint: &'static str) -> Result<(), Error> {
216+
assert_eq!(ty1.ctx, *self);
217+
assert_eq!(ty2.ctx, *self);
176218
let mut lock = self.lock();
177-
lock.unify(ty1, ty2).map_err(|e| Error::Bind {
178-
existing_bound: e.existing,
179-
new_bound: e.new,
180-
hint,
219+
lock.unify(&ty1.inner, &ty2.inner).map_err(|e| {
220+
let new_bound = lock.alloc_bound(Arc::as_ptr(&self.slab), e.new);
221+
Error::Bind {
222+
existing_bound: Type::wrap_bound(self, e.existing),
223+
new_bound: Type::wrap_bound(self, new_bound),
224+
hint,
225+
}
181226
})
182227
}
183228

@@ -257,7 +302,7 @@ pub struct OccursCheckId {
257302
}
258303

259304
struct BindError {
260-
existing: Bound,
305+
existing: BoundRef,
261306
new: Bound,
262307
}
263308

@@ -270,6 +315,16 @@ struct LockedContext<'ctx> {
270315
}
271316

272317
impl<'ctx> LockedContext<'ctx> {
318+
fn alloc_bound(&mut self, ctx_ptr: *const Mutex<Vec<Bound>>, bound: Bound) -> BoundRef {
319+
self.slab.push(bound);
320+
let index = self.slab.len() - 1;
321+
322+
BoundRef {
323+
context: ctx_ptr,
324+
index,
325+
}
326+
}
327+
273328
fn reassign_non_complete(&mut self, bound: BoundRef, new: Bound) {
274329
assert!(
275330
!matches!(self.slab[bound.index], Bound::Complete(..)),
@@ -278,10 +333,29 @@ impl<'ctx> LockedContext<'ctx> {
278333
self.slab[bound.index] = new;
279334
}
280335

336+
/// It is a common situation that we are pairing two types, and in the
337+
/// case that they are both complete, we want to pair the complete types.
338+
///
339+
/// This method deals with all the annoying/complicated member variable
340+
/// paths to get the actual complete data out.
341+
fn complete_pair_data(
342+
&self,
343+
inn1: &TypeInner,
344+
inn2: &TypeInner,
345+
) -> Option<(Arc<Final>, Arc<Final>)> {
346+
let bound1 = &self.slab[inn1.bound.root().index];
347+
let bound2 = &self.slab[inn2.bound.root().index];
348+
if let (Bound::Complete(ref data1), Bound::Complete(ref data2)) = (bound1, bound2) {
349+
Some((Arc::clone(data1), Arc::clone(data2)))
350+
} else {
351+
None
352+
}
353+
}
354+
281355
/// Unify the type with another one.
282356
///
283357
/// Fails if the bounds on the two types are incompatible
284-
fn unify(&mut self, existing: &Type, other: &Type) -> Result<(), BindError> {
358+
fn unify(&mut self, existing: &TypeInner, other: &TypeInner) -> Result<(), BindError> {
285359
existing.bound.unify(&other.bound, |x_bound, y_bound| {
286360
self.bind(x_bound, self.slab[y_bound.index].shallow_clone())
287361
})
@@ -290,7 +364,7 @@ impl<'ctx> LockedContext<'ctx> {
290364
fn bind(&mut self, existing: BoundRef, new: Bound) -> Result<(), BindError> {
291365
let existing_bound = self.slab[existing.index].shallow_clone();
292366
let bind_error = || BindError {
293-
existing: existing_bound.shallow_clone(),
367+
existing: existing.clone(),
294368
new: new.shallow_clone(),
295369
};
296370

@@ -342,24 +416,19 @@ impl<'ctx> LockedContext<'ctx> {
342416
//
343417
// It also gives the user access to more information about the type,
344418
// prior to finalization.
345-
let y1_bound = &self.slab[y1.bound.root().index];
346-
let y2_bound = &self.slab[y2.bound.root().index];
347-
if let (Bound::Complete(data1), Bound::Complete(data2)) = (y1_bound, y2_bound) {
419+
if let Some((data1, data2)) = self.complete_pair_data(y1, y2) {
348420
self.reassign_non_complete(
349421
existing,
350422
Bound::Complete(if let Bound::Sum(..) = existing_bound {
351-
Final::sum(Arc::clone(data1), Arc::clone(data2))
423+
Final::sum(data1, data2)
352424
} else {
353-
Final::product(Arc::clone(data1), Arc::clone(data2))
425+
Final::product(data1, data2)
354426
}),
355427
);
356428
}
357429
Ok(())
358430
}
359-
(x, y) => Err(BindError {
360-
existing: x.shallow_clone(),
361-
new: y.shallow_clone(),
362-
}),
431+
(_, _) => Err(bind_error()),
363432
}
364433
}
365434
}

0 commit comments

Comments
 (0)