1
//! Implementation of the Tor Vegas congestion control algorithm.
2
//!
3
//! This is used by the circuit reactor in order to decide when to send data and SENDMEs.
4
//!
5
//! Spec: prop324 section 3.3 (TOR_VEGAS)
6

            
7
use super::{
8
    CongestionControlAlgorithm, CongestionSignals, CongestionWindow, State,
9
    params::{Algorithm, VegasParams},
10
    rtt::{ClockStall, RoundtripTimeEstimator},
11
};
12
use crate::Result;
13

            
14
use tor_error::{error_report, internal};
15

            
16
/// Bandwidth-Delay Product (BDP) estimator.
17
///
18
/// Spec: prop324 section 3.1 (BDP_ESTIMATION).
19
#[derive(Clone, Debug, Default)]
20
pub(crate) struct BdpEstimator {
21
    /// The BDP value of this estimator.
22
    bdp: u32,
23
}
24

            
25
impl BdpEstimator {
26
    /// Return the current BDP value.
27
166
    fn get(&self) -> u32 {
28
166
        self.bdp
29
166
    }
30

            
31
    /// Update the estimator with the given congestion window, RTT estimator and any condition
32
    /// signals that we are currently experiencing.
33
    ///
34
    /// C-tor: congestion_control_update_circuit_bdp() in congestion_control_common.c
35
156
    fn update(
36
156
        &mut self,
37
156
        cwnd: &CongestionWindow,
38
156
        rtt: &RoundtripTimeEstimator,
39
156
        signals: &CongestionSignals,
40
156
    ) {
41
        // Stalled clock means our RTT value is invalid so set the BDP to the cwnd.
42
156
        if rtt.clock_stalled() {
43
            self.bdp = if signals.channel_blocked {
44
                // Set the BDP to the cwnd minus the outbound queue size, capping it to the minimum
45
                // cwnd.
46
                cwnd.get()
47
                    .saturating_sub(signals.channel_outbound_size)
48
                    .max(cwnd.min())
49
            } else {
50
                cwnd.get()
51
            };
52
156
        } else {
53
156
            // Congestion window based BDP will respond to changes in RTT only, and is relative to
54
156
            // cwnd growth. It is useful for correcting for BDP overestimation, but if BDP is
55
156
            // higher than the current cwnd, it will underestimate it.
56
156
            //
57
156
            // To clarify this is equivalent to: cwnd * min_rtt / ewma_rtt.
58
156
            let min_rtt_usec = rtt.min_rtt_usec().unwrap_or(u32::MAX);
59
156
            let ewma_rtt_usec = rtt.ewma_rtt_usec().unwrap_or(u32::MAX);
60
156
            self.bdp = cwnd
61
156
                .get()
62
156
                .saturating_mul(min_rtt_usec)
63
156
                .saturating_div(ewma_rtt_usec);
64
156
        }
65
156
    }
66
}
67

            
68
/// Congestion control Vegas algorithm.
69
///
70
/// TCP Vegas control algorithm estimates the queue lengths at relays by subtracting the current
71
/// BDP estimate from the current congestion window.
72
///
73
/// This object implements CongestionControlAlgorithm trait used by the ['CongestionControl'].
74
///
75
/// Spec: prop324 section 3.3 (TOR_VEGAS)
76
/// C-tor: Split between congestion_control_vegas.c and the congestion_control_t struct.
77
#[derive(Clone, Debug)]
78
pub(crate) struct Vegas {
79
    /// Congestion control parameters.
80
    params: VegasParams,
81
    /// Bandwidth delay product.
82
    /// C-tor: "bdp"
83
    bdp: BdpEstimator,
84
    /// Congestion window.
85
    /// C-tor: "cwnd", "cwnd_inc_pct_ss", "cwnd_inc", "cwnd_min", "cwnd_inc_rate", "cwnd_full",
86
    cwnd: CongestionWindow,
87
    /// Number of cells expected before we send a SENDME resulting in more data.
88
    num_cell_until_sendme: u32,
89
    /// The number of SENDME until we will acknowledge a congestion event again.
90
    /// C-tor: "next_cc_event"
91
    num_sendme_until_cwnd_update: u32,
92
    /// Counts down until we process a cwnd worth of SENDME acks. Used to track full cwnd status.
93
    /// C-tor: "next_cwnd_event"
94
    num_sendme_per_cwnd: u32,
95
    /// Number of cells in-flight (sent but awaiting SENDME ack).
96
    /// C-tor: "inflight"
97
    num_inflight: u32,
98
    /// Indicate if we noticed we were blocked on channel during an algorithm run. This is used to
99
    /// notice a change from blocked to non-blocked in order to reset the num_sendme_per_cwnd.
100
    /// C-tor: "blocked_chan"
101
    is_blocked_on_chan: bool,
102
}
103

            
104
impl Vegas {
105
    /// Create a new [`Vegas`] from the specified parameters, state, and cwnd.
106
392
    pub(crate) fn new(params: VegasParams, state: &State, cwnd: CongestionWindow) -> Self {
107
392
        Self {
108
392
            params,
109
392
            bdp: BdpEstimator::default(),
110
392
            num_cell_until_sendme: cwnd.sendme_inc(),
111
392
            num_inflight: 0,
112
392
            num_sendme_per_cwnd: 0,
113
392
            num_sendme_until_cwnd_update: cwnd.update_rate(state),
114
392
            cwnd,
115
392
            is_blocked_on_chan: false,
116
392
        }
117
392
    }
118
}
119

            
120
impl CongestionControlAlgorithm for Vegas {
121
16
    fn uses_stream_sendme(&self) -> bool {
122
        // Not allowed as in Vegas doesn't need them.
123
16
        false
124
16
    }
125

            
126
80
    fn uses_xon_xoff(&self) -> bool {
127
80
        true
128
80
    }
129

            
130
1200
    fn is_next_cell_sendme(&self) -> bool {
131
        // Matching inflight number to the SENDME increment, time to send a SENDME. Contrary to
132
        // C-tor, this is called after num_inflight is incremented.
133
1200
        self.num_inflight.is_multiple_of(self.cwnd.sendme_inc())
134
1200
    }
135

            
136
9402
    fn can_send(&self) -> bool {
137
9402
        self.num_inflight < self.cwnd.get()
138
9402
    }
139

            
140
500
    fn cwnd(&self) -> Option<CongestionWindow> {
141
500
        Some(self.cwnd)
142
500
    }
143

            
144
    /// Called when a SENDME cell is received.
145
    ///
146
    /// This is where the Vegas algorithm magic happens entirely. For every SENDME we get, the
147
    /// entire state is updated which usually result in the congestion window being changed.
148
    ///
149
    /// An error is returned if there is a protocol violation with regards to flow or congestion
150
    /// control.
151
    ///
152
    /// Spec: prop324 section 3.3 (TOR_VEGAS)
153
    /// C-tor: congestion_control_vegas_process_sendme() in congestion_control_vegas.c
154
156
    fn sendme_received(
155
156
        &mut self,
156
156
        state: &mut State,
157
156
        rtt: &mut RoundtripTimeEstimator,
158
156
        signals: CongestionSignals,
159
156
        clock_stall: ClockStall,
160
156
    ) -> Result<()> {
161
        // Do not update anything if we detected a clock stall or jump, as per [CLOCK_HEURISTICS].
162
156
        if clock_stall == ClockStall::Detected {
163
            // Update the inflight now that we have a SENDME and return early.
164
            self.num_inflight = self.num_inflight.saturating_sub(self.cwnd.sendme_inc());
165
            return Ok(());
166
156
        }
167

            
168
        // Update the countdown until we need to update the congestion window.
169
156
        self.num_sendme_until_cwnd_update = self.num_sendme_until_cwnd_update.saturating_sub(1);
170
        // We just got a SENDME so decrement the amount of expected SENDMEs for a cwnd.
171
156
        self.num_sendme_per_cwnd = self.num_sendme_per_cwnd.saturating_sub(1);
172

            
173
        // From here, C-tor proceeds to update the RTT and BDP (circuit estimates). The RTT is
174
        // updated before this is called and so the "rtt" object is up to date with the latest. As
175
        // for the BDP, we update it now. See C-tor congestion_control_update_circuit_estimates().
176

            
177
        // Update the BDP estimator even if the RTT estimator is not ready. If that is the case,
178
        // we'll estimate a BDP value to bootstrap.
179
156
        self.bdp.update(&self.cwnd, rtt, &signals);
180

            
181
        // Evaluate if we changed state on the blocked chan. This is done in the BDP update function
182
        // in C-tor. Instead, we do it now after the update of the BDP value.
183
156
        if rtt.is_ready() {
184
156
            if signals.channel_blocked {
185
                // Going from non blocked to block, it is an immediate congestion signal so reset the
186
                // number of sendme per cwnd because we are about to reevaluate it.
187
6
                if !self.is_blocked_on_chan {
188
                    self.num_sendme_until_cwnd_update = 0;
189
6
                }
190
            } else {
191
                // Going from blocked to non block, need to reevaluate the cwnd and so reset num
192
                // sendme.
193
150
                if self.is_blocked_on_chan {
194
                    self.num_sendme_until_cwnd_update = 0;
195
150
                }
196
            }
197
        }
198
156
        self.is_blocked_on_chan = signals.channel_blocked;
199

            
200
        // Only run the algorithm if the RTT estimator is ready or we have a blocked channel.
201
156
        if !rtt.is_ready() && !self.is_blocked_on_chan {
202
            // The inflight value can never be below a sendme_inc because every time a cell is sent,
203
            // inflight is incremented and we only end up decrementing if we receive a valid
204
            // authenticated SENDME which is always after the sendme_inc value that we get that.
205
            debug_assert!(self.num_inflight >= self.cwnd.sendme_inc());
206
            self.num_inflight = self.num_inflight.saturating_sub(self.cwnd.sendme_inc());
207
            return Ok(());
208
156
        }
209

            
210
        // The queue use is the amount in which our cwnd is above BDP;
211
        // if it is below, then 0 queue use.
212
156
        let queue_use = self.cwnd.get().saturating_sub(self.bdp.get());
213

            
214
        // Evaluate if the congestion window has became full or not.
215
156
        self.cwnd.eval_fullness(
216
156
            self.num_inflight,
217
156
            self.params.cwnd_full_gap(),
218
156
            self.params.cwnd_full_min_pct().as_percent(),
219
        );
220

            
221
        // Spec: See the pseudocode of TOR_VEGAS with RFC3742
222
156
        if state.in_slow_start() {
223
128
            if queue_use < self.params.cell_in_queue_params().gamma() && !self.is_blocked_on_chan {
224
                // If the congestion window is not fully in use, skip any increment in slow start.
225
122
                if self.cwnd.is_full() {
226
                    // This is the "Limited Slow Start" increment.
227
104
                    let inc = self
228
104
                        .cwnd
229
104
                        .rfc3742_ss_inc(self.params.cell_in_queue_params().ss_cwnd_cap());
230

            
231
                    // Check if inc is less than what we would do in steady-state avoidance. Note
232
                    // that this is likely never to happen in practice. If so, exit slow start.
233
104
                    if (inc * self.cwnd.sendme_per_cwnd())
234
104
                        <= (self.cwnd.increment() * self.cwnd.increment_rate())
235
                    {
236
                        *state = State::Steady;
237
104
                    }
238
18
                }
239
6
            } else {
240
6
                // Congestion signal: Set cwnd to gamma threshold
241
6
                self.cwnd
242
6
                    .set(self.bdp.get() + self.params.cell_in_queue_params().gamma());
243
6
                // Exit slow start due to congestion signal.
244
6
                *state = State::Steady;
245
6
            }
246

            
247
            // Max the window and exit slow start.
248
128
            if self.cwnd.get() >= self.params.ss_cwnd_max() {
249
                self.cwnd.set(self.params.ss_cwnd_max());
250
                *state = State::Steady;
251
128
            }
252
28
        } else if self.num_sendme_until_cwnd_update == 0 {
253
            // Once in steady state, we only update once per window.
254
28
            if queue_use > self.params.cell_in_queue_params().delta() {
255
4
                // Above delta threshold, drop cwnd down to the delta.
256
4
                self.cwnd.set(
257
4
                    self.bdp.get() + self.params.cell_in_queue_params().delta()
258
4
                        - self.cwnd.increment(),
259
4
                );
260
24
            } else if queue_use > self.params.cell_in_queue_params().beta()
261
20
                || self.is_blocked_on_chan
262
8
            {
263
8
                // Congestion signal: Above beta or if channel is blocked, decrement window.
264
8
                self.cwnd.dec();
265
16
            } else if self.cwnd.is_full() && queue_use < self.params.cell_in_queue_params().alpha()
266
10
            {
267
10
                // Congestion window is full and the queue usage is below alpha, increment.
268
10
                self.cwnd.inc();
269
10
            }
270
        }
271

            
272
        // Reset our counters if they reached their bottom.
273
156
        if self.num_sendme_until_cwnd_update == 0 {
274
156
            self.num_sendme_until_cwnd_update = self.cwnd.update_rate(state);
275
156
        }
276
156
        if self.num_sendme_per_cwnd == 0 {
277
26
            self.num_sendme_per_cwnd = self.cwnd.sendme_per_cwnd();
278
130
        }
279

            
280
        // Decide if enough time has passed to reset the cwnd.
281
156
        if self.params.cwnd_full_per_cwnd() != 0 {
282
156
            if self.num_sendme_per_cwnd == self.cwnd.sendme_per_cwnd() {
283
26
                self.cwnd.reset_full();
284
130
            }
285
        } else if self.num_sendme_until_cwnd_update == self.cwnd.update_rate(state) {
286
            self.cwnd.reset_full();
287
        }
288

            
289
        // Finally, update the inflight now that we have a SENDME.
290
156
        self.num_inflight = self.num_inflight.saturating_sub(self.cwnd.sendme_inc());
291
156
        Ok(())
292
156
    }
293

            
294
    fn sendme_sent(&mut self) -> Result<()> {
295
        // SENDME is on the wire, set our counter until next one.
296
        self.num_cell_until_sendme = self.cwnd.sendme_inc();
297
        Ok(())
298
    }
299

            
300
56
    fn data_received(&mut self) -> Result<bool> {
301
56
        if self.num_cell_until_sendme == 0 {
302
            // This is not a protocol violation, it is a code flow error and so don't close the
303
            // circuit by sending back an Error. Catching this prevents from sending two SENDMEs
304
            // back to back. We recover from this but scream very loudly.
305
            error_report!(internal!("Congestion control unexptected data cell"), "");
306
            return Ok(false);
307
56
        }
308

            
309
        // Decrement the expected window.
310
56
        self.num_cell_until_sendme = self.num_cell_until_sendme.saturating_sub(1);
311

            
312
        // Reaching zero, lets inform the caller a SENDME needs to be sent. This counter is reset
313
        // when the SENDME is actually sent.
314
56
        Ok(self.num_cell_until_sendme == 0)
315
56
    }
316

            
317
1200
    fn data_sent(&mut self) -> Result<()> {
318
        // This can be above cwnd because that cwnd can shrink while we are still sending data.
319
1200
        self.num_inflight = self.num_inflight.saturating_add(1);
320
1200
        Ok(())
321
1200
    }
322

            
323
    #[cfg(feature = "conflux")]
324
16
    fn inflight(&self) -> Option<u32> {
325
16
        Some(self.num_inflight)
326
16
    }
327

            
328
    #[cfg(test)]
329
12
    fn send_window(&self) -> u32 {
330
12
        self.cwnd.get()
331
12
    }
332

            
333
56
    fn algorithm(&self) -> Algorithm {
334
56
        Algorithm::Vegas(self.params)
335
56
    }
336
}
337

            
338
#[cfg(test)]
339
pub(crate) mod test {
340
    // @@ begin test lint list maintained by maint/add_warning @@
341
    #![allow(clippy::bool_assert_comparison)]
342
    #![allow(clippy::clone_on_copy)]
343
    #![allow(clippy::dbg_macro)]
344
    #![allow(clippy::mixed_attributes_style)]
345
    #![allow(clippy::print_stderr)]
346
    #![allow(clippy::print_stdout)]
347
    #![allow(clippy::single_char_pattern)]
348
    #![allow(clippy::unwrap_used)]
349
    #![allow(clippy::unchecked_time_subtraction)]
350
    #![allow(clippy::useless_vec)]
351
    #![allow(clippy::needless_pass_by_value)]
352
    #![allow(clippy::string_slice)] // See arti#2571
353
    //! <!-- @@ end test lint list maintained by maint/add_warning @@ -->
354

            
355
    use std::collections::VecDeque;
356
    use tor_units::Percentage;
357
    use web_time_compat::{Duration, Instant, InstantExt};
358

            
359
    use super::*;
360
    use crate::congestion::{
361
        params::VegasParamsBuilder,
362
        test_utils::{new_cwnd, new_rtt_estimator},
363
    };
364

            
365
    impl Vegas {
366
        /// Set the number of inflight cell.
367
        pub(crate) fn set_inflight(&mut self, v: u32) {
368
            self.num_inflight = v;
369
        }
370
        /// Return the state of the blocked on chan flag.
371
        fn is_blocked_on_chan(&self) -> bool {
372
            self.is_blocked_on_chan
373
        }
374
        /// Set the state of the blocked on chan flag.
375
        fn set_is_blocked_on_chan(&mut self, v: bool) {
376
            self.is_blocked_on_chan = v;
377
        }
378
    }
379

            
380
    /// The test vector parameters. They have the exact same name as in C-tor in order to help
381
    /// matching them and avoid confusion.
382
    #[derive(Debug)]
383
    struct TestVectorParams {
384
        // Inbound parameters.
385
        sent_usec_in: u64,
386
        got_sendme_usec_in: u64,
387
        or_conn_blocked_in: bool,
388
        inflight_in: u32,
389
        // Expected outbound parameters.
390
        ewma_rtt_usec_out: u32,
391
        min_rtt_usec_out: u32,
392
        cwnd_out: u32,
393
        in_slow_start_out: bool,
394
        cwnd_full_out: bool,
395
        blocked_chan_out: bool,
396
    }
397

            
398
    impl From<[u32; 10]> for TestVectorParams {
399
        fn from(arr: [u32; 10]) -> Self {
400
            Self {
401
                sent_usec_in: u64::from(arr[0]),
402
                got_sendme_usec_in: u64::from(arr[1]),
403
                or_conn_blocked_in: arr[2] == 1,
404
                inflight_in: arr[3],
405
                ewma_rtt_usec_out: arr[4],
406
                min_rtt_usec_out: arr[5],
407
                cwnd_out: arr[6],
408
                in_slow_start_out: arr[7] == 1,
409
                cwnd_full_out: arr[8] == 1,
410
                blocked_chan_out: arr[9] == 1,
411
            }
412
        }
413
    }
414

            
415
    struct VegasTest {
416
        params: VecDeque<TestVectorParams>,
417
        rtt: RoundtripTimeEstimator,
418
        state: State,
419
        vegas: Vegas,
420
    }
421

            
422
    impl VegasTest {
423
        fn new(vec: Vec<[u32; 10]>) -> Self {
424
            let mut params = VecDeque::new();
425
            for values in vec {
426
                params.push_back(values.into());
427
            }
428
            let state = State::default();
429
            Self {
430
                params,
431
                rtt: new_rtt_estimator(),
432
                vegas: Vegas::new(build_vegas_params(), &state, new_cwnd()),
433
                state,
434
            }
435
        }
436

            
437
        fn run_once(&mut self, p: &TestVectorParams) {
438
            eprintln!("Testing vector: {:?}", p);
439
            // Set the inflight and channel blocked value from the test vector.
440
            self.vegas.set_inflight(p.inflight_in);
441
            self.vegas.set_is_blocked_on_chan(p.or_conn_blocked_in);
442

            
443
            let now = Instant::get();
444
            self.rtt
445
                .expect_sendme(now + Duration::from_micros(p.sent_usec_in));
446
            let ret = self.rtt.update(
447
                now + Duration::from_micros(p.got_sendme_usec_in),
448
                &self.state,
449
                &self.vegas.cwnd().expect("No CWND"),
450
            );
451
            assert!(ret.is_ok());
452

            
453
            let signals = CongestionSignals::new(p.or_conn_blocked_in, 0);
454
            let clock_stall = ClockStall::NotDetected;
455
            let ret =
456
                self.vegas
457
                    .sendme_received(&mut self.state, &mut self.rtt, signals, clock_stall);
458
            assert!(ret.is_ok());
459

            
460
            assert_eq!(self.rtt.ewma_rtt_usec().unwrap(), p.ewma_rtt_usec_out);
461
            assert_eq!(self.rtt.min_rtt_usec().unwrap(), p.min_rtt_usec_out);
462
            assert_eq!(self.vegas.cwnd().expect("No CWND").get(), p.cwnd_out);
463
            assert_eq!(
464
                self.vegas.cwnd().expect("No CWND").is_full(),
465
                p.cwnd_full_out
466
            );
467
            assert_eq!(self.state.in_slow_start(), p.in_slow_start_out);
468
            assert_eq!(self.vegas.is_blocked_on_chan(), p.blocked_chan_out);
469
        }
470

            
471
        fn run(&mut self) {
472
            while let Some(param) = self.params.pop_front() {
473
                self.run_once(&param);
474
            }
475
        }
476
    }
477

            
478
    pub(crate) fn build_vegas_params() -> VegasParams {
479
        const OUTBUF_CELLS: u32 = 62;
480
        VegasParamsBuilder::default()
481
            .cell_in_queue_params(
482
                (
483
                    3 * OUTBUF_CELLS, // alpha
484
                    4 * OUTBUF_CELLS, // beta
485
                    5 * OUTBUF_CELLS, // delta
486
                    3 * OUTBUF_CELLS, // gamma
487
                    600,              // ss_cap
488
                )
489
                    .into(),
490
            )
491
            .ss_cwnd_max(5_000)
492
            .cwnd_full_gap(4)
493
            .cwnd_full_min_pct(Percentage::new(25))
494
            .cwnd_full_per_cwnd(1)
495
            .build()
496
            .expect("Unable to build Vegas parameters")
497
    }
498

            
499
    #[test]
500
    fn test_vectors() {
501
        let vec1 = vec![
502
            [100000, 200000, 0, 124, 100000, 100000, 155, 1, 0, 0],
503
            [200000, 300000, 0, 155, 100000, 100000, 186, 1, 1, 0],
504
            [350000, 500000, 0, 186, 133333, 100000, 217, 1, 1, 0],
505
            [500000, 550000, 0, 217, 77777, 77777, 248, 1, 1, 0],
506
            [600000, 700000, 0, 248, 92592, 77777, 279, 1, 1, 0],
507
            [700000, 750000, 0, 279, 64197, 64197, 310, 1, 0, 0], // Fullness expiry
508
            [750000, 875000, 0, 310, 104732, 64197, 341, 1, 1, 0],
509
            [875000, 900000, 0, 341, 51577, 51577, 372, 1, 1, 0],
510
            [900000, 950000, 0, 279, 50525, 50525, 403, 1, 1, 0],
511
            [950000, 1000000, 0, 279, 50175, 50175, 434, 1, 1, 0],
512
            [1000000, 1050000, 0, 279, 50058, 50058, 465, 1, 1, 0],
513
            [1050000, 1100000, 0, 279, 50019, 50019, 496, 1, 1, 0],
514
            [1100000, 1150000, 0, 279, 50006, 50006, 527, 1, 1, 0],
515
            [1150000, 1200000, 0, 279, 50002, 50002, 558, 1, 1, 0],
516
            [1200000, 1250000, 0, 550, 50000, 50000, 589, 1, 1, 0],
517
            [1250000, 1300000, 0, 550, 50000, 50000, 620, 1, 0, 0], // Fullness expiry
518
            [1300000, 1350000, 0, 550, 50000, 50000, 635, 1, 1, 0],
519
            [1350000, 1400000, 0, 550, 50000, 50000, 650, 1, 1, 0],
520
            [1400000, 1450000, 0, 150, 50000, 50000, 650, 1, 0, 0], // cwnd not full
521
            [1450000, 1500000, 0, 150, 50000, 50000, 650, 1, 0, 0], // cwnd not full
522
            [1500000, 1550000, 0, 550, 50000, 50000, 664, 1, 1, 0], // cwnd full
523
            [1500000, 1600000, 0, 550, 83333, 50000, 584, 0, 1, 0], // gamma exit
524
            [1600000, 1650000, 0, 550, 61111, 50000, 585, 0, 1, 0], // alpha
525
            [1650000, 1700000, 0, 550, 53703, 50000, 586, 0, 1, 0],
526
            [1700000, 1750000, 0, 100, 51234, 50000, 586, 0, 0, 0], // alpha, not full
527
            [1750000, 1900000, 0, 100, 117078, 50000, 559, 0, 0, 0], // delta, not full
528
            [1900000, 2000000, 0, 100, 105692, 50000, 558, 0, 0, 0], // beta, not full
529
            [2000000, 2075000, 0, 500, 85230, 50000, 558, 0, 1, 0], // no change
530
            [2075000, 2125000, 1, 500, 61743, 50000, 557, 0, 1, 1], // beta, blocked
531
            [2125000, 2150000, 0, 500, 37247, 37247, 558, 0, 1, 0], // alpha
532
            [2150000, 2350000, 0, 500, 145749, 37247, 451, 0, 1, 0], // delta
533
        ];
534
        VegasTest::new(vec1).run();
535

            
536
        let vec2 = vec![
537
            [100000, 200000, 0, 124, 100000, 100000, 155, 1, 0, 0],
538
            [200000, 300000, 0, 155, 100000, 100000, 186, 1, 1, 0],
539
            [350000, 500000, 0, 186, 133333, 100000, 217, 1, 1, 0],
540
            [500000, 550000, 1, 217, 77777, 77777, 403, 0, 1, 1], // ss exit, blocked
541
            [600000, 700000, 0, 248, 92592, 77777, 404, 0, 1, 0], // alpha
542
            [700000, 750000, 1, 404, 64197, 64197, 403, 0, 0, 1], // blocked beta
543
            [750000, 875000, 0, 403, 104732, 64197, 404, 0, 1, 0],
544
        ];
545
        VegasTest::new(vec2).run();
546

            
547
        let vec3 = vec![
548
            [18258527, 19002938, 0, 83, 744411, 744411, 155, 1, 0, 0],
549
            [18258580, 19254257, 0, 52, 911921, 744411, 186, 1, 1, 0],
550
            [20003224, 20645298, 0, 164, 732023, 732023, 217, 1, 1, 0],
551
            [20003367, 21021444, 0, 133, 922725, 732023, 248, 1, 1, 0],
552
            [20003845, 21265508, 0, 102, 1148683, 732023, 279, 1, 1, 0],
553
            [20003975, 21429157, 0, 71, 1333015, 732023, 310, 1, 0, 0],
554
            [20004309, 21707677, 0, 40, 1579917, 732023, 310, 1, 0, 0],
555
        ];
556
        VegasTest::new(vec3).run();
557

            
558
        let vec4 = vec![
559
            [358297091, 358854163, 0, 83, 557072, 557072, 155, 1, 0, 0],
560
            [358297649, 359123845, 0, 52, 736488, 557072, 186, 1, 1, 0],
561
            [359492879, 359995330, 0, 186, 580463, 557072, 217, 1, 1, 0],
562
            [359493043, 360489243, 0, 217, 857621, 557072, 248, 1, 1, 0],
563
            [359493232, 360489673, 0, 248, 950167, 557072, 279, 1, 1, 0],
564
            [359493795, 360489971, 0, 279, 980839, 557072, 310, 1, 0, 0],
565
            [359493918, 360490248, 0, 310, 991166, 557072, 341, 1, 1, 0],
566
            [359494029, 360716465, 0, 341, 1145346, 557072, 372, 1, 1, 0],
567
            [359996888, 360948867, 0, 372, 1016434, 557072, 403, 1, 1, 0],
568
            [359996979, 360949330, 0, 403, 973712, 557072, 434, 1, 1, 0],
569
            [360489528, 361113615, 0, 434, 740628, 557072, 465, 1, 1, 0],
570
            [360489656, 361281604, 0, 465, 774841, 557072, 496, 1, 1, 0],
571
            [360489837, 361500461, 0, 496, 932029, 557072, 482, 0, 1, 0],
572
            [360489963, 361500631, 0, 482, 984455, 557072, 482, 0, 1, 0],
573
            [360490117, 361842481, 0, 482, 1229727, 557072, 481, 0, 1, 0],
574
        ];
575
        VegasTest::new(vec4).run();
576
    }
577
}