Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

server serial task's send-to-listeners future is not cancel-safe #650

Open
gjcolombo opened this issue Feb 23, 2024 · 1 comment · May be fixed by #830
Open

server serial task's send-to-listeners future is not cancel-safe #650

gjcolombo opened this issue Feb 23, 2024 · 1 comment · May be fixed by #830
Labels
server Related specifically to the Propolis server API and its VM management functions.

Comments

@gjcolombo
Copy link
Contributor

Discovered this during some ad hoc testing with multiple connections to a single guest's serial console, where I observed some data being sent a second time to a listener that had already received it.

If the serial console task has read some data from the guest that needs to be written to listeners, it generates a future to do that writing and then includes it in the main select! that determines the task's next action (lines 116-125 below where cur_output is Some):

let (uart_read, ws_send) =
if ws_sinks.is_empty() || cur_output.is_none() {
(serial.read_source(&mut output).fuse(), Fuse::terminated())
} else {
let range = cur_output.clone().unwrap();
(
Fuse::terminated(),
if !ws_sinks.is_empty() {
futures::stream::iter(
ws_sinks.iter_mut().zip(std::iter::repeat(
Vec::from(&output[range]),
)),
)
.for_each_concurrent(4, |((_i, ws), bin)| {
ws.send(Message::binary(bin)).map(|_| ())
})
.fuse()
} else {
Fuse::terminated()
},
)
};

cur_output is only set to None if this future is the one chosen by select!:

// Transmit bytes from the UART through the WS
_ = ws_send => {
probes::serial_uart_out!(|| {});
cur_output = None;
}

But if this branch is not chosen, for_each_concurrent may have sent the current output bytes to some listeners and not to others. (The underlying websocket send may not be cancel-safe either but I haven't looked at its implementation.) This will cause the same data to be re-sent in the next loop iteration.

An approach along the lines suggested in RFD 400 section 5.2 might help here (and other approaches may work too):

  • If there's already a send-to-clients future active, try to resolve it
  • Else if there's no such future but there is some data waiting to be sent, create a new future to try to send that data
  • Else try to read more data from the guest
@gjcolombo gjcolombo added the server Related specifically to the Propolis server API and its VM management functions. label Feb 23, 2024
@gjcolombo
Copy link
Contributor Author

I'm also a little unsure about the read-from-clients side of this:

let (ws_recv, uart_write) = match &cur_input {
None => (
if !ws_streams.is_empty() {
futures::stream::iter(ws_streams.iter_mut())
.for_each_concurrent(4, |(i, ws)| {
ws.next()
.then(|msg| send_ch.send((*i, msg)))
.map(|_| ())
})
.fuse()

If ws.next() reads a message from a client, but another select! branch is chosen before the message can be sent through send_ch, I think the message is lost (per Sender::send's documentation of its cancel safety).

@gjcolombo gjcolombo linked a pull request Jan 10, 2025 that will close this issue
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
server Related specifically to the Propolis server API and its VM management functions.
Projects
None yet
Development

Successfully merging a pull request may close this issue.

1 participant