@@ -19,7 +19,6 @@ use std::sync::{Arc, Mutex, MutexGuard};
19
19
20
20
use crate :: dag:: { Dag , DagLike } ;
21
21
22
- use super :: bound_mutex:: BoundMutex ;
23
22
use super :: { Bound , CompleteBound , Error , Final , Type } ;
24
23
25
24
/// Type inference context, or handle to a context.
@@ -60,9 +59,13 @@ impl Context {
60
59
61
60
/// Helper function to allocate a bound and return a reference to it.
62
61
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
+
63
66
BoundRef {
64
67
context : Arc :: as_ptr ( & self . slab ) ,
65
- index : Arc :: new ( BoundMutex :: new ( bound ) ) ,
68
+ index,
66
69
}
67
70
}
68
71
@@ -132,7 +135,8 @@ impl Context {
132
135
/// Panics if passed a `BoundRef` that was not allocated by this context.
133
136
pub fn get ( & self , bound : & BoundRef ) -> Bound {
134
137
bound. assert_matches_context ( self ) ;
135
- bound. index . get ( ) . shallow_clone ( )
138
+ let lock = self . lock ( ) ;
139
+ lock. slab [ bound. index ] . shallow_clone ( )
136
140
}
137
141
138
142
/// Reassigns a bound to a different bound.
@@ -146,8 +150,8 @@ impl Context {
146
150
///
147
151
/// Also panics if passed a `BoundRef` that was not allocated by this context.
148
152
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) ;
151
155
}
152
156
153
157
/// Binds the type to a given bound. If this fails, attach the provided
@@ -156,15 +160,15 @@ impl Context {
156
160
/// Fails if the type has an existing incompatible bound.
157
161
pub fn bind ( & self , existing : & Type , new : Bound , hint : & ' static str ) -> Result < ( ) , Error > {
158
162
let existing_root = existing. bound . root ( ) ;
159
- let lock = self . lock ( ) ;
163
+ let mut lock = self . lock ( ) ;
160
164
lock. bind ( existing_root, new, hint)
161
165
}
162
166
163
167
/// Unify the type with another one.
164
168
///
165
169
/// Fails if the bounds on the two types are incompatible
166
170
pub fn unify ( & self , ty1 : & Type , ty2 : & Type , hint : & ' static str ) -> Result < ( ) , Error > {
167
- let lock = self . lock ( ) ;
171
+ let mut lock = self . lock ( ) ;
168
172
lock. unify ( ty1, ty2, hint)
169
173
}
170
174
@@ -179,9 +183,7 @@ impl Context {
179
183
#[ derive( Debug , Clone ) ]
180
184
pub struct BoundRef {
181
185
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 ,
185
187
}
186
188
187
189
impl BoundRef {
@@ -199,7 +201,7 @@ impl BoundRef {
199
201
pub fn occurs_check_id ( & self ) -> OccursCheckId {
200
202
OccursCheckId {
201
203
context : self . context ,
202
- index : Arc :: as_ptr ( & self . index ) ,
204
+ index : self . index ,
203
205
}
204
206
}
205
207
}
@@ -210,13 +212,13 @@ impl super::PointerLike for BoundRef {
210
212
self . context, other. context,
211
213
"tried to compare two bounds from different inference contexts"
212
214
) ;
213
- Arc :: ptr_eq ( & self . index , & other. index )
215
+ self . index == other. index
214
216
}
215
217
216
218
fn shallow_clone ( & self ) -> Self {
217
219
BoundRef {
218
220
context : self . context ,
219
- index : Arc :: clone ( & self . index ) ,
221
+ index : self . index ,
220
222
}
221
223
}
222
224
}
@@ -240,9 +242,7 @@ impl<'ctx> DagLike for (&'ctx Context, BoundRef) {
240
242
#[ derive( Copy , Clone , PartialEq , Eq , Debug , Hash ) ]
241
243
pub struct OccursCheckId {
242
244
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 ,
246
246
}
247
247
248
248
/// Structure representing an inference context with its slab allocator mutex locked.
@@ -254,17 +254,25 @@ struct LockedContext<'ctx> {
254
254
}
255
255
256
256
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
+
257
265
/// Unify the type with another one.
258
266
///
259
267
/// 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 > {
261
269
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)
263
271
} )
264
272
}
265
273
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 ( ) ;
268
276
let bind_error = || Error :: Bind {
269
277
existing_bound : existing_bound. shallow_clone ( ) ,
270
278
new_bound : new. shallow_clone ( ) ,
@@ -277,7 +285,7 @@ impl<'ctx> LockedContext<'ctx> {
277
285
// Free types are simply dropped and replaced by the new bound
278
286
( Bound :: Free ( _) , _) => {
279
287
// Free means non-finalized, so set() is ok.
280
- existing . index . set ( new) ;
288
+ self . reassign_non_complete ( existing , new) ;
281
289
Ok ( ( ) )
282
290
}
283
291
// Binding complete->complete shouldn't ever happen, but if so, we just
@@ -319,14 +327,17 @@ impl<'ctx> LockedContext<'ctx> {
319
327
//
320
328
// It also gives the user access to more information about the type,
321
329
// 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) )
327
337
} else {
328
- Final :: product ( data1, data2)
329
- } ) ) ;
338
+ Final :: product ( Arc :: clone ( data1) , Arc :: clone ( data2) )
339
+ } ) ,
340
+ ) ;
330
341
}
331
342
Ok ( ( ) )
332
343
}
0 commit comments