Skip to content

Commit c186f7c

Browse files
committed
Auto merge of #96455 - dtolnay:writetmp, r=m-ou-se
Make write/print macros eagerly drop temporaries This PR fixes the 2 regressions in #96434 (`println` and `eprintln`) and changes all the other similar macros (`write`, `writeln`, `print`, `eprint`) to match the old pre-#94868 behavior of `println` and `eprintln`. argument position | before #94868 | after #94868 | after this PR --- |:---:|:---:|:---: `write!($tmp, "…", …)` | :rage: | :rage: | :smiley_cat: `write!(…, "…", $tmp)` | :rage: | :rage: | :smiley_cat: `writeln!($tmp, "…", …)` | :rage: | :rage: | :smiley_cat: `writeln!(…, "…", $tmp)` | :rage: | :rage: | :smiley_cat: `print!("…", $tmp)` | :rage: | :rage: | :smiley_cat: `println!("…", $tmp)` | :smiley_cat: | :rage: | :smiley_cat: `eprint!("…", $tmp)` | :rage: | :rage: | :smiley_cat: `eprintln!("…", $tmp)` | :smiley_cat: | :rage: | :smiley_cat: `panic!("…", $tmp)` | :smiley_cat: | :smiley_cat: | :smiley_cat: Example of code that is affected by this change: ```rust use std::sync::Mutex; fn main() { let mutex = Mutex::new(0); print!("{}", mutex.lock().unwrap()) /* no semicolon */ } ``` You can see several real-world examples like this in the Crater links at the top of #96434. This code failed to compile prior to this PR as follows, but works after this PR. ```console error[E0597]: `mutex` does not live long enough --> src/main.rs:5:18 | 5 | print!("{}", mutex.lock().unwrap()) /* no semicolon */ | ^^^^^^^^^^^^--------- | | | borrowed value does not live long enough | a temporary with access to the borrow is created here ... 6 | } | - | | | `mutex` dropped here while still borrowed | ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `MutexGuard` ```
2 parents d125574 + a610098 commit c186f7c

File tree

4 files changed

+118
-14
lines changed

4 files changed

+118
-14
lines changed

library/core/src/macros/mod.rs

+8-6
Original file line numberDiff line numberDiff line change
@@ -496,9 +496,10 @@ macro_rules! r#try {
496496
#[stable(feature = "rust1", since = "1.0.0")]
497497
#[cfg_attr(not(test), rustc_diagnostic_item = "write_macro")]
498498
macro_rules! write {
499-
($dst:expr, $($arg:tt)*) => {
500-
$dst.write_fmt($crate::format_args!($($arg)*))
501-
};
499+
($dst:expr, $($arg:tt)*) => {{
500+
let result = $dst.write_fmt($crate::format_args!($($arg)*));
501+
result
502+
}};
502503
}
503504

504505
/// Write formatted data into a buffer, with a newline appended.
@@ -553,9 +554,10 @@ macro_rules! writeln {
553554
($dst:expr $(,)?) => {
554555
$crate::write!($dst, "\n")
555556
};
556-
($dst:expr, $($arg:tt)*) => {
557-
$dst.write_fmt($crate::format_args_nl!($($arg)*))
558-
};
557+
($dst:expr, $($arg:tt)*) => {{
558+
let result = $dst.write_fmt($crate::format_args_nl!($($arg)*));
559+
result
560+
}};
559561
}
560562

561563
/// Indicates unreachable code.

