5
5
use rustc_arena:: DroplessArena ;
6
6
use rustc_data_structures:: fx:: FxHashMap ;
7
7
use rustc_data_structures:: stable_hasher:: { HashStable , StableHasher , ToStableHashKey } ;
8
+ use rustc_data_structures:: sync:: Lock ;
8
9
use rustc_macros:: HashStable_Generic ;
9
10
use rustc_serialize:: { Decodable , Decoder , Encodable , Encoder } ;
10
11
@@ -1623,14 +1624,15 @@ impl Symbol {
1623
1624
1624
1625
/// Maps a string to its interned representation.
1625
1626
pub fn intern ( string : & str ) -> Self {
1626
- with_interner ( |interner| interner . intern ( string) )
1627
+ with_session_globals ( |session_globals| session_globals . symbol_interner . intern ( string) )
1627
1628
}
1628
1629
1629
1630
/// Convert to a `SymbolStr`. This is a slowish operation because it
1630
1631
/// requires locking the symbol interner.
1631
1632
pub fn as_str ( self ) -> SymbolStr {
1632
- with_interner ( |interner| unsafe {
1633
- SymbolStr { string : std:: mem:: transmute :: < & str , & str > ( interner. get ( self ) ) }
1633
+ with_session_globals ( |session_globals| {
1634
+ let symbol_str = session_globals. symbol_interner . get ( self ) ;
1635
+ unsafe { SymbolStr { string : std:: mem:: transmute :: < & str , & str > ( symbol_str) } }
1634
1636
} )
1635
1637
}
1636
1638
@@ -1639,7 +1641,7 @@ impl Symbol {
1639
1641
}
1640
1642
1641
1643
pub fn len ( self ) -> usize {
1642
- with_interner ( |interner| interner . get ( self ) . len ( ) )
1644
+ with_session_globals ( |session_globals| session_globals . symbol_interner . get ( self ) . len ( ) )
1643
1645
}
1644
1646
1645
1647
pub fn is_empty ( self ) -> bool {
@@ -1696,6 +1698,9 @@ impl<CTX> ToStableHashKey<CTX> for Symbol {
1696
1698
}
1697
1699
}
1698
1700
1701
+ #[ derive( Default ) ]
1702
+ pub ( crate ) struct Interner ( Lock < InternerInner > ) ;
1703
+
1699
1704
// The `&'static str`s in this type actually point into the arena.
1700
1705
//
1701
1706
// The `FxHashMap`+`Vec` pair could be replaced by `FxIndexSet`, but #75278
@@ -1705,45 +1710,46 @@ impl<CTX> ToStableHashKey<CTX> for Symbol {
1705
1710
// This type is private to prevent accidentally constructing more than one `Interner` on the same
1706
1711
// thread, which makes it easy to mixup `Symbol`s between `Interner`s.
1707
1712
#[ derive( Default ) ]
1708
- pub ( crate ) struct Interner {
1713
+ struct InternerInner {
1709
1714
arena : DroplessArena ,
1710
1715
names : FxHashMap < & ' static str , Symbol > ,
1711
1716
strings : Vec < & ' static str > ,
1712
1717
}
1713
1718
1714
1719
impl Interner {
1715
1720
fn prefill ( init : & [ & ' static str ] ) -> Self {
1716
- Interner {
1721
+ Interner ( Lock :: new ( InternerInner {
1717
1722
strings : init. into ( ) ,
1718
1723
names : init. iter ( ) . copied ( ) . zip ( ( 0 ..) . map ( Symbol :: new) ) . collect ( ) ,
1719
1724
..Default :: default ( )
1720
- }
1725
+ } ) )
1721
1726
}
1722
1727
1723
1728
#[ inline]
1724
- pub fn intern ( & mut self , string : & str ) -> Symbol {
1725
- if let Some ( & name) = self . names . get ( string) {
1729
+ fn intern ( & self , string : & str ) -> Symbol {
1730
+ let mut inner = self . 0 . lock ( ) ;
1731
+ if let Some ( & name) = inner. names . get ( string) {
1726
1732
return name;
1727
1733
}
1728
1734
1729
- let name = Symbol :: new ( self . strings . len ( ) as u32 ) ;
1735
+ let name = Symbol :: new ( inner . strings . len ( ) as u32 ) ;
1730
1736
1731
1737
// `from_utf8_unchecked` is safe since we just allocated a `&str` which is known to be
1732
1738
// UTF-8.
1733
1739
let string: & str =
1734
- unsafe { str:: from_utf8_unchecked ( self . arena . alloc_slice ( string. as_bytes ( ) ) ) } ;
1740
+ unsafe { str:: from_utf8_unchecked ( inner . arena . alloc_slice ( string. as_bytes ( ) ) ) } ;
1735
1741
// It is safe to extend the arena allocation to `'static` because we only access
1736
1742
// these while the arena is still alive.
1737
1743
let string: & ' static str = unsafe { & * ( string as * const str ) } ;
1738
- self . strings . push ( string) ;
1739
- self . names . insert ( string, name) ;
1744
+ inner . strings . push ( string) ;
1745
+ inner . names . insert ( string, name) ;
1740
1746
name
1741
1747
}
1742
1748
1743
1749
// Get the symbol as a string. `Symbol::as_str()` should be used in
1744
1750
// preference to this function.
1745
- pub fn get ( & self , symbol : Symbol ) -> & str {
1746
- self . strings [ symbol. 0 . as_usize ( ) ]
1751
+ fn get ( & self , symbol : Symbol ) -> & str {
1752
+ self . 0 . lock ( ) . strings [ symbol. 0 . as_usize ( ) ]
1747
1753
}
1748
1754
}
1749
1755
@@ -1874,11 +1880,6 @@ impl Ident {
1874
1880
}
1875
1881
}
1876
1882
1877
- #[ inline]
1878
- fn with_interner < T , F : FnOnce ( & mut Interner ) -> T > ( f : F ) -> T {
1879
- with_session_globals ( |session_globals| f ( & mut * session_globals. symbol_interner . lock ( ) ) )
1880
- }
1881
-
1882
1883
/// An alternative to [`Symbol`], useful when the chars within the symbol need to
1883
1884
/// be accessed. It deliberately has limited functionality and should only be
1884
1885
/// used for temporary values.
0 commit comments