2020-04-27 00:12:51 +02:00
|
|
|
// edition:2018
|
2019-04-20 09:33:13 +02:00
|
|
|
// aux-build:proc_macro_derive.rs
|
|
|
|
|
2019-11-06 10:36:04 -08:00
|
|
|
#![feature(rustc_private)]
|
2018-07-28 17:34:52 +02:00
|
|
|
#![warn(clippy::all)]
|
2020-10-08 23:02:16 +02:00
|
|
|
#![allow(clippy::blacklisted_name, clippy::eq_op)]
|
2018-07-28 17:34:52 +02:00
|
|
|
#![warn(clippy::used_underscore_binding)]
|
2016-02-22 15:42:24 +01:00
|
|
|
|
2019-04-20 09:33:13 +02:00
|
|
|
#[macro_use]
|
|
|
|
extern crate proc_macro_derive;
|
|
|
|
|
|
|
|
// This should not trigger the lint. There's underscore binding inside the external derive that
|
|
|
|
// would trigger the `used_underscore_binding` lint.
|
|
|
|
#[derive(DeriveSomething)]
|
|
|
|
struct Baz;
|
|
|
|
|
2016-06-15 16:27:56 +02:00
|
|
|
macro_rules! test_macro {
|
|
|
|
() => {{
|
|
|
|
let _foo = 42;
|
|
|
|
_foo + 1
|
2018-12-09 23:26:16 +01:00
|
|
|
}};
|
2016-06-15 16:27:56 +02:00
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we lint if we use a binding with a single leading underscore
|
2015-12-18 16:04:33 -08:00
|
|
|
fn prefix_underscore(_foo: u32) -> u32 {
|
2017-02-08 14:58:07 +01:00
|
|
|
_foo + 1
|
2015-12-10 12:54:43 -08:00
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we lint if we use a `_`-variable defined outside within a macro expansion
|
2019-05-11 20:40:05 -07:00
|
|
|
fn in_macro_or_desugar(_foo: u32) {
|
2016-06-15 16:27:56 +02:00
|
|
|
println!("{}", _foo);
|
|
|
|
assert_eq!(_foo, _foo);
|
2017-02-08 14:58:07 +01:00
|
|
|
|
2016-06-15 16:27:56 +02:00
|
|
|
test_macro!() + 1;
|
2015-12-21 01:03:12 -08:00
|
|
|
}
|
|
|
|
|
2015-12-18 16:29:22 -08:00
|
|
|
// Struct for testing use of fields prefixed with an underscore
|
|
|
|
struct StructFieldTest {
|
|
|
|
_underscore_field: u32,
|
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we lint the use of a struct field which is prefixed with an underscore
|
2015-12-18 16:29:22 -08:00
|
|
|
fn in_struct_field() {
|
|
|
|
let mut s = StructFieldTest { _underscore_field: 0 };
|
2017-02-08 14:58:07 +01:00
|
|
|
s._underscore_field += 1;
|
2015-12-18 16:29:22 -08:00
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we do not lint if the underscore is not a prefix
|
2015-12-12 16:31:34 -08:00
|
|
|
fn non_prefix_underscore(some_foo: u32) -> u32 {
|
|
|
|
some_foo + 1
|
2015-12-10 12:54:43 -08:00
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we do not lint if we do not use the binding (simple case)
|
2015-12-18 16:04:33 -08:00
|
|
|
fn unused_underscore_simple(_foo: u32) -> u32 {
|
|
|
|
1
|
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we do not lint if we do not use the binding (complex case). This checks for
|
2015-12-18 16:04:33 -08:00
|
|
|
/// compatibility with the built-in `unused_variables` lint.
|
|
|
|
fn unused_underscore_complex(mut _foo: u32) -> u32 {
|
|
|
|
_foo += 1;
|
|
|
|
_foo = 2;
|
2015-12-12 16:31:34 -08:00
|
|
|
1
|
2015-12-10 12:54:43 -08:00
|
|
|
}
|
2015-12-12 16:31:34 -08:00
|
|
|
|
2019-04-20 09:33:13 +02:00
|
|
|
/// Test that we do not lint for multiple underscores
|
2015-12-18 16:04:33 -08:00
|
|
|
fn multiple_underscores(__foo: u32) -> u32 {
|
|
|
|
__foo + 1
|
2015-12-18 13:47:12 -08:00
|
|
|
}
|
|
|
|
|
2015-12-17 13:52:30 -08:00
|
|
|
// Non-variable bindings with preceding underscore
|
|
|
|
fn _fn_test() {}
|
|
|
|
struct _StructTest;
|
|
|
|
enum _EnumTest {
|
2016-02-01 12:51:33 +01:00
|
|
|
_Empty,
|
2018-12-09 23:26:16 +01:00
|
|
|
_Value(_StructTest),
|
2015-12-17 13:52:30 -08:00
|
|
|
}
|
|
|
|
|
2019-01-31 01:15:29 +00:00
|
|
|
/// Tests that we do not lint for non-variable bindings
|
2015-12-17 13:52:30 -08:00
|
|
|
fn non_variables() {
|
|
|
|
_fn_test();
|
|
|
|
let _s = _StructTest;
|
2016-02-01 12:51:33 +01:00
|
|
|
let _e = match _EnumTest::_Value(_StructTest) {
|
|
|
|
_EnumTest::_Empty => 0,
|
|
|
|
_EnumTest::_Value(_st) => 1,
|
2015-12-17 13:52:30 -08:00
|
|
|
};
|
|
|
|
let f = _fn_test;
|
|
|
|
f();
|
|
|
|
}
|
2020-04-27 21:12:39 +02:00
|
|
|
|
|
|
|
// Tests that we do not lint if the binding comes from await desugaring,
|
|
|
|
// but we do lint the awaited expression. See issue 5360.
|
2020-04-27 00:12:51 +02:00
|
|
|
async fn await_desugaring() {
|
|
|
|
async fn foo() {}
|
2020-04-27 21:12:39 +02:00
|
|
|
fn uses_i(_i: i32) {}
|
|
|
|
|
2020-04-27 00:12:51 +02:00
|
|
|
foo().await;
|
2020-04-27 21:12:39 +02:00
|
|
|
({
|
|
|
|
let _i = 5;
|
|
|
|
uses_i(_i);
|
|
|
|
foo()
|
|
|
|
})
|
|
|
|
.await
|
2020-04-27 00:12:51 +02:00
|
|
|
}
|
2015-12-17 13:52:30 -08:00
|
|
|
|
2015-12-12 16:31:34 -08:00
|
|
|
fn main() {
|
|
|
|
let foo = 0u32;
|
|
|
|
// tests of unused_underscore lint
|
|
|
|
let _ = prefix_underscore(foo);
|
2019-05-11 20:40:05 -07:00
|
|
|
in_macro_or_desugar(foo);
|
2015-12-18 16:29:22 -08:00
|
|
|
in_struct_field();
|
2015-12-12 16:31:34 -08:00
|
|
|
// possible false positives
|
|
|
|
let _ = non_prefix_underscore(foo);
|
2015-12-18 16:04:33 -08:00
|
|
|
let _ = unused_underscore_simple(foo);
|
|
|
|
let _ = unused_underscore_complex(foo);
|
2015-12-18 13:47:12 -08:00
|
|
|
let _ = multiple_underscores(foo);
|
2015-12-17 13:52:30 -08:00
|
|
|
non_variables();
|
2020-04-27 00:12:51 +02:00
|
|
|
await_desugaring();
|
2015-12-12 16:31:34 -08:00
|
|
|
}
|