library/std/src/macros.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -62,9 +62,9 @@ macro_rules! panic {
6262
#[cfg_attr(not(test), rustc_diagnostic_item = "print_macro")]
6363
#[allow_internal_unstable(print_internals)]
6464
macro_rules! print {
65-
($($arg:tt)*) => {
66-
$crate::io::_print($crate::format_args!($($arg)*))
67-
};
65+
($($arg:tt)*) => {{
66+
$crate::io::_print($crate::format_args!($($arg)*));
67+
}};
6868
}
6969

7070
/// Prints to the standard output, with a newline.
@@ -133,9 +133,9 @@ macro_rules! println {
133133
#[cfg_attr(not(test), rustc_diagnostic_item = "eprint_macro")]
134134
#[allow_internal_unstable(print_internals)]
135135
macro_rules! eprint {
136-
($($arg:tt)*) => {
137-
$crate::io::_eprint($crate::format_args!($($arg)*))
138-
};
136+
($($arg:tt)*) => {{
137+
$crate::io::_eprint($crate::format_args!($($arg)*));
138+
}};
139139
}
140140

141141
/// Prints to the standard error, with a newline.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
// check-pass
2+
3+
use std::fmt::{self, Display};
4+
5+
struct Mutex;
6+
7+
impl Mutex {
8+
fn lock(&self) -> MutexGuard {
9+
MutexGuard(self)
10+
}
11+
}
12+
13+
struct MutexGuard<'a>(&'a Mutex);
14+
15+
impl<'a> Drop for MutexGuard<'a> {
16+
fn drop(&mut self) {
17+
// Empty but this is a necessary part of the repro. Otherwise borrow
18+
// checker is fine with 'a dangling at the time that MutexGuard goes out
19+
// of scope.
20+
}
21+
}
22+
23+
impl<'a> MutexGuard<'a> {
24+
fn write_fmt(&self, _args: fmt::Arguments) {}
25+
}
26+
27+
impl<'a> Display for MutexGuard<'a> {
28+
fn fmt(&self, _formatter: &mut fmt::Formatter) -> fmt::Result {
29+
Ok(())
30+
}
31+
}
32+
33+
fn main() {
34+
let _write = {
35+
let out = Mutex;
36+
let mutex = Mutex;
37+
write!(out.lock(), "{}", mutex.lock()) /* no semicolon */
38+
};
39+
40+
let _writeln = {
41+
let out = Mutex;
42+
let mutex = Mutex;
43+
writeln!(out.lock(), "{}", mutex.lock()) /* no semicolon */
44+
};
45+
46+
let _print = {
47+
let mutex = Mutex;
48+
print!("{}", mutex.lock()) /* no semicolon */
49+
};
50+
51+
let _println = {
52+
let mutex = Mutex;
53+
println!("{}", mutex.lock()) /* no semicolon */
54+
};
55+
56+
let _eprint = {
57+
let mutex = Mutex;
58+
eprint!("{}", mutex.lock()) /* no semicolon */
59+
};
60+
61+
let _eprintln = {
62+
let mutex = Mutex;
63+
eprintln!("{}", mutex.lock()) /* no semicolon */
64+
};
65+
66+
let _panic = {
67+
let mutex = Mutex;
68+
panic!("{}", mutex.lock()) /* no semicolon */
69+
};
70+
}

src/tools/clippy/clippy_lints/src/explicit_write.rs

+34-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ use clippy_utils::source::snippet_with_applicability;
44
use clippy_utils::{is_expn_of, match_function_call, paths};
55
use if_chain::if_chain;
66
use rustc_errors::Applicability;
7-
use rustc_hir::{Expr, ExprKind};
7+
use rustc_hir::def::Res;
8+
use rustc_hir::{BindingAnnotation, Block, BlockCheckMode, Expr, ExprKind, Node, PatKind, QPath, Stmt, StmtKind};
89
use rustc_lint::{LateContext, LateLintPass};
910
use rustc_session::{declare_lint_pass, declare_tool_lint};
1011
use rustc_span::sym;
@@ -39,7 +40,7 @@ impl<'tcx> LateLintPass<'tcx> for ExplicitWrite {
3940
if let ExprKind::MethodCall(unwrap_fun, [write_call], _) = expr.kind;
4041
if unwrap_fun.ident.name == sym::unwrap;
4142
// match call to write_fmt
42-
if let ExprKind::MethodCall(write_fun, [write_recv, write_arg], _) = write_call.kind;
43+
if let ExprKind::MethodCall(write_fun, [write_recv, write_arg], _) = look_in_block(cx, &write_call.kind);
4344
if write_fun.ident.name == sym!(write_fmt);
4445
// match calls to std::io::stdout() / std::io::stderr ()
4546
if let Some(dest_name) = if match_function_call(cx, write_recv, &paths::STDOUT).is_some() {
@@ -100,3 +101,34 @@ impl<'tcx> LateLintPass<'tcx> for ExplicitWrite {
100101
}
101102
}
102103
}
104+
105+
/// If `kind` is a block that looks like `{ let result = $expr; result }` then
106+
/// returns $expr. Otherwise returns `kind`.
107+
fn look_in_block<'tcx, 'hir>(cx: &LateContext<'tcx>, kind: &'tcx ExprKind<'hir>) -> &'tcx ExprKind<'hir> {
108+
if_chain! {
109+
if let ExprKind::Block(block, _label @ None) = kind;
110+
if let Block {
111+
stmts: [Stmt { kind: StmtKind::Local(local), .. }],
112+
expr: Some(expr_end_of_block),
113+
rules: BlockCheckMode::DefaultBlock,
114+
..
115+
} = block;
116+
117+
// Find id of the local that expr_end_of_block resolves to
118+
if let ExprKind::Path(QPath::Resolved(None, expr_path)) = expr_end_of_block.kind;
119+
if let Res::Local(expr_res) = expr_path.res;
120+
if let Some(Node::Binding(res_pat)) = cx.tcx.hir().find(expr_res);
121+
122+
// Find id of the local we found in the block
123+
if let PatKind::Binding(BindingAnnotation::Unannotated, local_hir_id, _ident, None) = local.pat.kind;
124+
125+
// If those two are the same hir id
126+
if res_pat.hir_id == local_hir_id;
127+
128+
if let Some(init) = local.init;
129+
then {
130+
return &init.kind;
131+
}
132+
}
133+
kind
134+
}

0 commit comments

Comments
 (0)