Skip to content

Commit 3f977ba

Browse files
committed
Auto merge of #43728 - zackmdavis:fnused, r=eddyb
#[must_use] for functions This implements [RFC 1940](rust-lang/rfcs#1940). The RFC and discussion thereof seem to suggest that tagging `PartialEq::eq` and friends as `#[must_use]` would automatically lint for unused comparisons, but it doesn't work out that way (at least the way I've implemented it): unused `.eq` method calls get linted, but not `==` expressions. (The lint operates on the HIR, which sees binary operations as their own thing, even if they ultimately just call `.eq` _&c._.) What do _you_ think?? Resolves #43302.
2 parents 0f9317d + f5ac228 commit 3f977ba

File tree

4 files changed

+79
-9
lines changed

4 files changed

+79
-9
lines changed

src/libcore/cmp.rs

+7
Original file line numberDiff line numberDiff line change
@@ -110,11 +110,13 @@ use self::Ordering::*;
110110
pub trait PartialEq<Rhs: ?Sized = Self> {
111111
/// This method tests for `self` and `other` values to be equal, and is used
112112
/// by `==`.
113+
#[must_use]
113114
#[stable(feature = "rust1", since = "1.0.0")]
114115
fn eq(&self, other: &Rhs) -> bool;
115116

116117
/// This method tests for `!=`.
117118
#[inline]
119+
#[must_use]
118120
#[stable(feature = "rust1", since = "1.0.0")]
119121
fn ne(&self, other: &Rhs) -> bool { !self.eq(other) }
120122
}
@@ -625,6 +627,7 @@ pub trait PartialOrd<Rhs: ?Sized = Self>: PartialEq<Rhs> {
625627
/// let result = std::f64::NAN.partial_cmp(&1.0);
626628
/// assert_eq!(result, None);
627629
/// ```
630+
#[must_use]
628631
#[stable(feature = "rust1", since = "1.0.0")]
629632
fn partial_cmp(&self, other: &Rhs) -> Option<Ordering>;
630633

@@ -640,6 +643,7 @@ pub trait PartialOrd<Rhs: ?Sized = Self>: PartialEq<Rhs> {
640643
/// assert_eq!(result, false);
641644
/// ```
642645
#[inline]
646+
#[must_use]
643647
#[stable(feature = "rust1", since = "1.0.0")]
644648
fn lt(&self, other: &Rhs) -> bool {
645649
match self.partial_cmp(other) {
@@ -661,6 +665,7 @@ pub trait PartialOrd<Rhs: ?Sized = Self>: PartialEq<Rhs> {
661665
/// assert_eq!(result, true);
662666
/// ```
663667
#[inline]
668+
#[must_use]
664669
#[stable(feature = "rust1", since = "1.0.0")]
665670
fn le(&self, other: &Rhs) -> bool {
666671
match self.partial_cmp(other) {
@@ -681,6 +686,7 @@ pub trait PartialOrd<Rhs: ?Sized = Self>: PartialEq<Rhs> {
681686
/// assert_eq!(result, false);
682687
/// ```
683688
#[inline]
689+
#[must_use]
684690
#[stable(feature = "rust1", since = "1.0.0")]
685691
fn gt(&self, other: &Rhs) -> bool {
686692
match self.partial_cmp(other) {
@@ -702,6 +708,7 @@ pub trait PartialOrd<Rhs: ?Sized = Self>: PartialEq<Rhs> {
702708
/// assert_eq!(result, true);
703709
/// ```
704710
#[inline]
711+
#[must_use]
705712
#[stable(feature = "rust1", since = "1.0.0")]
706713
fn ge(&self, other: &Rhs) -> bool {
707714
match self.partial_cmp(other) {

src/librustc_lint/unused.rs

+25-9
Original file line numberDiff line numberDiff line change
@@ -145,22 +145,38 @@ impl<'a, 'tcx> LateLintPass<'a, 'tcx> for UnusedResults {
145145
}
146146

147147
let t = cx.tables.expr_ty(&expr);
148-
let warned = match t.sty {
149-
ty::TyTuple(ref tys, _) if tys.is_empty() => return,
150-
ty::TyNever => return,
151-
ty::TyBool => return,
152-
ty::TyAdt(def, _) => check_must_use(cx, def.did, s.span),
148+
let ty_warned = match t.sty {
149+
ty::TyAdt(def, _) => check_must_use(cx, def.did, s.span, ""),
153150
_ => false,
154151
};
155-
if !warned {
152+
153+
let mut fn_warned = false;
154+
let maybe_def = match expr.node {
155+
hir::ExprCall(ref callee, _) => {
156+
match callee.node {
157+
hir::ExprPath(ref qpath) => Some(cx.tables.qpath_def(qpath, callee.id)),
158+
_ => None
159+
}
160+
},
161+
hir::ExprMethodCall(..) => {
162+
cx.tables.type_dependent_defs.get(&expr.id).cloned()
163+
},
164+
_ => { None }
165+
};
166+
if let Some(def) = maybe_def {
167+
let def_id = def.def_id();
168+
fn_warned = check_must_use(cx, def_id, s.span, "return value of ");
169+
}
170+
171+
if !(ty_warned || fn_warned) {
156172
cx.span_lint(UNUSED_RESULTS, s.span, "unused result");
157173
}
158174

159-
fn check_must_use(cx: &LateContext, def_id: DefId, sp: Span) -> bool {
175+
fn check_must_use(cx: &LateContext, def_id: DefId, sp: Span, describe_path: &str) -> bool {
160176
for attr in cx.tcx.get_attrs(def_id).iter() {
161177
if attr.check_name("must_use") {
162-
let mut msg = format!("unused `{}` which must be used",
163-
cx.tcx.item_path_str(def_id));
178+
let mut msg = format!("unused {}`{}` which must be used",
179+
describe_path, cx.tcx.item_path_str(def_id));
164180
// check for #[must_use="..."]
165181
if let Some(s) = attr.value_str() {
166182
msg.push_str(": ");

src/test/ui/lint/fn_must_use.rs

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
12+
struct MyStruct {
13+
n: usize
14+
}
15+
16+
impl MyStruct {
17+
#[must_use]
18+
fn need_to_use_this_method_value(&self) -> usize {
19+
self.n
20+
}
21+
}
22+
23+
#[must_use="it's important"]
24+
fn need_to_use_this_value() -> bool {
25+
false
26+
}
27+
28+
fn main() {
29+
need_to_use_this_value();
30+
31+
let m = MyStruct { n: 2 };
32+
m.need_to_use_this_method_value();
33+
}

src/test/ui/lint/fn_must_use.stderr

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
warning: unused return value of `need_to_use_this_value` which must be used: it's important
2+
--> $DIR/fn_must_use.rs:29:5
3+
|
4+
29 | need_to_use_this_value();
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^
6+
|
7+
= note: #[warn(unused_must_use)] on by default
8+
9+
warning: unused return value of `MyStruct::need_to_use_this_method_value` which must be used
10+
--> $DIR/fn_must_use.rs:32:5
11+
|
12+
32 | m.need_to_use_this_method_value();
13+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
14+

0 commit comments

Comments
 (0)