|
1 | 1 | use futures::channel::mpsc;
|
2 | 2 | use futures::executor::block_on;
|
| 3 | +use futures::future::Future; |
3 | 4 | use futures::sink::SinkExt;
|
4 | 5 | use futures::stream::StreamExt;
|
5 |
| -use std::sync::Arc; |
| 6 | +use futures::task::{Context, Poll}; |
| 7 | +use std::pin::Pin; |
| 8 | +use std::sync::{Arc, Weak}; |
6 | 9 | use std::thread;
|
| 10 | +use std::time::{Duration, Instant}; |
7 | 11 |
|
8 | 12 | #[test]
|
9 | 13 | fn smoke() {
|
@@ -142,3 +146,133 @@ fn single_receiver_drop_closes_channel_and_drains() {
|
142 | 146 | assert!(sender.is_closed());
|
143 | 147 | }
|
144 | 148 | }
|
| 149 | + |
| 150 | +// Stress test that `try_send()`s occurring concurrently with receiver |
| 151 | +// close/drops don't appear as successful sends. |
| 152 | +#[test] |
| 153 | +fn stress_try_send_as_receiver_closes() { |
| 154 | + const AMT: usize = 10000; |
| 155 | + // To provide variable timing characteristics (in the hopes of |
| 156 | + // reproducing the collision that leads to a race), we busy-re-poll |
| 157 | + // the test MPSC receiver a variable number of times before actually |
| 158 | + // stopping. We vary this countdown between 1 and the following |
| 159 | + // value. |
| 160 | + const MAX_COUNTDOWN: usize = 20; |
| 161 | + // When we detect that a successfully sent item is still in the |
| 162 | + // queue after a disconnect, we spin for up to 100ms to confirm that |
| 163 | + // it is a persistent condition and not a concurrency illusion. |
| 164 | + const SPIN_TIMEOUT_S: u64 = 10; |
| 165 | + const SPIN_SLEEP_MS: u64 = 10; |
| 166 | + struct TestRx { |
| 167 | + rx: mpsc::Receiver<Arc<()>>, |
| 168 | + // The number of times to query `rx` before dropping it. |
| 169 | + poll_count: usize |
| 170 | + } |
| 171 | + struct TestTask { |
| 172 | + command_rx: mpsc::Receiver<TestRx>, |
| 173 | + test_rx: Option<mpsc::Receiver<Arc<()>>>, |
| 174 | + countdown: usize, |
| 175 | + } |
| 176 | + impl TestTask { |
| 177 | + /// Create a new TestTask |
| 178 | + fn new() -> (TestTask, mpsc::Sender<TestRx>) { |
| 179 | + let (command_tx, command_rx) = mpsc::channel::<TestRx>(0); |
| 180 | + ( |
| 181 | + TestTask { |
| 182 | + command_rx, |
| 183 | + test_rx: None, |
| 184 | + countdown: 0, // 0 means no countdown is in progress. |
| 185 | + }, |
| 186 | + command_tx, |
| 187 | + ) |
| 188 | + } |
| 189 | + } |
| 190 | + impl Future for TestTask { |
| 191 | + type Output = (); |
| 192 | + |
| 193 | + fn poll( |
| 194 | + mut self: Pin<&mut Self>, |
| 195 | + cx: &mut Context<'_>, |
| 196 | + ) -> Poll<Self::Output> { |
| 197 | + // Poll the test channel, if one is present. |
| 198 | + if let Some(rx) = &mut self.test_rx { |
| 199 | + if let Poll::Ready(v) = rx.poll_next_unpin(cx) { |
| 200 | + let _ = v.expect("test finished unexpectedly!"); |
| 201 | + } |
| 202 | + self.countdown -= 1; |
| 203 | + // Busy-poll until the countdown is finished. |
| 204 | + cx.waker().wake_by_ref(); |
| 205 | + } |
| 206 | + // Accept any newly submitted MPSC channels for testing. |
| 207 | + match self.command_rx.poll_next_unpin(cx) { |
| 208 | + Poll::Ready(Some(TestRx { rx, poll_count })) => { |
| 209 | + self.test_rx = Some(rx); |
| 210 | + self.countdown = poll_count; |
| 211 | + cx.waker().wake_by_ref(); |
| 212 | + }, |
| 213 | + Poll::Ready(None) => return Poll::Ready(()), |
| 214 | + Poll::Pending => {}, |
| 215 | + } |
| 216 | + if self.countdown == 0 { |
| 217 | + // Countdown complete -- drop the Receiver. |
| 218 | + self.test_rx = None; |
| 219 | + } |
| 220 | + Poll::Pending |
| 221 | + } |
| 222 | + } |
| 223 | + let (f, mut cmd_tx) = TestTask::new(); |
| 224 | + let bg = thread::spawn(move || block_on(f)); |
| 225 | + for i in 0..AMT { |
| 226 | + let (mut test_tx, rx) = mpsc::channel(0); |
| 227 | + let poll_count = i % MAX_COUNTDOWN; |
| 228 | + cmd_tx.try_send(TestRx { rx, poll_count }).unwrap(); |
| 229 | + let mut prev_weak: Option<Weak<()>> = None; |
| 230 | + let mut attempted_sends = 0; |
| 231 | + let mut successful_sends = 0; |
| 232 | + loop { |
| 233 | + // Create a test item. |
| 234 | + let item = Arc::new(()); |
| 235 | + let weak = Arc::downgrade(&item); |
| 236 | + match test_tx.try_send(item) { |
| 237 | + Ok(_) => { |
| 238 | + prev_weak = Some(weak); |
| 239 | + successful_sends += 1; |
| 240 | + } |
| 241 | + Err(ref e) if e.is_full() => {} |
| 242 | + Err(ref e) if e.is_disconnected() => { |
| 243 | + // Test for evidence of the race condition. |
| 244 | + if let Some(prev_weak) = prev_weak { |
| 245 | + if prev_weak.upgrade().is_some() { |
| 246 | + // The previously sent item is still allocated. |
| 247 | + // However, there appears to be some aspect of the |
| 248 | + // concurrency that can legitimately cause the Arc |
| 249 | + // to be momentarily valid. Spin for up to 100ms |
| 250 | + // waiting for the previously sent item to be |
| 251 | + // dropped. |
| 252 | + let t0 = Instant::now(); |
| 253 | + let mut spins = 0; |
| 254 | + loop { |
| 255 | + if prev_weak.upgrade().is_none() { |
| 256 | + break; |
| 257 | + } |
| 258 | + assert!(t0.elapsed() < Duration::from_secs(SPIN_TIMEOUT_S), |
| 259 | + "item not dropped on iteration {} after \ |
| 260 | + {} sends ({} successful). spin=({})", |
| 261 | + i, attempted_sends, successful_sends, spins |
| 262 | + ); |
| 263 | + spins += 1; |
| 264 | + thread::sleep(Duration::from_millis(SPIN_SLEEP_MS)); |
| 265 | + } |
| 266 | + } |
| 267 | + } |
| 268 | + break; |
| 269 | + } |
| 270 | + Err(ref e) => panic!("unexpected error: {}", e), |
| 271 | + } |
| 272 | + attempted_sends += 1; |
| 273 | + } |
| 274 | + } |
| 275 | + drop(cmd_tx); |
| 276 | + bg.join() |
| 277 | + .expect("background thread join"); |
| 278 | +} |
0 commit comments