24
24
// because getting it wrong can lead to nested `HygieneData::with` calls that
25
25
// trigger runtime aborts. (Fortunately these are obvious and easy to fix.)
26
26
27
- use std:: cell:: RefCell ;
28
- use std:: collections:: hash_map:: Entry ;
29
- use std:: collections:: hash_set:: Entry as SetEntry ;
30
27
use std:: hash:: Hash ;
31
28
use std:: sync:: Arc ;
32
29
use std:: { fmt, iter, mem} ;
33
30
34
31
use rustc_data_structures:: fingerprint:: Fingerprint ;
35
32
use rustc_data_structures:: fx:: { FxHashMap , FxHashSet } ;
36
33
use rustc_data_structures:: stable_hasher:: { HashStable , HashingControls , StableHasher } ;
37
- use rustc_data_structures:: sync:: { Lock , WorkerLocal } ;
34
+ use rustc_data_structures:: sync:: Lock ;
38
35
use rustc_data_structures:: unhash:: UnhashMap ;
39
36
use rustc_hashes:: Hash64 ;
40
37
use rustc_index:: IndexVec ;
@@ -61,8 +58,8 @@ impl !PartialOrd for SyntaxContext {}
61
58
/// The other fields are only for caching.
62
59
type SyntaxContextKey = ( SyntaxContext , ExpnId , Transparency ) ;
63
60
64
- #[ derive( Clone , Copy , PartialEq , Debug , Encodable , Decodable ) ]
65
- pub struct SyntaxContextData {
61
+ #[ derive( Clone , Copy , Debug ) ]
62
+ struct SyntaxContextData {
66
63
outer_expn : ExpnId ,
67
64
outer_transparency : Transparency ,
68
65
parent : SyntaxContext ,
@@ -74,6 +71,17 @@ pub struct SyntaxContextData {
74
71
dollar_crate_name : Symbol ,
75
72
}
76
73
74
+ /// Same as `SyntaxContextData`, but `opaque(_and_semitransparent)` cannot be recursive
75
+ /// and use `None` if they need to refer to self. Used for encoding and decoding metadata.
76
+ #[ derive( Encodable , Decodable ) ]
77
+ pub struct SyntaxContextDataNonRecursive {
78
+ outer_expn : ExpnId ,
79
+ outer_transparency : Transparency ,
80
+ parent : SyntaxContext ,
81
+ opaque : Option < SyntaxContext > ,
82
+ opaque_and_semitransparent : Option < SyntaxContext > ,
83
+ }
84
+
77
85
impl SyntaxContextData {
78
86
fn new (
79
87
( parent, outer_expn, outer_transparency) : SyntaxContextKey ,
@@ -114,6 +122,19 @@ impl SyntaxContextData {
114
122
}
115
123
}
116
124
125
+ impl SyntaxContextDataNonRecursive {
126
+ fn recursive ( & self , ctxt : SyntaxContext ) -> SyntaxContextData {
127
+ SyntaxContextData {
128
+ outer_expn : self . outer_expn ,
129
+ outer_transparency : self . outer_transparency ,
130
+ parent : self . parent ,
131
+ opaque : self . opaque . unwrap_or ( ctxt) ,
132
+ opaque_and_semitransparent : self . opaque_and_semitransparent . unwrap_or ( ctxt) ,
133
+ dollar_crate_name : kw:: DollarCrate ,
134
+ }
135
+ }
136
+ }
137
+
117
138
rustc_index:: newtype_index! {
118
139
/// A unique ID associated with a macro invocation and expansion.
119
140
#[ orderable]
@@ -637,6 +658,19 @@ impl HygieneData {
637
658
SyntaxContextData :: new ( key, opaque, opaque_and_semitransparent) ;
638
659
ctxt
639
660
}
661
+
662
+ fn non_recursive_ctxt ( & self , ctxt : SyntaxContext ) -> SyntaxContextDataNonRecursive {
663
+ debug_assert ! ( !self . syntax_context_data[ ctxt. 0 as usize ] . is_decode_placeholder( ) ) ;
664
+ let data = & self . syntax_context_data [ ctxt. 0 as usize ] ;
665
+ SyntaxContextDataNonRecursive {
666
+ outer_expn : data. outer_expn ,
667
+ outer_transparency : data. outer_transparency ,
668
+ parent : data. parent ,
669
+ opaque : ( data. opaque != ctxt) . then_some ( data. opaque ) ,
670
+ opaque_and_semitransparent : ( data. opaque_and_semitransparent != ctxt)
671
+ . then_some ( data. opaque_and_semitransparent ) ,
672
+ }
673
+ }
640
674
}
641
675
642
676
pub fn walk_chain ( span : Span , to : SyntaxContext ) -> Span {
@@ -1266,7 +1300,7 @@ impl HygieneEncodeContext {
1266
1300
pub fn encode < T > (
1267
1301
& self ,
1268
1302
encoder : & mut T ,
1269
- mut encode_ctxt : impl FnMut ( & mut T , u32 , & SyntaxContextData ) ,
1303
+ mut encode_ctxt : impl FnMut ( & mut T , u32 , & SyntaxContextDataNonRecursive ) ,
1270
1304
mut encode_expn : impl FnMut ( & mut T , ExpnId , & ExpnData , ExpnHash ) ,
1271
1305
) {
1272
1306
// When we serialize a `SyntaxContextData`, we may end up serializing
@@ -1315,18 +1349,12 @@ struct HygieneDecodeContextInner {
1315
1349
// so that multiple occurrences of the same serialized id are decoded to the same
1316
1350
// `SyntaxContext`. This only stores `SyntaxContext`s which are completely decoded.
1317
1351
remapped_ctxts : Vec < Option < SyntaxContext > > ,
1318
-
1319
- /// Maps serialized `SyntaxContext` ids that are currently being decoded to a `SyntaxContext`.
1320
- decoding : FxHashMap < u32 , SyntaxContext > ,
1321
1352
}
1322
1353
1323
1354
#[ derive( Default ) ]
1324
1355
/// Additional information used to assist in decoding hygiene data
1325
1356
pub struct HygieneDecodeContext {
1326
1357
inner : Lock < HygieneDecodeContextInner > ,
1327
-
1328
- /// A set of serialized `SyntaxContext` ids that are currently being decoded on each thread.
1329
- local_in_progress : WorkerLocal < RefCell < FxHashSet < u32 > > > ,
1330
1358
}
1331
1359
1332
1360
/// Register an expansion which has been decoded from the on-disk-cache for the local crate.
@@ -1397,7 +1425,10 @@ pub fn decode_expn_id(
1397
1425
// to track which `SyntaxContext`s we have already decoded.
1398
1426
// The provided closure will be invoked to deserialize a `SyntaxContextData`
1399
1427
// if we haven't already seen the id of the `SyntaxContext` we are deserializing.
1400
- pub fn decode_syntax_context < D : Decoder , F : FnOnce ( & mut D , u32 ) -> SyntaxContextData > (
1428
+ pub fn decode_syntax_context <
1429
+ D : Decoder ,
1430
+ F : FnOnce ( & mut D , u32 ) -> SyntaxContextDataNonRecursive ,
1431
+ > (
1401
1432
d : & mut D ,
1402
1433
context : & HygieneDecodeContext ,
1403
1434
decode_data : F ,
@@ -1409,113 +1440,43 @@ pub fn decode_syntax_context<D: Decoder, F: FnOnce(&mut D, u32) -> SyntaxContext
1409
1440
return SyntaxContext :: root ( ) ;
1410
1441
}
1411
1442
1412
- let pending_ctxt = {
1413
- let mut inner = context. inner . lock ( ) ;
1414
-
1415
- // Reminder: `HygieneDecodeContext` is per-crate, so there are no collisions between
1416
- // raw ids from different crate metadatas.
1417
- if let Some ( ctxt) = inner. remapped_ctxts . get ( raw_id as usize ) . copied ( ) . flatten ( ) {
1418
- // This has already been decoded.
1419
- return ctxt;
1420
- }
1421
-
1422
- match inner. decoding . entry ( raw_id) {
1423
- Entry :: Occupied ( ctxt_entry) => {
1424
- let pending_ctxt = * ctxt_entry. get ( ) ;
1425
- match context. local_in_progress . borrow_mut ( ) . entry ( raw_id) {
1426
- // We're decoding this already on the current thread. Return here and let the
1427
- // function higher up the stack finish decoding to handle recursive cases.
1428
- // Hopefully having a `SyntaxContext` that refers to an incorrect data is ok
1429
- // during reminder of the decoding process, it's certainly not ok after the
1430
- // top level decoding function returns.
1431
- SetEntry :: Occupied ( ..) => return pending_ctxt,
1432
- // Some other thread is currently decoding this.
1433
- // Race with it (alternatively we could wait here).
1434
- // We cannot return this value, unlike in the recursive case above, because it
1435
- // may expose a `SyntaxContext` pointing to incorrect data to arbitrary code.
1436
- SetEntry :: Vacant ( entry) => {
1437
- entry. insert ( ) ;
1438
- pending_ctxt
1439
- }
1440
- }
1441
- }
1442
- Entry :: Vacant ( entry) => {
1443
- // We are the first thread to start decoding. Mark the current thread as being
1444
- // progress.
1445
- context. local_in_progress . borrow_mut ( ) . insert ( raw_id) ;
1446
-
1447
- // Allocate and store SyntaxContext id *before* calling the decoder function,
1448
- // as the SyntaxContextData may reference itself.
1449
- let new_ctxt = HygieneData :: with ( |hygiene_data| {
1450
- // Push a dummy SyntaxContextData to ensure that nobody else can get the
1451
- // same ID as us. This will be overwritten after call `decode_data`.
1452
- hygiene_data. syntax_context_data . push ( SyntaxContextData :: decode_placeholder ( ) ) ;
1453
- SyntaxContext :: from_usize ( hygiene_data. syntax_context_data . len ( ) - 1 )
1454
- } ) ;
1455
- entry. insert ( new_ctxt) ;
1456
- new_ctxt
1457
- }
1458
- }
1459
- } ;
1443
+ // Reminder: `HygieneDecodeContext` is per-crate, so there are no collisions between
1444
+ // raw ids from different crate metadatas.
1445
+ if let Some ( ctxt) = context. inner . lock ( ) . remapped_ctxts . get ( raw_id as usize ) . copied ( ) . flatten ( )
1446
+ {
1447
+ // This has already been decoded.
1448
+ return ctxt;
1449
+ }
1460
1450
1461
1451
// Don't try to decode data while holding the lock, since we need to
1462
1452
// be able to recursively decode a SyntaxContext
1463
1453
let ctxt_data = decode_data ( d, raw_id) ;
1464
- let ctxt_key = ctxt_data. key ( ) ;
1465
1454
1466
1455
let ctxt = HygieneData :: with ( |hygiene_data| {
1467
- match hygiene_data. syntax_context_map . get ( & ctxt_key) {
1468
- // Ensure that syntax contexts are unique.
1469
- // If syntax contexts with the given key already exists, reuse it instead of
1470
- // using `pending_ctxt`.
1471
- // `pending_ctxt` will leave an unused hole in the vector of syntax contexts.
1472
- // Hopefully its value isn't stored anywhere during decoding and its dummy data
1473
- // is never accessed later. The `is_decode_placeholder` asserts on all
1474
- // accesses to syntax context data attempt to ensure it.
1475
- Some ( & ctxt) => ctxt,
1476
- // This is a completely new context.
1477
- // Overwrite its placeholder data with our decoded data.
1478
- None => {
1479
- let ctxt_data_ref =
1480
- & mut hygiene_data. syntax_context_data [ pending_ctxt. as_u32 ( ) as usize ] ;
1481
- let prev_ctxt_data = mem:: replace ( ctxt_data_ref, ctxt_data) ;
1482
- // Reset `dollar_crate_name` so that it will be updated by `update_dollar_crate_names`.
1483
- // We don't care what the encoding crate set this to - we want to resolve it
1484
- // from the perspective of the current compilation session.
1485
- ctxt_data_ref. dollar_crate_name = kw:: DollarCrate ;
1486
- // Make sure nothing weird happened while `decode_data` was running.
1487
- if !prev_ctxt_data. is_decode_placeholder ( ) {
1488
- // Another thread may have already inserted the decoded data,
1489
- // but the decoded data should match.
1490
- assert_eq ! ( prev_ctxt_data, * ctxt_data_ref) ;
1491
- }
1492
- hygiene_data. syntax_context_map . insert ( ctxt_key, pending_ctxt) ;
1493
- pending_ctxt
1494
- }
1495
- }
1456
+ let ctxt_key = ( ctxt_data. parent , ctxt_data. outer_expn , ctxt_data. outer_transparency ) ;
1457
+ * hygiene_data. syntax_context_map . entry ( ctxt_key) . or_insert_with ( || {
1458
+ let ctxt = SyntaxContext :: from_usize ( hygiene_data. syntax_context_data . len ( ) ) ;
1459
+ hygiene_data. syntax_context_data . push ( ctxt_data. recursive ( ctxt) ) ;
1460
+ ctxt
1461
+ } )
1496
1462
} ) ;
1497
1463
1498
- // Mark the context as completed
1499
- context. local_in_progress . borrow_mut ( ) . remove ( & raw_id) ;
1500
-
1501
1464
let mut inner = context. inner . lock ( ) ;
1502
1465
let new_len = raw_id as usize + 1 ;
1503
1466
if inner. remapped_ctxts . len ( ) < new_len {
1504
1467
inner. remapped_ctxts . resize ( new_len, None ) ;
1505
1468
}
1506
1469
inner. remapped_ctxts [ raw_id as usize ] = Some ( ctxt) ;
1507
- inner. decoding . remove ( & raw_id) ;
1508
1470
1509
1471
ctxt
1510
1472
}
1511
1473
1512
- fn for_all_ctxts_in < F : FnMut ( u32 , SyntaxContext , & SyntaxContextData ) > (
1474
+ fn for_all_ctxts_in < F : FnMut ( u32 , SyntaxContext , & SyntaxContextDataNonRecursive ) > (
1513
1475
ctxts : impl Iterator < Item = SyntaxContext > ,
1514
1476
mut f : F ,
1515
1477
) {
1516
- let all_data: Vec < _ > = HygieneData :: with ( |data| {
1517
- ctxts. map ( |ctxt| ( ctxt, data. syntax_context_data [ ctxt. 0 as usize ] . clone ( ) ) ) . collect ( )
1518
- } ) ;
1478
+ let all_data: Vec < _ > =
1479
+ HygieneData :: with ( |data| ctxts. map ( |ctxt| ( ctxt, data. non_recursive_ctxt ( ctxt) ) ) . collect ( ) ) ;
1519
1480
for ( ctxt, data) in all_data. into_iter ( ) {
1520
1481
f ( ctxt. 0 , ctxt, & data) ;
1521
1482
}
0 commit comments