@@ -19,7 +19,7 @@ use std::sync::{Arc, Mutex, MutexGuard};
19
19
20
20
use crate :: dag:: { Dag , DagLike } ;
21
21
22
- use super :: { Bound , CompleteBound , Error , Final , Type } ;
22
+ use super :: { Bound , CompleteBound , Error , Final , Type , TypeInner } ;
23
23
24
24
/// Type inference context, or handle to a context.
25
25
///
@@ -60,13 +60,7 @@ impl Context {
60
60
/// Helper function to allocate a bound and return a reference to it.
61
61
fn alloc_bound ( & self , bound : Bound ) -> BoundRef {
62
62
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)
70
64
}
71
65
72
66
/// Allocate a new free type bound, and return a reference to it.
@@ -90,10 +84,24 @@ impl Context {
90
84
///
91
85
/// Panics if either of the child types are from a different inference context.
92
86
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
+ ) ;
95
95
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
+ }
97
105
}
98
106
99
107
/// Allocate a new product-type bound, and return a reference to it.
@@ -102,10 +110,27 @@ impl Context {
102
110
///
103
111
/// Panics if either of the child types are from a different inference context.
104
112
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
+ ) ;
107
121
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
+ }
109
134
}
110
135
111
136
/// Creates a new handle to the context.
@@ -133,7 +158,7 @@ impl Context {
133
158
/// # Panics
134
159
///
135
160
/// 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 {
137
162
bound. assert_matches_context ( self ) ;
138
163
let lock = self . lock ( ) ;
139
164
lock. slab [ bound. index ] . shallow_clone ( )
@@ -150,34 +175,54 @@ impl Context {
150
175
/// probably a bug.
151
176
///
152
177
/// 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 ) {
154
179
let mut lock = self . lock ( ) ;
155
180
lock. reassign_non_complete ( bound, new) ;
156
181
}
157
182
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.
160
185
///
161
186
/// 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
+
164
201
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
+ }
169
209
} )
170
210
}
171
211
172
212
/// Unify the type with another one.
173
213
///
174
214
/// Fails if the bounds on the two types are incompatible
175
215
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 ) ;
176
218
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
+ }
181
226
} )
182
227
}
183
228
@@ -257,7 +302,7 @@ pub struct OccursCheckId {
257
302
}
258
303
259
304
struct BindError {
260
- existing : Bound ,
305
+ existing : BoundRef ,
261
306
new : Bound ,
262
307
}
263
308
@@ -270,6 +315,16 @@ struct LockedContext<'ctx> {
270
315
}
271
316
272
317
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
+
273
328
fn reassign_non_complete ( & mut self , bound : BoundRef , new : Bound ) {
274
329
assert ! (
275
330
!matches!( self . slab[ bound. index] , Bound :: Complete ( ..) ) ,
@@ -278,10 +333,29 @@ impl<'ctx> LockedContext<'ctx> {
278
333
self . slab [ bound. index ] = new;
279
334
}
280
335
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
+
281
355
/// Unify the type with another one.
282
356
///
283
357
/// 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 > {
285
359
existing. bound . unify ( & other. bound , |x_bound, y_bound| {
286
360
self . bind ( x_bound, self . slab [ y_bound. index ] . shallow_clone ( ) )
287
361
} )
@@ -290,7 +364,7 @@ impl<'ctx> LockedContext<'ctx> {
290
364
fn bind ( & mut self , existing : BoundRef , new : Bound ) -> Result < ( ) , BindError > {
291
365
let existing_bound = self . slab [ existing. index ] . shallow_clone ( ) ;
292
366
let bind_error = || BindError {
293
- existing : existing_bound . shallow_clone ( ) ,
367
+ existing : existing . clone ( ) ,
294
368
new : new. shallow_clone ( ) ,
295
369
} ;
296
370
@@ -342,24 +416,19 @@ impl<'ctx> LockedContext<'ctx> {
342
416
//
343
417
// It also gives the user access to more information about the type,
344
418
// 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) {
348
420
self . reassign_non_complete (
349
421
existing,
350
422
Bound :: Complete ( if let Bound :: Sum ( ..) = existing_bound {
351
- Final :: sum ( Arc :: clone ( data1) , Arc :: clone ( data2) )
423
+ Final :: sum ( data1, data2)
352
424
} else {
353
- Final :: product ( Arc :: clone ( data1) , Arc :: clone ( data2) )
425
+ Final :: product ( data1, data2)
354
426
} ) ,
355
427
) ;
356
428
}
357
429
Ok ( ( ) )
358
430
}
359
- ( x, y) => Err ( BindError {
360
- existing : x. shallow_clone ( ) ,
361
- new : y. shallow_clone ( ) ,
362
- } ) ,
431
+ ( _, _) => Err ( bind_error ( ) ) ,
363
432
}
364
433
}
365
434
}
0 commit comments