rust
rust copied to clipboard
Tracking Issue for `unix_sigpipe`
The feature gate for the issue is #![feature(unix_sigpipe)]
.
It enables a new fn main()
attribute #[unix_sigpipe = "..."]
.
Usage
Any simple Rust program that writes a sizeable amount of data to stdout will panic if its output is limited via pipes.
fn main() {
loop {
println!("hello world");
}
}
% ./main | head -n 1
hello world
thread 'main' panicked at 'failed printing to stdout: Broken pipe (os error 32)', library/std/src/io/stdio.rs:1016:9
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrac
To prevent panicking we can use the new attribute:
#![feature(unix_sigpipe)]
#[unix_sigpipe = "sig_dfl"]
fn main() {
loop {
println!("hello world");
}
}
% ./main | head -n 1
hello world
More Info
Please refer to the unstable book section for more details. In short:
#[unix_sigpipe = "..."] |
Behaviour |
---|---|
sig_ign |
Set SIGPIPE handler to SIG_IGN before invoking fn main() . Default behaviour since 2014. |
sig_dfl |
Set SIGPIPE handler to SIG_DFL before invoking fn main() . |
inherit |
Leave SIGPIPE handler untounched before entering fn main() . |
The problem with the current SIGPIPE
code in libstd as well as several other aspects of this problem is discussed extensively at these places:
- https://github.com/rust-lang/rust/issues/62569
- https://github.com/rust-lang/rust/issues/46016
- https://rust-lang.zulipchat.com/#narrow/stream/219381-t-libs/topic/Proposal.3A.20First.20step.20towards.20solving.20the.20SIGPIPE.20problem
Naming convention
The naming follows the convention used by #![windows_subsystem = "windows|console"]
where the values "windows"
and "console"
have the same names as the actual linker flags: /SUBSYSTEM:WINDOWS
and /SUBSYSTEM:CONSOLE
.
The names sig_ign
and sig_dfl
comes from the signal handler names SIG_IGN
and SIG_DFL
.
Steps
- [x] Implement the feature:
- https://github.com/rust-lang/rust/pull/97802
- https://github.com/rust-lang/miri/pull/2532
- https://github.com/rust-lang/rust/pull/101077 (made
sigpipe::DEFAULT
distinct) - https://github.com/rust-lang/rust/pull/102110 (improved diagnostics)
- https://github.com/rust-lang/rust/pull/106092
- https://github.com/rust-lang/rust/pull/108980
- [ ] Use the attribute in a broad set of non-test-case places to learn how it works in practice.
- [x]
#[unix_sigpipe = "sig_dfl"]
- https://github.com/rust-lang/rust/pull/102587
- https://github.com/rust-lang/rust/pull/103495
- https://github.com/moonrepo/espresso/blob/e3f429b01bfd9a0a8956f11b1bc9120084c42d3c/crates/cli/src/main.rs#L18
- https://github.com/trinitronx/intro-to-rust-kvstore/blob/2c26260a837c33f193cf26cecf49279675c3a6a3/src/main.rs#L8
- [ ]
#[unix_sigpipe = "sig_ign"]
- [ ]
#[unix_sigpipe = "inherit"]
- [x]
- [x] Remove
rustc_driver::set_sigpipe_handler()
- https://github.com/rust-lang/rust/pull/103536
- [ ] Final comment period (FCP)
- https://github.com/rust-lang/rust/pull/120832
- [ ] Stabilization PR
- https://github.com/rust-lang/rust/pull/120832
Unresolved Questions That Blocks Stabilisation
-
#[unix_sigpipe = "sig_dfl"]
- None
-
#[unix_sigpipe = "sig_ign"]
- [ ] currently child processes get
SIG_IGN
, but arguably they should getSIG_DFL
since that is what most programs assume, and we explicitly made it that way before.- Note that we can implement this without running code right before
exec
, see https://github.com/rust-lang/rust/pull/121578 for the trick.
- Note that we can implement this without running code right before
- [ ] Should the current default be implemented with a noop signal handler?
- [ ] currently child processes get
-
#[unix_sigpipe = "inherit"]
- [ ] Is the name clear enough? Maybe rename to
unchanged
?
- [ ] Is the name clear enough? Maybe rename to
Unresolved Questions That Does Not Block Stabilisation
Because these questions can be resolved incrementally after stabilization.
- [ ] What is the long-term plan with regards to changing the default behaviour with regards to ignoring
SIGPIPE
, if we want to do it at all?- https://github.com/rust-lang/rust/issues/62569
Resolved Questions
- [x] We don't want to change to a
-Z unix_sigpipe
flag instead, see https://rust-lang.zulipchat.com/#narrow/stream/219381-t-libs/topic/Proposal.3A.20First.20step.20towards.20solving.20the.20SIGPIPE.20problem/near/285499895, at least not initially. - [x] Should we only have the third
sigpipe: u8
argument tofn lang_start()
on Unix platform viacfg
? Answer: No, this is not allowed, see top level comment in https://github.com/rust-lang/rust/blob/master/src/tools/tidy/src/pal.rs - [x] Should we stabilize
sig_dfl
or isinherit
andsig_ign
sufficient? Answer: There are noteworthy examples of real projects that has opted to useSIG_DFL
to solve theBrokenPipe
problem. Notably rustc itself. So if we don't stabilizesig_dfl
, such projects can't make use of our new attribute. Therefore, we also need to stabilizesig_dfl
. - [x] Should the attribute go on fn main() or on the top-level module (#![unix_sigpipe="..."])?
Answer: It makes a lot of semantic sense to have the attribute on fnmain()
, because it is a way to configure what the Rust runtime should do beforefn main()
is invoked. For libraries, no entry point code that modifiesSIGPIPE
is generated, so allowing the attribute in these situations does not make much sense. See https://github.com/rust-lang/rust/pull/101077#issuecomment-1242972299 for small-scale discussion. - [x] Can we write the code in a way that allows
lto
to remove the _signal stub code completely? With abool
it works (see https://github.com/rust-lang/rust/pull/97802#discussion_r893965842), but with the currentu8
we might need to do some tweaks. Answer: There are currently 4 values and I see no feasible way to reduce it to 2. - [x] Can and should we alter the BrokenPipe error message and make it suggest to use the new attribute? Answer: No, because that would mean we would end up giving developer advice to users that can't act on the advice.
- [x] Does this have any impact on defining a stable ABI? Answer: No, because ABI discussion are about enabling things such as calling Rust functions in binary artifacts produced by an older Rust compiler than the current one. That we changed the ABI of
fn lang_start()
is not relevant. And a stable Rust ABI is not even close (see https://github.com/rust-lang/rfcs/issues/600). - [x] Can we use
MSG_NOSIGNAL
withsend()
etc instead of settingSIGPIPE
globally? Answer: No, because there is no equivalent forwrite()
, and it would incur an extra syscall for each write-operation, which is likely to have significant performance drawbacks.
Disclaimer: I have taken the liberty to mark some questions resolved that I find unlikely to be controversial. If you would like me to create a proper discussion ticket for any of the resolved or unresolved questions, please let me know!
About tracking issues
Tracking issues are used to record the overall progress of implementation. They are also used as hubs connecting to other relevant issues, e.g., bugs or open design questions. A tracking issue is however not meant for large scale discussion, questions, or bug reports about a feature. Instead, open a dedicated issue for the specific matter and add the relevant feature gate label.
@rustbot label +T-libs