15
15
//!
16
16
17
17
use std:: fmt;
18
- use std:: sync:: { Arc , Mutex } ;
18
+ use std:: sync:: { Arc , Mutex , MutexGuard } ;
19
19
20
20
use crate :: dag:: { Dag , DagLike } ;
21
21
22
22
use super :: bound_mutex:: BoundMutex ;
23
- use super :: { Bound , Error , Final , Type } ;
23
+ use super :: { Bound , CompleteBound , Error , Final , Type } ;
24
24
25
25
/// Type inference context, or handle to a context.
26
26
///
@@ -155,14 +155,24 @@ impl Context {
155
155
///
156
156
/// Fails if the type has an existing incompatible bound.
157
157
pub fn bind ( & self , existing : & Type , new : Bound , hint : & ' static str ) -> Result < ( ) , Error > {
158
- existing. bind ( new, hint)
158
+ let existing_root = existing. bound . root ( ) ;
159
+ let lock = self . lock ( ) ;
160
+ lock. bind ( existing_root, new, hint)
159
161
}
160
162
161
163
/// Unify the type with another one.
162
164
///
163
165
/// Fails if the bounds on the two types are incompatible
164
166
pub fn unify ( & self , ty1 : & Type , ty2 : & Type , hint : & ' static str ) -> Result < ( ) , Error > {
165
- ty1. unify ( ty2, hint)
167
+ let lock = self . lock ( ) ;
168
+ lock. unify ( ty1, ty2, hint)
169
+ }
170
+
171
+ /// Locks the underlying slab mutex.
172
+ fn lock ( & self ) -> LockedContext {
173
+ LockedContext {
174
+ slab : self . slab . lock ( ) . unwrap ( ) ,
175
+ }
166
176
}
167
177
}
168
178
@@ -183,10 +193,6 @@ impl BoundRef {
183
193
) ;
184
194
}
185
195
186
- pub fn bind ( & self , bound : Bound , hint : & ' static str ) -> Result < ( ) , Error > {
187
- self . index . bind ( bound, hint)
188
- }
189
-
190
196
/// Creates an "occurs-check ID" which is just a copy of the [`BoundRef`]
191
197
/// with `PartialEq` and `Eq` implemented in terms of underlying pointer
192
198
/// equality.
@@ -238,3 +244,97 @@ pub struct OccursCheckId {
238
244
// now we set it to an Arc<BoundMutex> to preserve semantics.
239
245
index : * const BoundMutex ,
240
246
}
247
+
248
+ /// Structure representing an inference context with its slab allocator mutex locked.
249
+ ///
250
+ /// This type is never exposed outside of this module and should only exist
251
+ /// ephemerally within function calls into this module.
252
+ struct LockedContext < ' ctx > {
253
+ slab : MutexGuard < ' ctx , Vec < Bound > > ,
254
+ }
255
+
256
+ impl < ' ctx > LockedContext < ' ctx > {
257
+ /// Unify the type with another one.
258
+ ///
259
+ /// Fails if the bounds on the two types are incompatible
260
+ fn unify ( & self , existing : & Type , other : & Type , hint : & ' static str ) -> Result < ( ) , Error > {
261
+ existing. bound . unify ( & other. bound , |x_bound, y_bound| {
262
+ self . bind ( x_bound, y_bound. index . get ( ) , hint)
263
+ } )
264
+ }
265
+
266
+ fn bind ( & self , existing : BoundRef , new : Bound , hint : & ' static str ) -> Result < ( ) , Error > {
267
+ let existing_bound = existing. index . get ( ) ;
268
+ let bind_error = || Error :: Bind {
269
+ existing_bound : existing_bound. shallow_clone ( ) ,
270
+ new_bound : new. shallow_clone ( ) ,
271
+ hint,
272
+ } ;
273
+
274
+ match ( & existing_bound, & new) {
275
+ // Binding a free type to anything is a no-op
276
+ ( _, Bound :: Free ( _) ) => Ok ( ( ) ) ,
277
+ // Free types are simply dropped and replaced by the new bound
278
+ ( Bound :: Free ( _) , _) => {
279
+ // Free means non-finalized, so set() is ok.
280
+ existing. index . set ( new) ;
281
+ Ok ( ( ) )
282
+ }
283
+ // Binding complete->complete shouldn't ever happen, but if so, we just
284
+ // compare the two types and return a pass/fail
285
+ ( Bound :: Complete ( ref existing_final) , Bound :: Complete ( ref new_final) ) => {
286
+ if existing_final == new_final {
287
+ Ok ( ( ) )
288
+ } else {
289
+ Err ( bind_error ( ) )
290
+ }
291
+ }
292
+ // Binding an incomplete to a complete type requires recursion.
293
+ ( Bound :: Complete ( complete) , incomplete) | ( incomplete, Bound :: Complete ( complete) ) => {
294
+ match ( complete. bound ( ) , incomplete) {
295
+ // A unit might match a Bound::Free(..) or a Bound::Complete(..),
296
+ // and both cases were handled above. So this is an error.
297
+ ( CompleteBound :: Unit , _) => Err ( bind_error ( ) ) ,
298
+ (
299
+ CompleteBound :: Product ( ref comp1, ref comp2) ,
300
+ Bound :: Product ( ref ty1, ref ty2) ,
301
+ )
302
+ | ( CompleteBound :: Sum ( ref comp1, ref comp2) , Bound :: Sum ( ref ty1, ref ty2) ) => {
303
+ let bound1 = ty1. bound . root ( ) ;
304
+ let bound2 = ty2. bound . root ( ) ;
305
+ self . bind ( bound1, Bound :: Complete ( Arc :: clone ( comp1) ) , hint) ?;
306
+ self . bind ( bound2, Bound :: Complete ( Arc :: clone ( comp2) ) , hint)
307
+ }
308
+ _ => Err ( bind_error ( ) ) ,
309
+ }
310
+ }
311
+ ( Bound :: Sum ( ref x1, ref x2) , Bound :: Sum ( ref y1, ref y2) )
312
+ | ( Bound :: Product ( ref x1, ref x2) , Bound :: Product ( ref y1, ref y2) ) => {
313
+ self . unify ( x1, y1, hint) ?;
314
+ self . unify ( x2, y2, hint) ?;
315
+ // This type was not complete, but it may be after unification, giving us
316
+ // an opportunity to finaliize it. We do this eagerly to make sure that
317
+ // "complete" (no free children) is always equivalent to "finalized" (the
318
+ // bound field having variant Bound::Complete(..)), even during inference.
319
+ //
320
+ // It also gives the user access to more information about the type,
321
+ // 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)
327
+ } else {
328
+ Final :: product ( data1, data2)
329
+ } ) ) ;
330
+ }
331
+ Ok ( ( ) )
332
+ }
333
+ ( x, y) => Err ( Error :: Bind {
334
+ existing_bound : x. shallow_clone ( ) ,
335
+ new_bound : y. shallow_clone ( ) ,
336
+ hint,
337
+ } ) ,
338
+ }
339
+ }
340
+ }
0 commit comments