@@ -81,12 +81,14 @@ use crate::hir;
81
81
82
82
pub struct AllArenas {
83
83
pub interner : SyncDroplessArena ,
84
+ pub local_interner : SyncDroplessArena ,
84
85
}
85
86
86
87
impl AllArenas {
87
88
pub fn new ( ) -> Self {
88
89
AllArenas {
89
90
interner : SyncDroplessArena :: default ( ) ,
91
+ local_interner : SyncDroplessArena :: default ( ) ,
90
92
}
91
93
}
92
94
}
@@ -135,7 +137,7 @@ impl<'gcx: 'tcx, 'tcx> CtxtInterners<'tcx> {
135
137
/// Intern a type
136
138
#[ inline( never) ]
137
139
fn intern_ty (
138
- local : & CtxtInterners < ' tcx > ,
140
+ local : & CtxtInterners < ' gcx > ,
139
141
global : & CtxtInterners < ' gcx > ,
140
142
st : TyKind < ' tcx >
141
143
) -> Ty < ' tcx > {
@@ -160,6 +162,12 @@ impl<'gcx: 'tcx, 'tcx> CtxtInterners<'tcx> {
160
162
& ty_struct) ;
161
163
}
162
164
165
+ // This is safe because all the types the ty_struct can point to
166
+ // already is in the local arena or the global arena
167
+ let ty_struct: TyS < ' gcx > = unsafe {
168
+ mem:: transmute ( ty_struct)
169
+ } ;
170
+
163
171
Interned ( local. arena . alloc ( ty_struct) )
164
172
} ) . 0
165
173
} else {
@@ -1010,8 +1018,8 @@ pub struct FreeRegionInfo {
1010
1018
#[ derive( Copy , Clone ) ]
1011
1019
pub struct TyCtxt < ' a , ' gcx : ' tcx , ' tcx : ' a > {
1012
1020
gcx : & ' gcx GlobalCtxt < ' gcx > ,
1013
- interners : & ' tcx CtxtInterners < ' tcx > ,
1014
- dummy : PhantomData < & ' a ( ) > ,
1021
+ interners : & ' gcx CtxtInterners < ' gcx > ,
1022
+ dummy : PhantomData < ( & ' a ( ) , & ' tcx ( ) ) > ,
1015
1023
}
1016
1024
1017
1025
impl < ' gcx > Deref for TyCtxt < ' _ , ' gcx , ' _ > {
@@ -1026,6 +1034,7 @@ pub struct GlobalCtxt<'tcx> {
1026
1034
pub arena : WorkerLocal < Arena < ' tcx > > ,
1027
1035
1028
1036
global_interners : CtxtInterners < ' tcx > ,
1037
+ local_interners : CtxtInterners < ' tcx > ,
1029
1038
1030
1039
cstore : & ' tcx CrateStoreDyn ,
1031
1040
@@ -1222,6 +1231,7 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
1222
1231
s. fatal ( & err) ;
1223
1232
} ) ;
1224
1233
let interners = CtxtInterners :: new ( & arenas. interner ) ;
1234
+ let local_interners = CtxtInterners :: new ( & arenas. local_interner ) ;
1225
1235
let common = Common {
1226
1236
empty_predicates : ty:: GenericPredicates {
1227
1237
parent : None ,
@@ -1280,6 +1290,7 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
1280
1290
cstore,
1281
1291
arena : WorkerLocal :: new ( |_| Arena :: default ( ) ) ,
1282
1292
global_interners : interners,
1293
+ local_interners : local_interners,
1283
1294
dep_graph,
1284
1295
common,
1285
1296
types : common_types,
@@ -1669,18 +1680,15 @@ impl<'gcx> GlobalCtxt<'gcx> {
1669
1680
/// with the same lifetime as `arena`.
1670
1681
pub fn enter_local < ' tcx , F , R > (
1671
1682
& ' gcx self ,
1672
- arena : & ' tcx SyncDroplessArena ,
1673
- interners : & ' tcx mut Option < CtxtInterners < ' tcx > > ,
1674
1683
f : F
1675
1684
) -> R
1676
1685
where
1677
1686
F : FnOnce ( TyCtxt < ' tcx , ' gcx , ' tcx > ) -> R ,
1678
1687
' gcx : ' tcx ,
1679
1688
{
1680
- * interners = Some ( CtxtInterners :: new ( & arena) ) ;
1681
1689
let tcx = TyCtxt {
1682
1690
gcx : self ,
1683
- interners : interners . as_ref ( ) . unwrap ( ) ,
1691
+ interners : & self . local_interners ,
1684
1692
dummy : PhantomData ,
1685
1693
} ;
1686
1694
ty:: tls:: with_related_context ( tcx. global_tcx ( ) , |icx| {
@@ -2286,6 +2294,17 @@ macro_rules! intern_method {
2286
2294
pub fn $method( self , v: $alloc) -> & $lt_tcx $ty {
2287
2295
let key = ( $alloc_to_key) ( & v) ;
2288
2296
2297
+ let alloc = |v, interners: & ' gcx CtxtInterners <' gcx>| {
2298
+ // This transmutes $alloc<'tcx> to $alloc<'gcx>
2299
+ let v = unsafe {
2300
+ mem:: transmute( v)
2301
+ } ;
2302
+ let i: & $lt_tcx $ty = $alloc_method( & interners. arena, v) ;
2303
+ // Cast to 'gcx
2304
+ let i = unsafe { mem:: transmute( i) } ;
2305
+ Interned ( i)
2306
+ } ;
2307
+
2289
2308
// HACK(eddyb) Depend on flags being accurate to
2290
2309
// determine that all contents are in the global tcx.
2291
2310
// See comments on Lift for why we can't use that.
@@ -2299,18 +2318,11 @@ macro_rules! intern_method {
2299
2318
v) ;
2300
2319
}
2301
2320
2302
- Interned ( $alloc_method ( & self . interners. arena , v ) )
2321
+ alloc ( v , & self . interners)
2303
2322
} ) . 0
2304
2323
} else {
2305
2324
self . global_interners. $name. borrow_mut( ) . intern_ref( key, || {
2306
- // This transmutes $alloc<'tcx> to $alloc<'gcx>
2307
- let v = unsafe {
2308
- mem:: transmute( v)
2309
- } ;
2310
- let i: & $lt_tcx $ty = $alloc_method( & self . global_interners. arena, v) ;
2311
- // Cast to 'gcx
2312
- let i = unsafe { mem:: transmute( i) } ;
2313
- Interned ( i)
2325
+ alloc( v, & self . global_interners)
2314
2326
} ) . 0
2315
2327
}
2316
2328
}
0 commit comments