2022-04-07 17:39:59 +00:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_help;
|
|
|
|
use clippy_utils::source::walk_span_to_context;
|
2022-05-21 11:24:00 +00:00
|
|
|
use clippy_utils::{get_parent_node, is_lint_allowed};
|
2022-04-07 17:39:59 +00:00
|
|
|
use rustc_data_structures::sync::Lrc;
|
2022-05-21 11:24:00 +00:00
|
|
|
use rustc_hir as hir;
|
|
|
|
use rustc_hir::{Block, BlockCheckMode, ItemKind, Node, UnsafeSource};
|
2022-04-07 17:39:59 +00:00
|
|
|
use rustc_lexer::{tokenize, TokenKind};
|
|
|
|
use rustc_lint::{LateContext, LateLintPass, LintContext};
|
2021-10-21 11:11:36 +00:00
|
|
|
use rustc_middle::lint::in_external_macro;
|
2022-04-07 17:39:59 +00:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
2022-05-21 11:24:00 +00:00
|
|
|
use rustc_span::{BytePos, Pos, Span, SyntaxContext};
|
2021-10-21 11:11:36 +00:00
|
|
|
|
|
|
|
declare_clippy_lint! {
|
|
|
|
/// ### What it does
|
2022-05-21 11:24:00 +00:00
|
|
|
/// Checks for `unsafe` blocks and impls without a `// SAFETY: ` comment
|
2021-10-21 11:11:36 +00:00
|
|
|
/// explaining why the unsafe operations performed inside
|
|
|
|
/// the block are safe.
|
|
|
|
///
|
2022-04-07 17:39:59 +00:00
|
|
|
/// Note the comment must appear on the line(s) preceding the unsafe block
|
|
|
|
/// with nothing appearing in between. The following is ok:
|
|
|
|
/// ```ignore
|
|
|
|
/// foo(
|
|
|
|
/// // SAFETY:
|
|
|
|
/// // This is a valid safety comment
|
|
|
|
/// unsafe { *x }
|
|
|
|
/// )
|
|
|
|
/// ```
|
|
|
|
/// But neither of these are:
|
|
|
|
/// ```ignore
|
|
|
|
/// // SAFETY:
|
|
|
|
/// // This is not a valid safety comment
|
|
|
|
/// foo(
|
|
|
|
/// /* SAFETY: Neither is this */ unsafe { *x },
|
|
|
|
/// );
|
|
|
|
/// ```
|
|
|
|
///
|
2021-10-21 11:11:36 +00:00
|
|
|
/// ### Why is this bad?
|
2022-05-21 11:24:00 +00:00
|
|
|
/// Undocumented unsafe blocks and impls can make it difficult to
|
2021-10-21 11:11:36 +00:00
|
|
|
/// read and maintain code, as well as uncover unsoundness
|
|
|
|
/// and bugs.
|
|
|
|
///
|
|
|
|
/// ### Example
|
|
|
|
/// ```rust
|
|
|
|
/// use std::ptr::NonNull;
|
|
|
|
/// let a = &mut 42;
|
|
|
|
///
|
|
|
|
/// let ptr = unsafe { NonNull::new_unchecked(a) };
|
|
|
|
/// ```
|
|
|
|
/// Use instead:
|
|
|
|
/// ```rust
|
|
|
|
/// use std::ptr::NonNull;
|
|
|
|
/// let a = &mut 42;
|
|
|
|
///
|
2021-12-30 14:10:43 +00:00
|
|
|
/// // SAFETY: references are guaranteed to be non-null.
|
2021-10-21 11:11:36 +00:00
|
|
|
/// let ptr = unsafe { NonNull::new_unchecked(a) };
|
|
|
|
/// ```
|
2021-12-06 11:33:31 +00:00
|
|
|
#[clippy::version = "1.58.0"]
|
2021-10-21 11:11:36 +00:00
|
|
|
pub UNDOCUMENTED_UNSAFE_BLOCKS,
|
|
|
|
restriction,
|
|
|
|
"creating an unsafe block without explaining why it is safe"
|
|
|
|
}
|
|
|
|
|
2022-04-07 17:39:59 +00:00
|
|
|
declare_lint_pass!(UndocumentedUnsafeBlocks => [UNDOCUMENTED_UNSAFE_BLOCKS]);
|
2021-10-21 11:11:36 +00:00
|
|
|
|
|
|
|
impl LateLintPass<'_> for UndocumentedUnsafeBlocks {
|
|
|
|
fn check_block(&mut self, cx: &LateContext<'_>, block: &'_ Block<'_>) {
|
2022-04-07 17:39:59 +00:00
|
|
|
if block.rules == BlockCheckMode::UnsafeBlock(UnsafeSource::UserProvided)
|
|
|
|
&& !in_external_macro(cx.tcx.sess, block.span)
|
|
|
|
&& !is_lint_allowed(cx, UNDOCUMENTED_UNSAFE_BLOCKS, block.hir_id)
|
2022-05-21 11:24:00 +00:00
|
|
|
&& !is_unsafe_from_proc_macro(cx, block.span)
|
2022-04-07 17:39:59 +00:00
|
|
|
&& !block_has_safety_comment(cx, block)
|
|
|
|
{
|
|
|
|
let source_map = cx.tcx.sess.source_map();
|
|
|
|
let span = if source_map.is_multiline(block.span) {
|
|
|
|
source_map.span_until_char(block.span, '\n')
|
|
|
|
} else {
|
|
|
|
block.span
|
|
|
|
};
|
2021-10-21 11:11:36 +00:00
|
|
|
|
2022-04-07 17:39:59 +00:00
|
|
|
span_lint_and_help(
|
|
|
|
cx,
|
|
|
|
UNDOCUMENTED_UNSAFE_BLOCKS,
|
|
|
|
span,
|
|
|
|
"unsafe block missing a safety comment",
|
|
|
|
None,
|
|
|
|
"consider adding a safety comment on the preceding line",
|
|
|
|
);
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
|
|
|
}
|
2022-05-21 11:24:00 +00:00
|
|
|
|
|
|
|
fn check_item(&mut self, cx: &LateContext<'_>, item: &hir::Item<'_>) {
|
|
|
|
if let hir::ItemKind::Impl(imple) = item.kind
|
|
|
|
&& imple.unsafety == hir::Unsafety::Unsafe
|
|
|
|
&& !in_external_macro(cx.tcx.sess, item.span)
|
|
|
|
&& !is_lint_allowed(cx, UNDOCUMENTED_UNSAFE_BLOCKS, item.hir_id())
|
|
|
|
&& !is_unsafe_from_proc_macro(cx, item.span)
|
|
|
|
&& !item_has_safety_comment(cx, item)
|
|
|
|
{
|
|
|
|
let source_map = cx.tcx.sess.source_map();
|
|
|
|
let span = if source_map.is_multiline(item.span) {
|
|
|
|
source_map.span_until_char(item.span, '\n')
|
|
|
|
} else {
|
|
|
|
item.span
|
|
|
|
};
|
|
|
|
|
|
|
|
span_lint_and_help(
|
|
|
|
cx,
|
|
|
|
UNDOCUMENTED_UNSAFE_BLOCKS,
|
|
|
|
span,
|
|
|
|
"unsafe impl missing a safety comment",
|
|
|
|
None,
|
|
|
|
"consider adding a safety comment on the preceding line",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2022-04-07 17:39:59 +00:00
|
|
|
}
|
2021-10-21 11:11:36 +00:00
|
|
|
|
2022-05-21 11:24:00 +00:00
|
|
|
fn is_unsafe_from_proc_macro(cx: &LateContext<'_>, span: Span) -> bool {
|
2022-04-07 17:39:59 +00:00
|
|
|
let source_map = cx.sess().source_map();
|
2022-05-21 11:24:00 +00:00
|
|
|
let file_pos = source_map.lookup_byte_offset(span.lo());
|
2022-04-07 17:39:59 +00:00
|
|
|
file_pos
|
|
|
|
.sf
|
|
|
|
.src
|
|
|
|
.as_deref()
|
|
|
|
.and_then(|src| src.get(file_pos.pos.to_usize()..))
|
|
|
|
.map_or(true, |src| !src.starts_with("unsafe"))
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
|
|
|
|
2022-04-07 17:39:59 +00:00
|
|
|
/// Checks if the lines immediately preceding the block contain a safety comment.
|
2022-05-21 11:24:00 +00:00
|
|
|
fn block_has_safety_comment(cx: &LateContext<'_>, block: &hir::Block<'_>) -> bool {
|
2022-04-07 17:39:59 +00:00
|
|
|
// This intentionally ignores text before the start of a function so something like:
|
|
|
|
// ```
|
|
|
|
// // SAFETY: reason
|
|
|
|
// fn foo() { unsafe { .. } }
|
|
|
|
// ```
|
|
|
|
// won't work. This is to avoid dealing with where such a comment should be place relative to
|
|
|
|
// attributes and doc comments.
|
|
|
|
|
2022-05-21 11:24:00 +00:00
|
|
|
span_from_macro_expansion_has_safety_comment(cx, block.span) || span_in_body_has_safety_comment(cx, block.span)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks if the lines immediately preceding the item contain a safety comment.
|
|
|
|
#[allow(clippy::collapsible_match)]
|
|
|
|
fn item_has_safety_comment(cx: &LateContext<'_>, item: &hir::Item<'_>) -> bool {
|
|
|
|
if span_from_macro_expansion_has_safety_comment(cx, item.span) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if item.span.ctxt() == SyntaxContext::root() {
|
|
|
|
if let Some(parent_node) = get_parent_node(cx.tcx, item.hir_id()) {
|
|
|
|
let comment_start = match parent_node {
|
|
|
|
Node::Crate(parent_mod) => {
|
|
|
|
comment_start_before_impl_in_mod(cx, parent_mod, parent_mod.spans.inner_span, item)
|
|
|
|
},
|
|
|
|
Node::Item(parent_item) => {
|
|
|
|
if let ItemKind::Mod(parent_mod) = &parent_item.kind {
|
|
|
|
comment_start_before_impl_in_mod(cx, parent_mod, parent_item.span, item)
|
|
|
|
} else {
|
|
|
|
// Doesn't support impls in this position. Pretend a comment was found.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Node::Stmt(stmt) => {
|
|
|
|
if let Some(stmt_parent) = get_parent_node(cx.tcx, stmt.hir_id) {
|
|
|
|
match stmt_parent {
|
|
|
|
Node::Block(block) => walk_span_to_context(block.span, SyntaxContext::root()).map(Span::lo),
|
|
|
|
_ => {
|
|
|
|
// Doesn't support impls in this position. Pretend a comment was found.
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Problem getting the parent node. Pretend a comment was found.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
// Doesn't support impls in this position. Pretend a comment was found.
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
let source_map = cx.sess().source_map();
|
|
|
|
if let Some(comment_start) = comment_start
|
|
|
|
&& let Ok(unsafe_line) = source_map.lookup_line(item.span.lo())
|
|
|
|
&& let Ok(comment_start_line) = source_map.lookup_line(comment_start)
|
|
|
|
&& Lrc::ptr_eq(&unsafe_line.sf, &comment_start_line.sf)
|
|
|
|
&& let Some(src) = unsafe_line.sf.src.as_deref()
|
|
|
|
{
|
2022-05-30 05:59:45 +00:00
|
|
|
unsafe_line.sf.lines(|lines| {
|
|
|
|
comment_start_line.line < unsafe_line.line && text_has_safety_comment(
|
|
|
|
src,
|
|
|
|
&lines[comment_start_line.line + 1..=unsafe_line.line],
|
|
|
|
unsafe_line.sf.start_pos.to_usize(),
|
|
|
|
)
|
|
|
|
})
|
2022-05-21 11:24:00 +00:00
|
|
|
} else {
|
|
|
|
// Problem getting source text. Pretend a comment was found.
|
|
|
|
true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// No parent node. Pretend a comment was found.
|
|
|
|
true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn comment_start_before_impl_in_mod(
|
|
|
|
cx: &LateContext<'_>,
|
|
|
|
parent_mod: &hir::Mod<'_>,
|
|
|
|
parent_mod_span: Span,
|
|
|
|
imple: &hir::Item<'_>,
|
|
|
|
) -> Option<BytePos> {
|
|
|
|
parent_mod.item_ids.iter().enumerate().find_map(|(idx, item_id)| {
|
|
|
|
if *item_id == imple.item_id() {
|
|
|
|
if idx == 0 {
|
|
|
|
// mod A { /* comment */ unsafe impl T {} ... }
|
|
|
|
// ^------------------------------------------^ returns the start of this span
|
|
|
|
// ^---------------------^ finally checks comments in this range
|
|
|
|
if let Some(sp) = walk_span_to_context(parent_mod_span, SyntaxContext::root()) {
|
|
|
|
return Some(sp.lo());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// some_item /* comment */ unsafe impl T {}
|
|
|
|
// ^-------^ returns the end of this span
|
|
|
|
// ^---------------^ finally checks comments in this range
|
|
|
|
let prev_item = cx.tcx.hir().item(parent_mod.item_ids[idx - 1]);
|
|
|
|
if let Some(sp) = walk_span_to_context(prev_item.span, SyntaxContext::root()) {
|
|
|
|
return Some(sp.hi());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn span_from_macro_expansion_has_safety_comment(cx: &LateContext<'_>, span: Span) -> bool {
|
2022-04-07 17:39:59 +00:00
|
|
|
let source_map = cx.sess().source_map();
|
2022-05-21 11:24:00 +00:00
|
|
|
let ctxt = span.ctxt();
|
|
|
|
if ctxt == SyntaxContext::root() {
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
// From a macro expansion. Get the text from the start of the macro declaration to start of the
|
|
|
|
// unsafe block.
|
2022-04-07 17:39:59 +00:00
|
|
|
// macro_rules! foo { () => { stuff }; (x) => { unsafe { stuff } }; }
|
|
|
|
// ^--------------------------------------------^
|
2022-05-21 11:24:00 +00:00
|
|
|
if let Ok(unsafe_line) = source_map.lookup_line(span.lo())
|
2022-04-07 17:39:59 +00:00
|
|
|
&& let Ok(macro_line) = source_map.lookup_line(ctxt.outer_expn_data().def_site.lo())
|
|
|
|
&& Lrc::ptr_eq(&unsafe_line.sf, ¯o_line.sf)
|
|
|
|
&& let Some(src) = unsafe_line.sf.src.as_deref()
|
|
|
|
{
|
2022-05-30 05:59:45 +00:00
|
|
|
unsafe_line.sf.lines(|lines| {
|
|
|
|
macro_line.line < unsafe_line.line && text_has_safety_comment(
|
|
|
|
src,
|
|
|
|
&lines[macro_line.line + 1..=unsafe_line.line],
|
|
|
|
unsafe_line.sf.start_pos.to_usize(),
|
|
|
|
)
|
|
|
|
})
|
2022-04-07 17:39:59 +00:00
|
|
|
} else {
|
|
|
|
// Problem getting source text. Pretend a comment was found.
|
|
|
|
true
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
2022-05-21 11:24:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-18 07:39:37 +00:00
|
|
|
fn get_body_search_span(cx: &LateContext<'_>) -> Option<Span> {
|
|
|
|
let body = cx.enclosing_body?;
|
|
|
|
let map = cx.tcx.hir();
|
|
|
|
let mut span = map.body(body).value.span;
|
|
|
|
for (_, node) in map.parent_iter(body.hir_id) {
|
|
|
|
match node {
|
|
|
|
Node::Expr(e) => span = e.span,
|
|
|
|
Node::Block(_) | Node::Arm(_) | Node::Stmt(_) | Node::Local(_) => (),
|
|
|
|
_ => break,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(span)
|
|
|
|
}
|
|
|
|
|
2022-05-21 11:24:00 +00:00
|
|
|
fn span_in_body_has_safety_comment(cx: &LateContext<'_>, span: Span) -> bool {
|
|
|
|
let source_map = cx.sess().source_map();
|
|
|
|
let ctxt = span.ctxt();
|
|
|
|
if ctxt == SyntaxContext::root()
|
2022-07-18 07:39:37 +00:00
|
|
|
&& let Some(search_span) = get_body_search_span(cx)
|
2022-04-07 17:39:59 +00:00
|
|
|
{
|
2022-05-21 11:24:00 +00:00
|
|
|
if let Ok(unsafe_line) = source_map.lookup_line(span.lo())
|
2022-07-18 07:39:37 +00:00
|
|
|
&& let Some(body_span) = walk_span_to_context(search_span, SyntaxContext::root())
|
2022-05-21 11:24:00 +00:00
|
|
|
&& let Ok(body_line) = source_map.lookup_line(body_span.lo())
|
|
|
|
&& Lrc::ptr_eq(&unsafe_line.sf, &body_line.sf)
|
|
|
|
&& let Some(src) = unsafe_line.sf.src.as_deref()
|
|
|
|
{
|
|
|
|
// Get the text from the start of function body to the unsafe block.
|
|
|
|
// fn foo() { some_stuff; unsafe { stuff }; other_stuff; }
|
|
|
|
// ^-------------^
|
2022-05-30 05:59:45 +00:00
|
|
|
unsafe_line.sf.lines(|lines| {
|
|
|
|
body_line.line < unsafe_line.line && text_has_safety_comment(
|
|
|
|
src,
|
|
|
|
&lines[body_line.line + 1..=unsafe_line.line],
|
|
|
|
unsafe_line.sf.start_pos.to_usize(),
|
|
|
|
)
|
|
|
|
})
|
2022-05-21 11:24:00 +00:00
|
|
|
} else {
|
|
|
|
// Problem getting source text. Pretend a comment was found.
|
|
|
|
true
|
|
|
|
}
|
2022-04-07 17:39:59 +00:00
|
|
|
} else {
|
2022-05-21 11:24:00 +00:00
|
|
|
false
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-07 17:39:59 +00:00
|
|
|
/// Checks if the given text has a safety comment for the immediately proceeding line.
|
|
|
|
fn text_has_safety_comment(src: &str, line_starts: &[BytePos], offset: usize) -> bool {
|
|
|
|
let mut lines = line_starts
|
|
|
|
.array_windows::<2>()
|
|
|
|
.rev()
|
|
|
|
.map_while(|[start, end]| {
|
2022-05-05 14:12:52 +00:00
|
|
|
let start = start.to_usize() - offset;
|
|
|
|
let end = end.to_usize() - offset;
|
|
|
|
src.get(start..end).map(|text| (start, text.trim_start()))
|
2022-04-07 17:39:59 +00:00
|
|
|
})
|
|
|
|
.filter(|(_, text)| !text.is_empty());
|
|
|
|
|
|
|
|
let Some((line_start, line)) = lines.next() else {
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
// Check for a sequence of line comments.
|
|
|
|
if line.starts_with("//") {
|
|
|
|
let mut line = line;
|
|
|
|
loop {
|
|
|
|
if line.to_ascii_uppercase().contains("SAFETY:") {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
match lines.next() {
|
|
|
|
Some((_, x)) if x.starts_with("//") => line = x,
|
|
|
|
_ => return false,
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-07 17:39:59 +00:00
|
|
|
// No line comments; look for the start of a block comment.
|
|
|
|
// This will only find them if they are at the start of a line.
|
|
|
|
let (mut line_start, mut line) = (line_start, line);
|
|
|
|
loop {
|
|
|
|
if line.starts_with("/*") {
|
2022-05-05 14:12:52 +00:00
|
|
|
let src = src[line_start..line_starts.last().unwrap().to_usize() - offset].trim_start();
|
2022-04-07 17:39:59 +00:00
|
|
|
let mut tokens = tokenize(src);
|
2022-07-27 03:59:30 +00:00
|
|
|
return src[..tokens.next().unwrap().len as usize]
|
2022-04-07 17:39:59 +00:00
|
|
|
.to_ascii_uppercase()
|
|
|
|
.contains("SAFETY:")
|
|
|
|
&& tokens.all(|t| t.kind == TokenKind::Whitespace);
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
2022-04-07 17:39:59 +00:00
|
|
|
match lines.next() {
|
|
|
|
Some(x) => (line_start, line) = x,
|
|
|
|
None => return false,
|
2021-10-21 11:11:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|