|  | 
|  | 1 | +//@ check-pass | 
|  | 2 | +// | 
|  | 3 | +// During development of #124141 at one point expression on attributes were | 
|  | 4 | +// being duplicated and `m1` caused an exponential blowup that caused OOM. | 
|  | 5 | +// The number of recursive calls depends on the number of doc comments on the | 
|  | 6 | +// expr block. On each recursive call, the `#[allow(deprecated)]` attribute(s) on | 
|  | 7 | +// the `0` somehow get duplicated, resulting in 1, 2, 4, 8, ... identical | 
|  | 8 | +// attributes. | 
|  | 9 | +// | 
|  | 10 | +// After the fix, the code compiles quickly and normally. | 
|  | 11 | + | 
|  | 12 | +macro_rules! m1 { | 
|  | 13 | + ($(#[$meta:meta])* { $e:expr }) => { | 
|  | 14 | + m1! { expr: { $e }, unprocessed: [$(#[$meta])*] } | 
|  | 15 | + }; | 
|  | 16 | + | 
|  | 17 | + (expr: { $e:expr }, unprocessed: [ #[$meta:meta] $($metas:tt)* ]) => { | 
|  | 18 | + m1! { expr: { $e }, unprocessed: [ $($metas)* ] } | 
|  | 19 | + }; | 
|  | 20 | + | 
|  | 21 | + (expr: { $e:expr }, unprocessed: []) => { | 
|  | 22 | + { $e } | 
|  | 23 | + } | 
|  | 24 | +} | 
|  | 25 | + | 
|  | 26 | +macro_rules! m2 { | 
|  | 27 | + ($(#[$meta:meta])* { $e:stmt }) => { | 
|  | 28 | + m2! { stmt: { $e }, unprocessed: [$(#[$meta])*] } | 
|  | 29 | + }; | 
|  | 30 | + | 
|  | 31 | + (stmt: { $e:stmt }, unprocessed: [ #[$meta:meta] $($metas:tt)* ]) => { | 
|  | 32 | + m2! { stmt: { $e }, unprocessed: [ $($metas)* ] } | 
|  | 33 | + }; | 
|  | 34 | + | 
|  | 35 | + (stmt: { $e:stmt }, unprocessed: []) => { | 
|  | 36 | + { $e } | 
|  | 37 | + } | 
|  | 38 | +} | 
|  | 39 | + | 
|  | 40 | +macro_rules! m3 { | 
|  | 41 | + ($(#[$meta:meta])* { $e:item }) => { | 
|  | 42 | + m3! { item: { $e }, unprocessed: [$(#[$meta])*] } | 
|  | 43 | + }; | 
|  | 44 | + | 
|  | 45 | + (item: { $e:item }, unprocessed: [ #[$meta:meta] $($metas:tt)* ]) => { | 
|  | 46 | + m3! { item: { $e }, unprocessed: [ $($metas)* ] } | 
|  | 47 | + }; | 
|  | 48 | + | 
|  | 49 | + (item: { $e:item }, unprocessed: []) => { | 
|  | 50 | + { $e } | 
|  | 51 | + } | 
|  | 52 | +} | 
|  | 53 | + | 
|  | 54 | +fn main() { | 
|  | 55 | + // Each additional doc comment line doubles the compile time. | 
|  | 56 | + m1!( | 
|  | 57 | + /// a1 | 
|  | 58 | + /// a2 | 
|  | 59 | + /// a3 | 
|  | 60 | + /// a4 | 
|  | 61 | + /// a5 | 
|  | 62 | + /// a6 | 
|  | 63 | + /// a7 | 
|  | 64 | + /// a8 | 
|  | 65 | + /// a9 | 
|  | 66 | + /// a10 | 
|  | 67 | + /// a11 | 
|  | 68 | + /// a12 | 
|  | 69 | + /// a13 | 
|  | 70 | + /// a14 | 
|  | 71 | + /// a15 | 
|  | 72 | + /// a16 | 
|  | 73 | + /// a17 | 
|  | 74 | + /// a18 | 
|  | 75 | + /// a19 | 
|  | 76 | + /// a20 | 
|  | 77 | + { | 
|  | 78 | + #[allow(deprecated)] 0 | 
|  | 79 | + } | 
|  | 80 | + ); | 
|  | 81 | + | 
|  | 82 | + m2!( | 
|  | 83 | + /// a1 | 
|  | 84 | + /// a2 | 
|  | 85 | + /// a3 | 
|  | 86 | + /// a4 | 
|  | 87 | + /// a5 | 
|  | 88 | + /// a6 | 
|  | 89 | + /// a7 | 
|  | 90 | + /// a8 | 
|  | 91 | + /// a9 | 
|  | 92 | + /// a10 | 
|  | 93 | + /// a11 | 
|  | 94 | + /// a12 | 
|  | 95 | + /// a13 | 
|  | 96 | + /// a14 | 
|  | 97 | + /// a15 | 
|  | 98 | + /// a16 | 
|  | 99 | + /// a17 | 
|  | 100 | + /// a18 | 
|  | 101 | + /// a19 | 
|  | 102 | + /// a20 | 
|  | 103 | + { | 
|  | 104 | + #[allow(deprecated)] let x = 5 | 
|  | 105 | + } | 
|  | 106 | + ); | 
|  | 107 | + | 
|  | 108 | + m3!( | 
|  | 109 | + /// a1 | 
|  | 110 | + /// a2 | 
|  | 111 | + /// a3 | 
|  | 112 | + /// a4 | 
|  | 113 | + /// a5 | 
|  | 114 | + /// a6 | 
|  | 115 | + /// a7 | 
|  | 116 | + /// a8 | 
|  | 117 | + /// a9 | 
|  | 118 | + /// a10 | 
|  | 119 | + /// a11 | 
|  | 120 | + /// a12 | 
|  | 121 | + /// a13 | 
|  | 122 | + /// a14 | 
|  | 123 | + /// a15 | 
|  | 124 | + /// a16 | 
|  | 125 | + /// a17 | 
|  | 126 | + /// a18 | 
|  | 127 | + /// a19 | 
|  | 128 | + /// a20 | 
|  | 129 | + { | 
|  | 130 | + #[allow(deprecated)] struct S; | 
|  | 131 | + } | 
|  | 132 | + ); | 
|  | 133 | +} | 
0 commit comments