Skip to content

Commit d36ebb7

Browse files
committed
Block shutdown until all pending messages are sent (Fixes librespot-org#27).
Ideally the task within our "mdns-responder" thread would block until all the futures actually return Poll::Ready but I can't get that to work, hence the explicit extra blocking I have added.
1 parent 4829c5c commit d36ebb7

File tree

2 files changed

+33
-14
lines changed

2 files changed

+33
-14
lines changed

src/fsm.rs

+21-6
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ use std::{
1111
pin::Pin,
1212
task::{Context, Poll},
1313
};
14+
use std::sync::mpsc::{sync_channel, SyncSender, Receiver};
1415

1516
use tokio::{net::UdpSocket, sync::mpsc};
1617

@@ -31,6 +32,7 @@ pub enum Command {
3132
}
3233

3334
pub struct FSM<AF: AddressFamily> {
35+
shutdown_done: Receiver<bool>,
3436
socket: UdpSocket,
3537
services: Services,
3638
commands: mpsc::UnboundedReceiver<Command>,
@@ -40,21 +42,23 @@ pub struct FSM<AF: AddressFamily> {
4042

4143
impl<AF: AddressFamily> FSM<AF> {
4244
// Will panic if called from outside the context of a runtime
43-
pub fn new(services: &Services) -> io::Result<(FSM<AF>, mpsc::UnboundedSender<Command>)> {
45+
pub fn new(services: &Services) -> io::Result<(FSM<AF>, mpsc::UnboundedSender<Command>, SyncSender<bool>)> {
4446
let std_socket = AF::bind()?;
4547
let socket = UdpSocket::from_std(std_socket)?;
4648

4749
let (tx, rx) = mpsc::unbounded_channel();
50+
let (tx_fin, rx_fin) = sync_channel::<bool>(0);
4851

4952
let fsm = FSM {
53+
shutdown_done: rx_fin,
5054
socket: socket,
5155
services: services.clone(),
5256
commands: rx,
5357
outgoing: VecDeque::new(),
5458
_af: PhantomData,
5559
};
5660

57-
Ok((fsm, tx))
61+
Ok((fsm, tx, tx_fin))
5862
}
5963

6064
fn recv_packets(&mut self, cx: &mut Context) -> io::Result<()> {
@@ -221,9 +225,13 @@ impl<AF: Unpin + AddressFamily> Future for FSM<AF> {
221225
type Output = ();
222226
fn poll(self: Pin<&mut Self>, cx: &mut Context) -> Poll<()> {
223227
let pinned = Pin::get_mut(self);
228+
let mut shutdown = false;
224229
while let Poll::Ready(cmd) = Pin::new(&mut pinned.commands).poll_recv(cx) {
225230
match cmd {
226-
Some(Command::Shutdown) => return Poll::Ready(()),
231+
Some(Command::Shutdown) => {
232+
shutdown = true;
233+
break;
234+
}
227235
Some(Command::SendUnsolicited {
228236
svc,
229237
ttl,
@@ -238,9 +246,11 @@ impl<AF: Unpin + AddressFamily> Future for FSM<AF> {
238246
}
239247
}
240248

241-
match pinned.recv_packets(cx) {
242-
Ok(_) => (),
243-
Err(e) => error!("ResponderRecvPacket Error: {:?}", e),
249+
if !shutdown {
250+
match pinned.recv_packets(cx) {
251+
Ok(_) => (),
252+
Err(e) => error!("ResponderRecvPacket Error: {:?}", e),
253+
}
244254
}
245255

246256
while let Some((ref response, addr)) = pinned.outgoing.pop_front() {
@@ -255,6 +265,11 @@ impl<AF: Unpin + AddressFamily> Future for FSM<AF> {
255265
}
256266
}
257267

268+
if shutdown {
269+
pinned.shutdown_done.recv().unwrap();
270+
return Poll::Ready(());
271+
}
272+
258273
Poll::Pending
259274
}
260275
}

src/lib.rs

+12-8
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ use std::future::Future;
55
use std::io;
66
use std::marker::Unpin;
77
use std::sync::{Arc, RwLock};
8+
use std::sync::mpsc::SyncSender;
89

910
use std::thread;
1011
use tokio::{runtime::Handle, sync::mpsc};
@@ -105,15 +106,15 @@ impl Responder {
105106
let v4 = FSM::<Inet>::new(&services);
106107
let v6 = FSM::<Inet6>::new(&services);
107108

108-
let (task, commands): (ResponderTask, _) = match (v4, v6) {
109-
(Ok((v4_task, v4_command)), Ok((v6_task, v6_command))) => {
109+
let (task, commands, shutdown): (ResponderTask, _, _) = match (v4, v6) {
110+
(Ok((v4_task, v4_command, v4_shutdown)), Ok((v6_task, v6_command, v6_shutdown))) => {
110111
let tasks = future::join(v4_task, v6_task).map(|((), ())| ());
111-
(Box::new(tasks), vec![v4_command, v6_command])
112+
(Box::new(tasks), vec![v4_command, v6_command], vec![v4_shutdown, v6_shutdown])
112113
}
113114

114-
(Ok((v4_task, v4_command)), Err(err)) => {
115+
(Ok((v4_task, v4_command, v4_shutdown)), Err(err)) => {
115116
warn!("Failed to register IPv6 receiver: {:?}", err);
116-
(Box::new(v4_task), vec![v4_command])
117+
(Box::new(v4_task), vec![v4_command], vec![v4_shutdown])
117118
}
118119

119120
(Err(err), _) => return Err(err),
@@ -123,7 +124,7 @@ impl Responder {
123124
let responder = Responder {
124125
services: services,
125126
commands: RefCell::new(commands.clone()),
126-
shutdown: Arc::new(Shutdown(commands)),
127+
shutdown: Arc::new(Shutdown(commands, shutdown)),
127128
};
128129

129130
Ok((responder, task))
@@ -197,12 +198,15 @@ impl Drop for Service {
197198
}
198199
}
199200

200-
struct Shutdown(CommandSender);
201+
struct Shutdown(CommandSender, Vec<SyncSender<bool>>);
201202

202203
impl Drop for Shutdown {
203204
fn drop(&mut self) {
204205
self.0.send_shutdown();
205-
// TODO wait for tasks to shutdown
206+
for x in self.1.iter() {
207+
// Blocks until service has acked the shutdown.
208+
let _ = x.send(false).unwrap();
209+
}
206210
}
207211
}
208212

0 commit comments

Comments
 (0)