mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-07 18:58:51 +00:00
476 lines
10 KiB
Rust
476 lines
10 KiB
Rust
use hir::db::ExpandDatabase;
|
|
use hir::HirFileIdExt;
|
|
use ide_db::{assists::Assist, source_change::SourceChange};
|
|
use syntax::{ast, SyntaxNode};
|
|
use syntax::{match_ast, AstNode};
|
|
use text_edit::TextEdit;
|
|
|
|
use crate::{fix, Diagnostic, DiagnosticCode, DiagnosticsContext};
|
|
|
|
// Diagnostic: missing-unsafe
|
|
//
|
|
// This diagnostic is triggered if an operation marked as `unsafe` is used outside of an `unsafe` function or block.
|
|
pub(crate) fn missing_unsafe(ctx: &DiagnosticsContext<'_>, d: &hir::MissingUnsafe) -> Diagnostic {
|
|
Diagnostic::new_with_syntax_node_ptr(
|
|
ctx,
|
|
DiagnosticCode::RustcHardError("E0133"),
|
|
"this operation is unsafe and requires an unsafe function or block",
|
|
d.expr.map(|it| it.into()),
|
|
)
|
|
.with_fixes(fixes(ctx, d))
|
|
}
|
|
|
|
fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::MissingUnsafe) -> Option<Vec<Assist>> {
|
|
// The fixit will not work correctly for macro expansions, so we don't offer it in that case.
|
|
if d.expr.file_id.is_macro() {
|
|
return None;
|
|
}
|
|
|
|
let root = ctx.sema.db.parse_or_expand(d.expr.file_id);
|
|
let expr = d.expr.value.to_node(&root);
|
|
|
|
let node_to_add_unsafe_block = pick_best_node_to_add_unsafe_block(&expr)?;
|
|
|
|
let replacement = format!("unsafe {{ {} }}", node_to_add_unsafe_block.text());
|
|
let edit = TextEdit::replace(node_to_add_unsafe_block.text_range(), replacement);
|
|
let source_change =
|
|
SourceChange::from_text_edit(d.expr.file_id.original_file(ctx.sema.db), edit);
|
|
Some(vec![fix("add_unsafe", "Add unsafe block", source_change, expr.syntax().text_range())])
|
|
}
|
|
|
|
// Pick the first ancestor expression of the unsafe `expr` that is not a
|
|
// receiver of a method call, a field access, the left-hand side of an
|
|
// assignment, or a reference. As all of those cases would incur a forced move
|
|
// if wrapped which might not be wanted. That is:
|
|
// - `unsafe_expr.foo` -> `unsafe { unsafe_expr.foo }`
|
|
// - `unsafe_expr.foo.bar` -> `unsafe { unsafe_expr.foo.bar }`
|
|
// - `unsafe_expr.foo()` -> `unsafe { unsafe_expr.foo() }`
|
|
// - `unsafe_expr.foo.bar()` -> `unsafe { unsafe_expr.foo.bar() }`
|
|
// - `unsafe_expr += 1` -> `unsafe { unsafe_expr += 1 }`
|
|
// - `&unsafe_expr` -> `unsafe { &unsafe_expr }`
|
|
// - `&&unsafe_expr` -> `unsafe { &&unsafe_expr }`
|
|
fn pick_best_node_to_add_unsafe_block(unsafe_expr: &ast::Expr) -> Option<SyntaxNode> {
|
|
// The `unsafe_expr` might be:
|
|
// - `ast::CallExpr`: call an unsafe function
|
|
// - `ast::MethodCallExpr`: call an unsafe method
|
|
// - `ast::PrefixExpr`: dereference a raw pointer
|
|
// - `ast::PathExpr`: access a static mut variable
|
|
for (node, parent) in
|
|
unsafe_expr.syntax().ancestors().zip(unsafe_expr.syntax().ancestors().skip(1))
|
|
{
|
|
match_ast! {
|
|
match parent {
|
|
// If the `parent` is a `MethodCallExpr`, that means the `node`
|
|
// is the receiver of the method call, because only the receiver
|
|
// can be a direct child of a method call. The method name
|
|
// itself is not an expression but a `NameRef`, and an argument
|
|
// is a direct child of an `ArgList`.
|
|
ast::MethodCallExpr(_) => continue,
|
|
ast::FieldExpr(_) => continue,
|
|
ast::RefExpr(_) => continue,
|
|
ast::BinExpr(it) => {
|
|
// Check if the `node` is the left-hand side of an
|
|
// assignment, if so, we don't want to wrap it in an unsafe
|
|
// block, e.g. `unsafe_expr += 1`
|
|
let is_left_hand_side_of_assignment = {
|
|
if let Some(ast::BinaryOp::Assignment { .. }) = it.op_kind() {
|
|
it.lhs().map(|lhs| lhs.syntax().text_range().contains_range(node.text_range())).unwrap_or(false)
|
|
} else {
|
|
false
|
|
}
|
|
};
|
|
if !is_left_hand_side_of_assignment {
|
|
return Some(node);
|
|
}
|
|
},
|
|
_ => { return Some(node); }
|
|
|
|
}
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use crate::tests::{check_diagnostics, check_fix, check_no_fix};
|
|
|
|
#[test]
|
|
fn missing_unsafe_diagnostic_with_raw_ptr() {
|
|
check_diagnostics(
|
|
r#"
|
|
fn main() {
|
|
let x = &5 as *const usize;
|
|
unsafe { let _y = *x; }
|
|
let _z = *x;
|
|
} //^^💡 error: this operation is unsafe and requires an unsafe function or block
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn missing_unsafe_diagnostic_with_unsafe_call() {
|
|
check_diagnostics(
|
|
r#"
|
|
struct HasUnsafe;
|
|
|
|
impl HasUnsafe {
|
|
unsafe fn unsafe_fn(&self) {
|
|
let x = &5 as *const usize;
|
|
let _y = *x;
|
|
}
|
|
}
|
|
|
|
unsafe fn unsafe_fn() {
|
|
let x = &5 as *const usize;
|
|
let _y = *x;
|
|
}
|
|
|
|
fn main() {
|
|
unsafe_fn();
|
|
//^^^^^^^^^^^💡 error: this operation is unsafe and requires an unsafe function or block
|
|
HasUnsafe.unsafe_fn();
|
|
//^^^^^^^^^^^^^^^^^^^^^💡 error: this operation is unsafe and requires an unsafe function or block
|
|
unsafe {
|
|
unsafe_fn();
|
|
HasUnsafe.unsafe_fn();
|
|
}
|
|
}
|
|
"#,
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn missing_unsafe_diagnostic_with_static_mut() {
|
|
check_diagnostics(
|
|
r#"
|
|
//- minicore: copy
|
|
|
|
struct Ty {
|
|
a: u8,
|
|
}
|
|
|
|
static mut STATIC_MUT: Ty = Ty { a: 0 };
|
|
|
|
fn main() {
|
|
let _x = STATIC_MUT.a;
|
|
//^^^^^^^^^^💡 error: this operation is unsafe and requires an unsafe function or block
|
|
unsafe {
|
|
let _x = STATIC_MUT.a;
|
|
}
|
|
}
|
|
"#,
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn no_missing_unsafe_diagnostic_with_safe_intrinsic() {
|
|
check_diagnostics(
|
|
r#"
|
|
extern "rust-intrinsic" {
|
|
#[rustc_safe_intrinsic]
|
|
pub fn bitreverse(x: u32) -> u32; // Safe intrinsic
|
|
pub fn floorf32(x: f32) -> f32; // Unsafe intrinsic
|
|
}
|
|
|
|
fn main() {
|
|
let _ = bitreverse(12);
|
|
let _ = floorf32(12.0);
|
|
//^^^^^^^^^^^^^^💡 error: this operation is unsafe and requires an unsafe function or block
|
|
}
|
|
"#,
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn add_unsafe_block_when_dereferencing_a_raw_pointer() {
|
|
check_fix(
|
|
r#"
|
|
fn main() {
|
|
let x = &5 as *const usize;
|
|
let _z = *x$0;
|
|
}
|
|
"#,
|
|
r#"
|
|
fn main() {
|
|
let x = &5 as *const usize;
|
|
let _z = unsafe { *x };
|
|
}
|
|
"#,
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn add_unsafe_block_when_calling_unsafe_function() {
|
|
check_fix(
|
|
r#"
|
|
unsafe fn func() {
|
|
let x = &5 as *const usize;
|
|
let z = *x;
|
|
}
|
|
fn main() {
|
|
func$0();
|
|
}
|
|
"#,
|
|
r#"
|
|
unsafe fn func() {
|
|
let x = &5 as *const usize;
|
|
let z = *x;
|
|
}
|
|
fn main() {
|
|
unsafe { func() };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn add_unsafe_block_when_calling_unsafe_method() {
|
|
check_fix(
|
|
r#"
|
|
struct S(usize);
|
|
impl S {
|
|
unsafe fn func(&self) {
|
|
let x = &self.0 as *const usize;
|
|
let _z = *x;
|
|
}
|
|
}
|
|
fn main() {
|
|
let s = S(5);
|
|
s.func$0();
|
|
}
|
|
"#,
|
|
r#"
|
|
struct S(usize);
|
|
impl S {
|
|
unsafe fn func(&self) {
|
|
let x = &self.0 as *const usize;
|
|
let _z = *x;
|
|
}
|
|
}
|
|
fn main() {
|
|
let s = S(5);
|
|
unsafe { s.func() };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn add_unsafe_block_when_accessing_mutable_static() {
|
|
check_fix(
|
|
r#"
|
|
//- minicore: copy
|
|
struct Ty {
|
|
a: u8,
|
|
}
|
|
|
|
static mut STATIC_MUT: Ty = Ty { a: 0 };
|
|
|
|
fn main() {
|
|
let _x = STATIC_MUT$0.a;
|
|
}
|
|
"#,
|
|
r#"
|
|
struct Ty {
|
|
a: u8,
|
|
}
|
|
|
|
static mut STATIC_MUT: Ty = Ty { a: 0 };
|
|
|
|
fn main() {
|
|
let _x = unsafe { STATIC_MUT.a };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn add_unsafe_block_when_calling_unsafe_intrinsic() {
|
|
check_fix(
|
|
r#"
|
|
extern "rust-intrinsic" {
|
|
pub fn floorf32(x: f32) -> f32;
|
|
}
|
|
|
|
fn main() {
|
|
let _ = floorf32$0(12.0);
|
|
}
|
|
"#,
|
|
r#"
|
|
extern "rust-intrinsic" {
|
|
pub fn floorf32(x: f32) -> f32;
|
|
}
|
|
|
|
fn main() {
|
|
let _ = unsafe { floorf32(12.0) };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_as_a_receiver_of_a_method_call() {
|
|
check_fix(
|
|
r#"
|
|
unsafe fn foo() -> String {
|
|
"string".to_string()
|
|
}
|
|
|
|
fn main() {
|
|
foo$0().len();
|
|
}
|
|
"#,
|
|
r#"
|
|
unsafe fn foo() -> String {
|
|
"string".to_string()
|
|
}
|
|
|
|
fn main() {
|
|
unsafe { foo().len() };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_as_an_argument_of_a_method_call() {
|
|
check_fix(
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let mut v = vec![];
|
|
v.push(STATIC_MUT$0);
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let mut v = vec![];
|
|
v.push(unsafe { STATIC_MUT });
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_as_left_hand_side_of_assignment() {
|
|
check_fix(
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
STATIC_MUT$0 = 1;
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
unsafe { STATIC_MUT = 1 };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_as_right_hand_side_of_assignment() {
|
|
check_fix(
|
|
r#"
|
|
//- minicore: copy
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x;
|
|
_x = STATIC_MUT$0;
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x;
|
|
_x = unsafe { STATIC_MUT };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_in_binary_plus() {
|
|
check_fix(
|
|
r#"
|
|
//- minicore: copy
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = STATIC_MUT$0 + 1;
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = unsafe { STATIC_MUT } + 1;
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn ref_to_unsafe_expr() {
|
|
check_fix(
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = &STATIC_MUT$0;
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = unsafe { &STATIC_MUT };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn ref_ref_to_unsafe_expr() {
|
|
check_fix(
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = &&STATIC_MUT$0;
|
|
}
|
|
"#,
|
|
r#"
|
|
static mut STATIC_MUT: u8 = 0;
|
|
|
|
fn main() {
|
|
let _x = unsafe { &&STATIC_MUT };
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn unsafe_expr_in_macro_call() {
|
|
check_no_fix(
|
|
r#"
|
|
unsafe fn foo() -> u8 {
|
|
0
|
|
}
|
|
|
|
fn main() {
|
|
let x = format!("foo: {}", foo$0());
|
|
}
|
|
"#,
|
|
)
|
|
}
|
|
}
|