@@ -190,6 +190,7 @@ use syntax_pos::{Span, DUMMY_SP};
190
190
191
191
use arena:: TypedArena ;
192
192
193
+ use smallvec:: { SmallVec , smallvec} ;
193
194
use std:: cmp:: { self , Ordering , min, max} ;
194
195
use std:: fmt;
195
196
use std:: iter:: { FromIterator , IntoIterator } ;
@@ -237,14 +238,16 @@ impl<'tcx> Pattern<'tcx> {
237
238
}
238
239
}
239
240
240
- pub struct Matrix < ' a , ' tcx : ' a > ( Vec < Vec < & ' a Pattern < ' tcx > > > ) ;
241
+ /// A 2D matrix. Nx1 matrices are very common, which is why `SmallVec[_; 2]`
242
+ /// works well for each row.
243
+ pub struct Matrix < ' p , ' tcx : ' p > ( Vec < SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > > ) ;
241
244
242
- impl < ' a , ' tcx > Matrix < ' a , ' tcx > {
245
+ impl < ' p , ' tcx > Matrix < ' p , ' tcx > {
243
246
pub fn empty ( ) -> Self {
244
247
Matrix ( vec ! [ ] )
245
248
}
246
249
247
- pub fn push ( & mut self , row : Vec < & ' a Pattern < ' tcx > > ) {
250
+ pub fn push ( & mut self , row : SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > ) {
248
251
self . 0 . push ( row)
249
252
}
250
253
}
@@ -261,7 +264,7 @@ impl<'a, 'tcx> Matrix<'a, 'tcx> {
261
264
/// ++++++++++++++++++++++++++
262
265
/// + _ + [_, _, ..tail] +
263
266
/// ++++++++++++++++++++++++++
264
- impl < ' a , ' tcx > fmt:: Debug for Matrix < ' a , ' tcx > {
267
+ impl < ' p , ' tcx > fmt:: Debug for Matrix < ' p , ' tcx > {
265
268
fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
266
269
write ! ( f, "\n " ) ?;
267
270
@@ -293,8 +296,9 @@ impl<'a, 'tcx> fmt::Debug for Matrix<'a, 'tcx> {
293
296
}
294
297
}
295
298
296
- impl < ' a , ' tcx > FromIterator < Vec < & ' a Pattern < ' tcx > > > for Matrix < ' a , ' tcx > {
297
- fn from_iter < T : IntoIterator < Item =Vec < & ' a Pattern < ' tcx > > > > ( iter : T ) -> Self
299
+ impl < ' p , ' tcx > FromIterator < SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > > for Matrix < ' p , ' tcx > {
300
+ fn from_iter < T > ( iter : T ) -> Self
301
+ where T : IntoIterator < Item =SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > >
298
302
{
299
303
Matrix ( iter. into_iter ( ) . collect ( ) )
300
304
}
@@ -998,7 +1002,7 @@ fn compute_missing_ctors<'a, 'tcx: 'a>(
998
1002
/// matrix isn't exhaustive).
999
1003
pub fn is_useful < ' p , ' a : ' p , ' tcx : ' a > ( cx : & mut MatchCheckCtxt < ' a , ' tcx > ,
1000
1004
matrix : & Matrix < ' p , ' tcx > ,
1001
- v : & [ & ' p Pattern < ' tcx > ] ,
1005
+ v : & [ & Pattern < ' tcx > ] ,
1002
1006
witness : WitnessPreference )
1003
1007
-> Usefulness < ' tcx > {
1004
1008
let & Matrix ( ref rows) = matrix;
@@ -1108,7 +1112,7 @@ pub fn is_useful<'p, 'a: 'p, 'tcx: 'a>(cx: &mut MatchCheckCtxt<'a, 'tcx>,
1108
1112
} else {
1109
1113
let matrix = rows. iter ( ) . filter_map ( |r| {
1110
1114
if r[ 0 ] . is_wildcard ( ) {
1111
- Some ( r[ 1 ..] . to_vec ( ) )
1115
+ Some ( SmallVec :: from_slice ( & r[ 1 ..] ) )
1112
1116
} else {
1113
1117
None
1114
1118
}
@@ -1199,10 +1203,10 @@ pub fn is_useful<'p, 'a: 'p, 'tcx: 'a>(cx: &mut MatchCheckCtxt<'a, 'tcx>,
1199
1203
1200
1204
/// A shorthand for the `U(S(c, P), S(c, q))` operation from the paper. I.e., `is_useful` applied
1201
1205
/// to the specialised version of both the pattern matrix `P` and the new pattern `q`.
1202
- fn is_useful_specialized < ' p , ' a : ' p , ' tcx : ' a > (
1206
+ fn is_useful_specialized < ' p , ' a : ' p , ' tcx : ' a > (
1203
1207
cx : & mut MatchCheckCtxt < ' a , ' tcx > ,
1204
1208
& Matrix ( ref m) : & Matrix < ' p , ' tcx > ,
1205
- v : & [ & ' p Pattern < ' tcx > ] ,
1209
+ v : & [ & Pattern < ' tcx > ] ,
1206
1210
ctor : Constructor < ' tcx > ,
1207
1211
lty : Ty < ' tcx > ,
1208
1212
witness : WitnessPreference ,
@@ -1521,15 +1525,15 @@ fn constructor_intersects_pattern<'p, 'a: 'p, 'tcx: 'a>(
1521
1525
tcx : TyCtxt < ' a , ' tcx , ' tcx > ,
1522
1526
ctor : & Constructor < ' tcx > ,
1523
1527
pat : & ' p Pattern < ' tcx > ,
1524
- ) -> Option < Vec < & ' p Pattern < ' tcx > > > {
1528
+ ) -> Option < SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > > {
1525
1529
if should_treat_range_exhaustively ( tcx, ctor) {
1526
1530
match ( IntRange :: from_ctor ( tcx, ctor) , IntRange :: from_pat ( tcx, pat) ) {
1527
1531
( Some ( ctor) , Some ( pat) ) => {
1528
1532
ctor. intersection ( & pat) . map ( |_| {
1529
1533
let ( pat_lo, pat_hi) = pat. range . into_inner ( ) ;
1530
1534
let ( ctor_lo, ctor_hi) = ctor. range . into_inner ( ) ;
1531
1535
assert ! ( pat_lo <= ctor_lo && ctor_hi <= pat_hi) ;
1532
- vec ! [ ]
1536
+ smallvec ! [ ]
1533
1537
} )
1534
1538
}
1535
1539
_ => None ,
@@ -1539,7 +1543,7 @@ fn constructor_intersects_pattern<'p, 'a: 'p, 'tcx: 'a>(
1539
1543
// conveniently handled by `IntRange`. For these cases, the constructor may not be a range
1540
1544
// so intersection actually devolves into being covered by the pattern.
1541
1545
match constructor_covered_by_range ( tcx, ctor, pat) {
1542
- Ok ( true ) => Some ( vec ! [ ] ) ,
1546
+ Ok ( true ) => Some ( smallvec ! [ ] ) ,
1543
1547
Ok ( false ) | Err ( ErrorReported ) => None ,
1544
1548
}
1545
1549
}
@@ -1610,9 +1614,9 @@ fn constructor_covered_by_range<'a, 'tcx>(
1610
1614
fn patterns_for_variant < ' p , ' a : ' p , ' tcx : ' a > (
1611
1615
subpatterns : & ' p [ FieldPattern < ' tcx > ] ,
1612
1616
wild_patterns : & [ & ' p Pattern < ' tcx > ] )
1613
- -> Vec < & ' p Pattern < ' tcx > >
1617
+ -> SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] >
1614
1618
{
1615
- let mut result = wild_patterns . to_owned ( ) ;
1619
+ let mut result = SmallVec :: from_slice ( wild_patterns ) ;
1616
1620
1617
1621
for subpat in subpatterns {
1618
1622
result[ subpat. field . index ( ) ] = & subpat. pattern ;
@@ -1635,15 +1639,16 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
1635
1639
r : & [ & ' p Pattern < ' tcx > ] ,
1636
1640
constructor : & Constructor < ' tcx > ,
1637
1641
wild_patterns : & [ & ' p Pattern < ' tcx > ] ,
1638
- ) -> Option < Vec < & ' p Pattern < ' tcx > > > {
1642
+ ) -> Option < SmallVec < [ & ' p Pattern < ' tcx > ; 2 ] > > {
1639
1643
let pat = & r[ 0 ] ;
1640
1644
1641
- let head: Option < Vec < & Pattern > > = match * pat. kind {
1642
- PatternKind :: AscribeUserType { ref subpattern, .. } =>
1643
- specialize ( cx, :: std:: slice:: from_ref ( & subpattern) , constructor, wild_patterns) ,
1645
+ let head = match * pat. kind {
1646
+ PatternKind :: AscribeUserType { ref subpattern, .. } => {
1647
+ specialize ( cx, :: std:: slice:: from_ref ( & subpattern) , constructor, wild_patterns)
1648
+ }
1644
1649
1645
1650
PatternKind :: Binding { .. } | PatternKind :: Wild => {
1646
- Some ( wild_patterns . to_owned ( ) )
1651
+ Some ( SmallVec :: from_slice ( wild_patterns ) )
1647
1652
}
1648
1653
1649
1654
PatternKind :: Variant { adt_def, variant_index, ref subpatterns, .. } => {
@@ -1660,7 +1665,7 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
1660
1665
}
1661
1666
1662
1667
PatternKind :: Deref { ref subpattern } => {
1663
- Some ( vec ! [ subpattern] )
1668
+ Some ( smallvec ! [ subpattern] )
1664
1669
}
1665
1670
1666
1671
PatternKind :: Constant { value } => {
@@ -1696,7 +1701,7 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
1696
1701
if wild_patterns. len ( ) as u64 == n {
1697
1702
// convert a constant slice/array pattern to a list of patterns.
1698
1703
match ( n, opt_ptr) {
1699
- ( 0 , _) => Some ( Vec :: new ( ) ) ,
1704
+ ( 0 , _) => Some ( SmallVec :: new ( ) ) ,
1700
1705
( _, Some ( ptr) ) => {
1701
1706
let alloc = cx. tcx . alloc_map . lock ( ) . unwrap_memory ( ptr. alloc_id ) ;
1702
1707
let layout = cx. tcx . layout_of ( cx. param_env . and ( ty) ) . ok ( ) ?;
@@ -1765,7 +1770,7 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
1765
1770
match slice_pat_covered_by_constructor (
1766
1771
cx. tcx , pat. span , constructor, prefix, slice, suffix
1767
1772
) {
1768
- Ok ( true ) => Some ( vec ! [ ] ) ,
1773
+ Ok ( true ) => Some ( smallvec ! [ ] ) ,
1769
1774
Ok ( false ) => None ,
1770
1775
Err ( ErrorReported ) => None
1771
1776
}
0 commit comments