types.rs 49.9 KB
Newer Older
1
use crate::{LateContext, LateLintPass, LintContext};
U
Ujjwal Sharma 已提交
2
use rustc_ast as ast;
3
use rustc_attr as attr;
4
use rustc_data_structures::fx::FxHashSet;
5
use rustc_errors::Applicability;
6 7
use rustc_hir as hir;
use rustc_hir::{is_range_literal, ExprKind, Node};
8
use rustc_index::vec::Idx;
M
Mazdak Farrokhzad 已提交
9
use rustc_middle::mir::interpret::{sign_extend, truncate};
10
use rustc_middle::ty::layout::{IntegerExt, SizeSkeleton};
M
Mazdak Farrokhzad 已提交
11
use rustc_middle::ty::subst::SubstsRef;
12
use rustc_middle::ty::{self, AdtKind, Ty, TyCtxt, TypeFoldable};
13
use rustc_span::source_map;
14
use rustc_span::symbol::sym;
15
use rustc_span::{Span, DUMMY_SP};
J
jumbatm 已提交
16
use rustc_target::abi::Abi;
R
Ralf Jung 已提交
17
use rustc_target::abi::{Integer, LayoutOf, TagEncoding, VariantIdx, Variants};
J
jumbatm 已提交
18
use rustc_target::spec::abi::Abi as SpecAbi;
K
kenta7777 已提交
19

20
use std::cmp;
G
Gurpreet Singh 已提交
21
use tracing::debug;
T
Taiki Endo 已提交
22

23
declare_lint! {
E
Eric Huss 已提交
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
    /// The `unused_comparisons` lint detects comparisons made useless by
    /// limits of the types involved.
    ///
    /// ### Example
    ///
    /// ```rust
    /// fn foo(x: u8) {
    ///     x >= 0;
    /// }
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// A useless comparison may indicate a mistake, and should be fixed or
    /// removed.
41 42 43 44 45 46
    UNUSED_COMPARISONS,
    Warn,
    "comparisons made useless by limits of the types involved"
}

declare_lint! {
E
Eric Huss 已提交
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
    /// The `overflowing_literals` lint detects literal out of range for its
    /// type.
    ///
    /// ### Example
    ///
    /// ```rust,compile_fail
    /// let x: u8 = 1000;
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// It is usually a mistake to use a literal that overflows the type where
    /// it is used. Either use a literal that is within range, or change the
    /// type to be within the range of the literal.
63
    OVERFLOWING_LITERALS,
64 65
    Deny,
    "literal out of range for its type"
66 67
}

68
declare_lint! {
E
Eric Huss 已提交
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
    /// The `variant_size_differences` lint detects enums with widely varying
    /// variant sizes.
    ///
    /// ### Example
    ///
    /// ```rust,compile_fail
    /// #![deny(variant_size_differences)]
    /// enum En {
    ///     V0(u8),
    ///     VBig([u8; 1024]),
    /// }
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// It can be a mistake to add a variant to an enum that is much larger
    /// than the other variants, bloating the overall size required for all
    /// variants. This can impact performance and memory usage. This is
    /// triggered if one variant is more than 3 times larger than the
    /// second-largest variant.
    ///
    /// Consider placing the large variant's contents on the heap (for example
    /// via [`Box`]) to keep the overall size of the enum itself down.
    ///
    /// This lint is "allow" by default because it can be noisy, and may not be
    /// an actual problem. Decisions about this should be guided with
    /// profiling and benchmarking.
    ///
    /// [`Box`]: https://doc.rust-lang.org/std/boxed/index.html
100 101 102 103 104
    VARIANT_SIZE_DIFFERENCES,
    Allow,
    "detects enums with widely varying variant sizes"
}

105 106 107
#[derive(Copy, Clone)]
pub struct TypeLimits {
    /// Id of the last visited negated expression
108
    negated_expr_id: Option<hir::HirId>,
109 110
}

111 112
impl_lint_pass!(TypeLimits => [UNUSED_COMPARISONS, OVERFLOWING_LITERALS]);

113 114
impl TypeLimits {
    pub fn new() -> TypeLimits {
115
        TypeLimits { negated_expr_id: None }
116 117 118
    }
}

119 120
/// Attempts to special-case the overflowing literal lint when it occurs as a range endpoint.
/// Returns `true` iff the lint was overridden.
121 122
fn lint_overflowing_range_endpoint<'tcx>(
    cx: &LateContext<'tcx>,
123
    lit: &hir::Lit,
124 125
    lit_val: u128,
    max: u128,
C
Camille GILLOT 已提交
126 127
    expr: &'tcx hir::Expr<'tcx>,
    parent_expr: &'tcx hir::Expr<'tcx>,
128
    ty: &str,
129 130 131
) -> bool {
    // We only want to handle exclusive (`..`) ranges,
    // which are represented as `ExprKind::Struct`.
J
jumbatm 已提交
132
    let mut overwritten = false;
V
varkor 已提交
133
    if let ExprKind::Struct(_, eps, _) = &parent_expr.kind {
134 135 136
        if eps.len() != 2 {
            return false;
        }
137 138 139 140
        // We can suggest using an inclusive range
        // (`..=`) instead only if it is the `end` that is
        // overflowing and only by 1.
        if eps[1].expr.hir_id == expr.hir_id && lit_val - 1 == max {
J
jumbatm 已提交
141 142 143 144 145 146 147
            cx.struct_span_lint(OVERFLOWING_LITERALS, parent_expr.span, |lint| {
                let mut err = lint.build(&format!("range endpoint is out of range for `{}`", ty));
                if let Ok(start) = cx.sess().source_map().span_to_snippet(eps[0].span) {
                    use ast::{LitIntType, LitKind};
                    // We need to preserve the literal's suffix,
                    // as it may determine typing information.
                    let suffix = match lit.node {
148 149 150
                        LitKind::Int(_, LitIntType::Signed(s)) => s.name_str().to_string(),
                        LitKind::Int(_, LitIntType::Unsigned(s)) => s.name_str().to_string(),
                        LitKind::Int(_, LitIntType::Unsuffixed) => "".to_string(),
J
jumbatm 已提交
151 152 153 154 155 156 157 158 159 160 161 162 163
                        _ => bug!(),
                    };
                    let suggestion = format!("{}..={}{}", start, lit_val - 1, suffix);
                    err.span_suggestion(
                        parent_expr.span,
                        &"use an inclusive range instead",
                        suggestion,
                        Applicability::MachineApplicable,
                    );
                    err.emit();
                    overwritten = true;
                }
            });
164 165
        }
    }
J
jumbatm 已提交
166
    overwritten
167 168
}

V
varkor 已提交
169 170 171 172
// For `isize` & `usize`, be conservative with the warnings, so that the
// warnings are consistent between 32- and 64-bit platforms.
fn int_ty_range(int_ty: ast::IntTy) -> (i128, i128) {
    match int_ty {
L
Lzu Tao 已提交
173 174 175 176 177 178
        ast::IntTy::Isize => (i64::MIN as i128, i64::MAX as i128),
        ast::IntTy::I8 => (i8::MIN as i64 as i128, i8::MAX as i128),
        ast::IntTy::I16 => (i16::MIN as i64 as i128, i16::MAX as i128),
        ast::IntTy::I32 => (i32::MIN as i64 as i128, i32::MAX as i128),
        ast::IntTy::I64 => (i64::MIN as i128, i64::MAX as i128),
        ast::IntTy::I128 => (i128::MIN as i128, i128::MAX),
V
varkor 已提交
179 180 181 182 183
    }
}

fn uint_ty_range(uint_ty: ast::UintTy) -> (u128, u128) {
    match uint_ty {
L
Lzu Tao 已提交
184 185 186 187 188 189
        ast::UintTy::Usize => (u64::MIN as u128, u64::MAX as u128),
        ast::UintTy::U8 => (u8::MIN as u128, u8::MAX as u128),
        ast::UintTy::U16 => (u16::MIN as u128, u16::MAX as u128),
        ast::UintTy::U32 => (u32::MIN as u128, u32::MAX as u128),
        ast::UintTy::U64 => (u64::MIN as u128, u64::MAX as u128),
        ast::UintTy::U128 => (u128::MIN, u128::MAX),
V
varkor 已提交
190 191 192
    }
}

193
fn get_bin_hex_repr(cx: &LateContext<'_>, lit: &hir::Lit) -> Option<String> {
V
varkor 已提交
194 195 196 197 198
    let src = cx.sess().source_map().span_to_snippet(lit.span).ok()?;
    let firstch = src.chars().next()?;

    if firstch == '0' {
        match src.chars().nth(1) {
199
            Some('x' | 'b') => return Some(src),
V
varkor 已提交
200 201 202 203 204 205 206 207
            _ => return None,
        }
    }

    None
}

fn report_bin_hex_error(
208
    cx: &LateContext<'_>,
C
Camille GILLOT 已提交
209
    expr: &hir::Expr<'_>,
V
varkor 已提交
210 211 212 213 214
    ty: attr::IntType,
    repr_str: String,
    val: u128,
    negative: bool,
) {
215
    let size = Integer::from_attr(&cx.tcx, ty).size();
J
jumbatm 已提交
216
    cx.struct_span_lint(OVERFLOWING_LITERALS, expr.span, |lint| {
217 218 219 220 221 222 223 224 225 226
        let (t, actually) = match ty {
            attr::IntType::SignedInt(t) => {
                let actually = sign_extend(val, size) as i128;
                (t.name_str(), actually.to_string())
            }
            attr::IntType::UnsignedInt(t) => {
                let actually = truncate(val, size);
                (t.name_str(), actually.to_string())
            }
        };
J
jumbatm 已提交
227 228 229
        let mut err = lint.build(&format!("literal out of range for {}", t));
        err.note(&format!(
            "the literal `{}` (decimal `{}`) does not fit into \
230
             the type `{}` and will become `{}{}`",
J
jumbatm 已提交
231 232
            repr_str, val, t, actually, t
        ));
233
        if let Some(sugg_ty) =
234
            get_type_suggestion(&cx.typeck_results().node_type(expr.hir_id), val, negative)
J
jumbatm 已提交
235 236 237 238 239 240 241 242 243 244 245
        {
            if let Some(pos) = repr_str.chars().position(|c| c == 'i' || c == 'u') {
                let (sans_suffix, _) = repr_str.split_at(pos);
                err.span_suggestion(
                    expr.span,
                    &format!("consider using `{}` instead", sugg_ty),
                    format!("{}{}", sans_suffix, sugg_ty),
                    Applicability::MachineApplicable,
                );
            } else {
                err.help(&format!("consider using `{}` instead", sugg_ty));
J
jumbatm 已提交
246
            }
J
jumbatm 已提交
247 248 249
        }
        err.emit();
    });
V
varkor 已提交
250 251 252 253 254 255 256 257 258
}

// This function finds the next fitting type and generates a suggestion string.
// It searches for fitting types in the following way (`X < Y`):
//  - `iX`: if literal fits in `uX` => `uX`, else => `iY`
//  - `-iX` => `iY`
//  - `uX` => `uY`
//
// No suggestion for: `isize`, `usize`.
259
fn get_type_suggestion(t: Ty<'_>, val: u128, negative: bool) -> Option<&'static str> {
U
Ujjwal Sharma 已提交
260 261
    use rustc_ast::IntTy::*;
    use rustc_ast::UintTy::*;
V
varkor 已提交
262 263 264 265 266 267 268 269
    macro_rules! find_fit {
        ($ty:expr, $val:expr, $negative:expr,
         $($type:ident => [$($utypes:expr),*] => [$($itypes:expr),*]),+) => {
            {
                let _neg = if negative { 1 } else { 0 };
                match $ty {
                    $($type => {
                        $(if !negative && val <= uint_ty_range($utypes).1 {
270
                            return Some($utypes.name_str())
V
varkor 已提交
271 272
                        })*
                        $(if val <= int_ty_range($itypes).1 as u128 + _neg {
273
                            return Some($itypes.name_str())
V
varkor 已提交
274 275
                        })*
                        None
276
                    },)+
V
varkor 已提交
277 278 279 280 281
                    _ => None
                }
            }
        }
    }
L
LeSeulArtichaut 已提交
282
    match t.kind() {
V
varkor 已提交
283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298
        ty::Int(i) => find_fit!(i, val, negative,
                      I8 => [U8] => [I16, I32, I64, I128],
                      I16 => [U16] => [I32, I64, I128],
                      I32 => [U32] => [I64, I128],
                      I64 => [U64] => [I128],
                      I128 => [U128] => []),
        ty::Uint(u) => find_fit!(u, val, negative,
                      U8 => [U8, U16, U32, U64, U128] => [],
                      U16 => [U16, U32, U64, U128] => [],
                      U32 => [U32, U64, U128] => [],
                      U64 => [U64, U128] => [],
                      U128 => [U128] => []),
        _ => None,
    }
}

299 300
fn lint_int_literal<'tcx>(
    cx: &LateContext<'tcx>,
V
varkor 已提交
301
    type_limits: &TypeLimits,
C
Camille GILLOT 已提交
302
    e: &'tcx hir::Expr<'tcx>,
303
    lit: &hir::Lit,
V
varkor 已提交
304 305 306
    t: ast::IntTy,
    v: u128,
) {
M
Mark Rousskov 已提交
307
    let int_type = t.normalize(cx.sess().target.ptr_width);
308
    let (min, max) = int_ty_range(int_type);
V
varkor 已提交
309
    let max = max as u128;
310
    let negative = type_limits.negated_expr_id == Some(e.hir_id);
V
varkor 已提交
311 312 313 314 315

    // Detect literal value out of range [min, max] inclusive
    // avoiding use of -min to prevent overflow/panic
    if (negative && v > max + 1) || (!negative && v > max) {
        if let Some(repr_str) = get_bin_hex_repr(cx, lit) {
M
Mark Rousskov 已提交
316
            report_bin_hex_error(cx, e, attr::IntType::SignedInt(t), repr_str, v, negative);
V
varkor 已提交
317 318 319
            return;
        }

320
        let par_id = cx.tcx.hir().get_parent_node(e.hir_id);
L
ljedrz 已提交
321
        if let Node::Expr(par_e) = cx.tcx.hir().get(par_id) {
V
varkor 已提交
322
            if let hir::ExprKind::Struct(..) = par_e.kind {
D
David Wood 已提交
323
                if is_range_literal(par_e)
324
                    && lint_overflowing_range_endpoint(cx, lit, v, max, e, par_e, t.name_str())
V
varkor 已提交
325 326 327 328 329 330 331
                {
                    // The overflowing literal lint was overridden.
                    return;
                }
            }
        }

J
jumbatm 已提交
332
        cx.struct_span_lint(OVERFLOWING_LITERALS, e.span, |lint| {
333 334 335 336 337 338 339 340 341 342 343 344
            lint.build(&format!("literal out of range for `{}`", t.name_str()))
                .note(&format!(
                    "the literal `{}` does not fit into the type `{}` whose range is `{}..={}`",
                    cx.sess()
                        .source_map()
                        .span_to_snippet(lit.span)
                        .expect("must get snippet from literal"),
                    t.name_str(),
                    min,
                    max,
                ))
                .emit();
J
jumbatm 已提交
345
        });
V
varkor 已提交
346 347 348
    }
}

349 350
fn lint_uint_literal<'tcx>(
    cx: &LateContext<'tcx>,
C
Camille GILLOT 已提交
351
    e: &'tcx hir::Expr<'tcx>,
352
    lit: &hir::Lit,
V
varkor 已提交
353 354
    t: ast::UintTy,
) {
M
Mark Rousskov 已提交
355
    let uint_type = t.normalize(cx.sess().target.ptr_width);
V
varkor 已提交
356 357 358 359 360 361 362 363
    let (min, max) = uint_ty_range(uint_type);
    let lit_val: u128 = match lit.node {
        // _v is u8, within range by definition
        ast::LitKind::Byte(_v) => return,
        ast::LitKind::Int(v, _) => v,
        _ => bug!(),
    };
    if lit_val < min || lit_val > max {
364
        let parent_id = cx.tcx.hir().get_parent_node(e.hir_id);
L
ljedrz 已提交
365
        if let Node::Expr(par_e) = cx.tcx.hir().get(parent_id) {
V
varkor 已提交
366
            match par_e.kind {
V
varkor 已提交
367
                hir::ExprKind::Cast(..) => {
L
LeSeulArtichaut 已提交
368
                    if let ty::Char = cx.typeck_results().expr_ty(par_e).kind() {
J
jumbatm 已提交
369 370 371 372 373 374 375 376 377 378
                        cx.struct_span_lint(OVERFLOWING_LITERALS, par_e.span, |lint| {
                            lint.build("only `u8` can be cast into `char`")
                                .span_suggestion(
                                    par_e.span,
                                    &"use a `char` literal instead",
                                    format!("'\\u{{{:X}}}'", lit_val),
                                    Applicability::MachineApplicable,
                                )
                                .emit();
                        });
V
varkor 已提交
379 380 381
                        return;
                    }
                }
D
David Wood 已提交
382
                hir::ExprKind::Struct(..) if is_range_literal(par_e) => {
M
Mark Rousskov 已提交
383 384 385 386
                    let t = t.name_str();
                    if lint_overflowing_range_endpoint(cx, lit, lit_val, max, e, par_e, t) {
                        // The overflowing literal lint was overridden.
                        return;
V
varkor 已提交
387
                    }
M
Mark Rousskov 已提交
388
                }
V
varkor 已提交
389 390 391 392 393 394 395
                _ => {}
            }
        }
        if let Some(repr_str) = get_bin_hex_repr(cx, lit) {
            report_bin_hex_error(cx, e, attr::IntType::UnsignedInt(t), repr_str, lit_val, false);
            return;
        }
J
jumbatm 已提交
396
        cx.struct_span_lint(OVERFLOWING_LITERALS, e.span, |lint| {
397 398 399 400 401 402 403 404 405 406 407 408
            lint.build(&format!("literal out of range for `{}`", t.name_str()))
                .note(&format!(
                    "the literal `{}` does not fit into the type `{}` whose range is `{}..={}`",
                    cx.sess()
                        .source_map()
                        .span_to_snippet(lit.span)
                        .expect("must get snippet from literal"),
                    t.name_str(),
                    min,
                    max,
                ))
                .emit()
J
jumbatm 已提交
409
        });
V
varkor 已提交
410 411 412
    }
}

413 414
fn lint_literal<'tcx>(
    cx: &LateContext<'tcx>,
V
varkor 已提交
415
    type_limits: &TypeLimits,
C
Camille GILLOT 已提交
416
    e: &'tcx hir::Expr<'tcx>,
417
    lit: &hir::Lit,
V
varkor 已提交
418
) {
L
LeSeulArtichaut 已提交
419
    match *cx.typeck_results().node_type(e.hir_id).kind() {
V
varkor 已提交
420 421
        ty::Int(t) => {
            match lit.node {
422
                ast::LitKind::Int(v, ast::LitIntType::Signed(_) | ast::LitIntType::Unsuffixed) => {
V
varkor 已提交
423 424 425 426 427
                    lint_int_literal(cx, type_limits, e, lit, t, v)
                }
                _ => bug!(),
            };
        }
M
Mark Rousskov 已提交
428
        ty::Uint(t) => lint_uint_literal(cx, e, lit, t),
V
varkor 已提交
429 430
        ty::Float(t) => {
            let is_infinite = match lit.node {
M
Mark Rousskov 已提交
431 432 433 434
                ast::LitKind::Float(v, _) => match t {
                    ast::FloatTy::F32 => v.as_str().parse().map(f32::is_infinite),
                    ast::FloatTy::F64 => v.as_str().parse().map(f64::is_infinite),
                },
V
varkor 已提交
435 436 437
                _ => bug!(),
            };
            if is_infinite == Ok(true) {
J
jumbatm 已提交
438
                cx.struct_span_lint(OVERFLOWING_LITERALS, e.span, |lint| {
439 440 441 442 443 444 445 446 447 448 449
                    lint.build(&format!("literal out of range for `{}`", t.name_str()))
                        .note(&format!(
                            "the literal `{}` does not fit into the type `{}` and will be converted to `std::{}::INFINITY`",
                            cx.sess()
                                .source_map()
                                .span_to_snippet(lit.span)
                                .expect("must get snippet from literal"),
                            t.name_str(),
                            t.name_str(),
                        ))
                        .emit();
J
jumbatm 已提交
450
                });
V
varkor 已提交
451 452 453 454 455 456
            }
        }
        _ => {}
    }
}

457 458
impl<'tcx> LateLintPass<'tcx> for TypeLimits {
    fn check_expr(&mut self, cx: &LateContext<'tcx>, e: &'tcx hir::Expr<'tcx>) {
V
varkor 已提交
459
        match e.kind {
460
            hir::ExprKind::Unary(hir::UnOp::UnNeg, ref expr) => {
461
                // propagate negation, if the negation itself isn't negated
462 463
                if self.negated_expr_id != Some(e.hir_id) {
                    self.negated_expr_id = Some(expr.hir_id);
464
                }
465
            }
C
csmoe 已提交
466
            hir::ExprKind::Binary(binop, ref l, ref r) => {
467
                if is_comparison(binop) && !check_limits(cx, binop, &l, &r) {
J
jumbatm 已提交
468 469 470
                    cx.struct_span_lint(UNUSED_COMPARISONS, e.span, |lint| {
                        lint.build("comparison is useless due to type limits").emit()
                    });
471
                }
472
            }
V
varkor 已提交
473 474
            hir::ExprKind::Lit(ref lit) => lint_literal(cx, self, e, lit),
            _ => {}
475 476
        };

477
        fn is_valid<T: cmp::PartialOrd>(binop: hir::BinOp, v: T, min: T, max: T) -> bool {
478
            match binop.node {
C
csmoe 已提交
479 480 481 482 483
                hir::BinOpKind::Lt => v > min && v <= max,
                hir::BinOpKind::Le => v >= min && v < max,
                hir::BinOpKind::Gt => v >= min && v < max,
                hir::BinOpKind::Ge => v > min && v <= max,
                hir::BinOpKind::Eq | hir::BinOpKind::Ne => v >= min && v <= max,
484
                _ => bug!(),
485 486 487 488
            }
        }

        fn rev_binop(binop: hir::BinOp) -> hir::BinOp {
M
Mark Rousskov 已提交
489 490 491 492 493 494 495 496 497 498
            source_map::respan(
                binop.span,
                match binop.node {
                    hir::BinOpKind::Lt => hir::BinOpKind::Gt,
                    hir::BinOpKind::Le => hir::BinOpKind::Ge,
                    hir::BinOpKind::Gt => hir::BinOpKind::Lt,
                    hir::BinOpKind::Ge => hir::BinOpKind::Le,
                    _ => return binop,
                },
            )
499 500
        }

M
Mark Rousskov 已提交
501
        fn check_limits(
502
            cx: &LateContext<'_>,
M
Mark Rousskov 已提交
503
            binop: hir::BinOp,
C
Camille GILLOT 已提交
504 505
            l: &hir::Expr<'_>,
            r: &hir::Expr<'_>,
M
Mark Rousskov 已提交
506
        ) -> bool {
V
varkor 已提交
507
            let (lit, expr, swap) = match (&l.kind, &r.kind) {
C
csmoe 已提交
508 509
                (&hir::ExprKind::Lit(_), _) => (l, r, true),
                (_, &hir::ExprKind::Lit(_)) => (r, l, false),
510
                _ => return true,
511 512 513
            };
            // Normalize the binop so that the literal is always on the RHS in
            // the comparison
514
            let norm_binop = if swap { rev_binop(binop) } else { binop };
L
LeSeulArtichaut 已提交
515
            match *cx.typeck_results().node_type(expr.hir_id).kind() {
516
                ty::Int(int_ty) => {
517
                    let (min, max) = int_ty_range(int_ty);
V
varkor 已提交
518
                    let lit_val: i128 = match lit.kind {
M
Mark Rousskov 已提交
519
                        hir::ExprKind::Lit(ref li) => match li.node {
520 521 522 523
                            ast::LitKind::Int(
                                v,
                                ast::LitIntType::Signed(_) | ast::LitIntType::Unsuffixed,
                            ) => v as i128,
M
Mark Rousskov 已提交
524
                            _ => return true,
525
                        },
M
Mark Rousskov 已提交
526
                        _ => bug!(),
527 528 529
                    };
                    is_valid(norm_binop, lit_val, min, max)
                }
530
                ty::Uint(uint_ty) => {
M
Mark Rousskov 已提交
531
                    let (min, max): (u128, u128) = uint_ty_range(uint_ty);
V
varkor 已提交
532
                    let lit_val: u128 = match lit.kind {
M
Mark Rousskov 已提交
533 534 535
                        hir::ExprKind::Lit(ref li) => match li.node {
                            ast::LitKind::Int(v, _) => v,
                            _ => return true,
536
                        },
M
Mark Rousskov 已提交
537
                        _ => bug!(),
538 539 540
                    };
                    is_valid(norm_binop, lit_val, min, max)
                }
541
                _ => true,
542 543 544 545 546
            }
        }

        fn is_comparison(binop: hir::BinOp) -> bool {
            match binop.node {
M
Mark Rousskov 已提交
547 548 549 550 551 552
                hir::BinOpKind::Eq
                | hir::BinOpKind::Lt
                | hir::BinOpKind::Le
                | hir::BinOpKind::Ne
                | hir::BinOpKind::Ge
                | hir::BinOpKind::Gt => true,
553
                _ => false,
554 555 556 557 558 559
            }
        }
    }
}

declare_lint! {
E
Eric Huss 已提交
560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580
    /// The `improper_ctypes` lint detects incorrect use of types in foreign
    /// modules.
    ///
    /// ### Example
    ///
    /// ```rust
    /// extern "C" {
    ///     static STATIC: String;
    /// }
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// The compiler has several checks to verify that types used in `extern`
    /// blocks are safe and follow certain rules to ensure proper
    /// compatibility with the foreign interfaces. This lint is issued when it
    /// detects a probable mistake in a definition. The lint usually should
    /// provide a description of the issue, along with possibly a hint on how
    /// to resolve it.
581 582 583 584 585
    IMPROPER_CTYPES,
    Warn,
    "proper use of libc types in foreign modules"
}

586 587 588
declare_lint_pass!(ImproperCTypesDeclarations => [IMPROPER_CTYPES]);

declare_lint! {
E
Eric Huss 已提交
589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609
    /// The `improper_ctypes_definitions` lint detects incorrect use of
    /// [`extern` function] definitions.
    ///
    /// [`extern` function]: https://doc.rust-lang.org/reference/items/functions.html#extern-function-qualifier
    ///
    /// ### Example
    ///
    /// ```rust
    /// # #![allow(unused)]
    /// pub extern "C" fn str_type(p: &str) { }
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// There are many parameter and return types that may be specified in an
    /// `extern` function that are not compatible with the given ABI. This
    /// lint is an alert that these types should not be used. The lint usually
    /// should provide a description of the issue, along with possibly a hint
    /// on how to resolve it.
610 611 612 613 614 615 616
    IMPROPER_CTYPES_DEFINITIONS,
    Warn,
    "proper use of libc types in foreign item definitions"
}

declare_lint_pass!(ImproperCTypesDefinitions => [IMPROPER_CTYPES_DEFINITIONS]);

J
jumbatm 已提交
617 618 619 620
#[derive(Clone, Copy)]
crate enum CItemKind {
    Declaration,
    Definition,
621
}
622

J
jumbatm 已提交
623 624 625
struct ImproperCTypesVisitor<'a, 'tcx> {
    cx: &'a LateContext<'tcx>,
    mode: CItemKind,
626 627
}

628
enum FfiResult<'tcx> {
629
    FfiSafe,
630
    FfiPhantom(Ty<'tcx>),
631
    FfiUnsafe { ty: Ty<'tcx>, reason: String, help: Option<String> },
632 633
}

J
jumbatm 已提交
634
crate fn nonnull_optimization_guaranteed<'tcx>(tcx: TyCtxt<'tcx>, def: &ty::AdtDef) -> bool {
635 636 637 638 639
    tcx.get_attrs(def.did)
        .iter()
        .any(|a| tcx.sess.check_name(a, sym::rustc_nonnull_optimization_guaranteed))
}

640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659
/// `repr(transparent)` structs can have a single non-ZST field, this function returns that
/// field.
pub fn transparent_newtype_field<'a, 'tcx>(
    tcx: TyCtxt<'tcx>,
    variant: &'a ty::VariantDef,
) -> Option<&'a ty::FieldDef> {
    let param_env = tcx.param_env(variant.def_id);
    for field in &variant.fields {
        let field_ty = tcx.type_of(field.did);
        let is_zst =
            tcx.layout_of(param_env.and(field_ty)).map(|layout| layout.is_zst()).unwrap_or(false);

        if !is_zst {
            return Some(field);
        }
    }

    None
}

J
jumbatm 已提交
660
/// Is type known to be non-null?
661
crate fn ty_is_known_nonnull<'tcx>(cx: &LateContext<'tcx>, ty: Ty<'tcx>, mode: CItemKind) -> bool {
J
jumbatm 已提交
662
    let tcx = cx.tcx;
L
LeSeulArtichaut 已提交
663
    match ty.kind() {
J
jumbatm 已提交
664 665 666 667
        ty::FnPtr(_) => true,
        ty::Ref(..) => true,
        ty::Adt(def, _) if def.is_box() && matches!(mode, CItemKind::Definition) => true,
        ty::Adt(def, substs) if def.repr.transparent() && !def.is_union() => {
J
jumbatm 已提交
668
            let marked_non_null = nonnull_optimization_guaranteed(tcx, &def);
J
jumbatm 已提交
669

670
            if marked_non_null {
J
jumbatm 已提交
671
                return true;
672
            }
673

J
jumbatm 已提交
674
            for variant in &def.variants {
675
                if let Some(field) = transparent_newtype_field(cx.tcx, variant) {
J
jumbatm 已提交
676 677
                    if ty_is_known_nonnull(cx, field.ty(tcx, substs), mode) {
                        return true;
678
                    }
679
                }
680
            }
J
jumbatm 已提交
681 682

            false
683
        }
J
jumbatm 已提交
684
        _ => false,
685
    }
J
jumbatm 已提交
686
}
687

J
jumbatm 已提交
688 689 690 691
/// Given a non-null scalar (or transparent) type `ty`, return the nullable version of that type.
/// If the type passed in was not scalar, returns None.
fn get_nullable_type<'tcx>(cx: &LateContext<'tcx>, ty: Ty<'tcx>) -> Option<Ty<'tcx>> {
    let tcx = cx.tcx;
L
LeSeulArtichaut 已提交
692
    Some(match *ty.kind() {
J
jumbatm 已提交
693
        ty::Adt(field_def, field_substs) => {
J
jumbatm 已提交
694 695
            let inner_field_ty = {
                let first_non_zst_ty =
696
                    field_def.variants.iter().filter_map(|v| transparent_newtype_field(cx.tcx, v));
J
jumbatm 已提交
697 698 699 700 701 702 703 704 705 706 707
                debug_assert_eq!(
                    first_non_zst_ty.clone().count(),
                    1,
                    "Wrong number of fields for transparent type"
                );
                first_non_zst_ty
                    .last()
                    .expect("No non-zst fields in transparent type.")
                    .ty(tcx, field_substs)
            };
            return get_nullable_type(cx, inner_field_ty);
J
jumbatm 已提交
708 709 710 711 712 713 714 715 716 717 718 719
        }
        ty::Int(ty) => tcx.mk_mach_int(ty),
        ty::Uint(ty) => tcx.mk_mach_uint(ty),
        ty::RawPtr(ty_mut) => tcx.mk_ptr(ty_mut),
        // As these types are always non-null, the nullable equivalent of
        // Option<T> of these types are their raw pointer counterparts.
        ty::Ref(_region, ty, mutbl) => tcx.mk_ptr(ty::TypeAndMut { ty, mutbl }),
        ty::FnPtr(..) => {
            // There is no nullable equivalent for Rust's function pointers -- you
            // must use an Option<fn(..) -> _> to represent it.
            ty
        }
720

J
jumbatm 已提交
721 722 723
        // We should only ever reach this case if ty_is_known_nonnull is extended
        // to other types.
        ref unhandled => {
J
jumbatm 已提交
724 725 726 727 728
            debug!(
                "get_nullable_type: Unhandled scalar kind: {:?} while checking {:?}",
                unhandled, ty
            );
            return None;
J
jumbatm 已提交
729
        }
J
jumbatm 已提交
730
    })
J
jumbatm 已提交
731 732 733
}

/// Check if this enum can be safely exported based on the "nullable pointer optimization". If it
Y
yuk1ty 已提交
734
/// can, return the type that `ty` can be safely converted to, otherwise return `None`.
J
jumbatm 已提交
735 736 737 738 739 740 741 742 743
/// Currently restricted to function pointers, boxes, references, `core::num::NonZero*`,
/// `core::ptr::NonNull`, and `#[repr(transparent)]` newtypes.
/// FIXME: This duplicates code in codegen.
crate fn repr_nullable_ptr<'tcx>(
    cx: &LateContext<'tcx>,
    ty: Ty<'tcx>,
    ckind: CItemKind,
) -> Option<Ty<'tcx>> {
    debug!("is_repr_nullable_ptr(cx, ty = {:?})", ty);
L
LeSeulArtichaut 已提交
744
    if let ty::Adt(ty_def, substs) = ty.kind() {
745
        if ty_def.variants.len() != 2 {
746
            return None;
747
        }
748

749 750 751 752 753 754 755
        let get_variant_fields = |index| &ty_def.variants[VariantIdx::new(index)].fields;
        let variant_fields = [get_variant_fields(0), get_variant_fields(1)];
        let fields = if variant_fields[0].is_empty() {
            &variant_fields[1]
        } else if variant_fields[1].is_empty() {
            &variant_fields[0]
        } else {
756
            return None;
757
        };
758

759
        if fields.len() != 1 {
760
            return None;
761
        }
762

J
jumbatm 已提交
763 764
        let field_ty = fields[0].ty(cx.tcx, substs);
        if !ty_is_known_nonnull(cx, field_ty, ckind) {
765
            return None;
766
        }
767

J
jumbatm 已提交
768 769 770 771
        // At this point, the field's type is known to be nonnull and the parent enum is Option-like.
        // If the computed size for the field and the enum are different, the nonnull optimization isn't
        // being applied (and we've got a problem somewhere).
        let compute_size_skeleton = |t| SizeSkeleton::compute(t, cx.tcx, cx.param_env).unwrap();
772 773 774
        if !compute_size_skeleton(ty).same_size(compute_size_skeleton(field_ty)) {
            bug!("improper_ctypes: Option nonnull optimization not applied?");
        }
775

J
jumbatm 已提交
776 777 778 779
        // Return the nullable type this Option-like enum can be safely represented with.
        let field_ty_abi = &cx.layout_of(field_ty).unwrap().abi;
        if let Abi::Scalar(field_ty_scalar) = field_ty_abi {
            match (field_ty_scalar.valid_range.start(), field_ty_scalar.valid_range.end()) {
J
jumbatm 已提交
780
                (0, _) => unreachable!("Non-null optimisation extended to a non-zero value."),
J
jumbatm 已提交
781
                (1, _) => {
J
jumbatm 已提交
782
                    return Some(get_nullable_type(cx, field_ty).unwrap());
J
jumbatm 已提交
783 784 785 786
                }
                (start, end) => unreachable!("Unhandled start and end range: ({}, {})", start, end),
            };
        }
787
    }
J
jumbatm 已提交
788 789
    None
}
790

J
jumbatm 已提交
791
impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
E
Elichai Turkel 已提交
792 793
    /// Check if the type is array and emit an unsafe type lint.
    fn check_for_array_ty(&mut self, sp: Span, ty: Ty<'tcx>) -> bool {
L
LeSeulArtichaut 已提交
794
        if let ty::Array(..) = ty.kind() {
E
Elichai Turkel 已提交
795 796 797 798 799 800 801 802 803 804 805 806
            self.emit_ffi_unsafe_type_lint(
                ty,
                sp,
                "passing raw arrays by value is not FFI-safe",
                Some("consider passing a pointer to the array"),
            );
            true
        } else {
            false
        }
    }

807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822
    /// Checks if the given field's type is "ffi-safe".
    fn check_field_type_for_ffi(
        &self,
        cache: &mut FxHashSet<Ty<'tcx>>,
        field: &ty::FieldDef,
        substs: SubstsRef<'tcx>,
    ) -> FfiResult<'tcx> {
        let field_ty = field.ty(self.cx.tcx, substs);
        if field_ty.has_opaque_types() {
            self.check_type_for_ffi(cache, field_ty)
        } else {
            let field_ty = self.cx.tcx.normalize_erasing_regions(self.cx.param_env, field_ty);
            self.check_type_for_ffi(cache, field_ty)
        }
    }

823 824 825 826 827 828 829 830 831 832 833 834 835 836
    /// Checks if the given `VariantDef`'s field types are "ffi-safe".
    fn check_variant_for_ffi(
        &self,
        cache: &mut FxHashSet<Ty<'tcx>>,
        ty: Ty<'tcx>,
        def: &ty::AdtDef,
        variant: &ty::VariantDef,
        substs: SubstsRef<'tcx>,
    ) -> FfiResult<'tcx> {
        use FfiResult::*;

        if def.repr.transparent() {
            // Can assume that only one field is not a ZST, so only check
            // that field's type for FFI-safety.
837
            if let Some(field) = transparent_newtype_field(self.cx.tcx, variant) {
838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866
                self.check_field_type_for_ffi(cache, field, substs)
            } else {
                bug!("malformed transparent type");
            }
        } else {
            // We can't completely trust repr(C) markings; make sure the fields are
            // actually safe.
            let mut all_phantom = !variant.fields.is_empty();
            for field in &variant.fields {
                match self.check_field_type_for_ffi(cache, &field, substs) {
                    FfiSafe => {
                        all_phantom = false;
                    }
                    FfiPhantom(..) if def.is_enum() => {
                        return FfiUnsafe {
                            ty,
                            reason: "this enum contains a PhantomData field".into(),
                            help: None,
                        };
                    }
                    FfiPhantom(..) => {}
                    r => return r,
                }
            }

            if all_phantom { FfiPhantom(ty) } else { FfiSafe }
        }
    }

A
Alexander Regueiro 已提交
867
    /// Checks if the given type is "ffi-safe" (has a stable, well-defined
868
    /// representation which can be exported to C code).
M
Mark Rousskov 已提交
869
    fn check_type_for_ffi(&self, cache: &mut FxHashSet<Ty<'tcx>>, ty: Ty<'tcx>) -> FfiResult<'tcx> {
T
Taiki Endo 已提交
870
        use FfiResult::*;
871

J
jumbatm 已提交
872
        let tcx = self.cx.tcx;
873 874 875 876

        // Protect against infinite recursion, for example
        // `struct S(*mut S);`.
        // FIXME: A recursion limit is necessary as well, for irregular
M
Matthias Krüger 已提交
877
        // recursive types.
878 879 880 881
        if !cache.insert(ty) {
            return FfiSafe;
        }

L
LeSeulArtichaut 已提交
882
        match ty.kind() {
J
jumbatm 已提交
883
            ty::Adt(def, _) if def.is_box() && matches!(self.mode, CItemKind::Definition) => {
884 885 886
                FfiSafe
            }

V
varkor 已提交
887
            ty::Adt(def, substs) => {
888
                if def.is_phantom_data() {
889
                    return FfiPhantom(ty);
890
                }
891
                match def.adt_kind() {
892 893 894
                    AdtKind::Struct | AdtKind::Union => {
                        let kind = if def.is_struct() { "struct" } else { "union" };

R
Robin Kruppe 已提交
895
                        if !def.repr.c() && !def.repr.transparent() {
896
                            return FfiUnsafe {
897
                                ty,
898 899
                                reason: format!("this {} has unspecified layout", kind),
                                help: Some(format!(
M
Mark Rousskov 已提交
900
                                    "consider adding a `#[repr(C)]` or \
901 902 903
                                             `#[repr(transparent)]` attribute to this {}",
                                    kind
                                )),
904
                            };
905
                        }
906

907 908 909 910 911
                        let is_non_exhaustive =
                            def.non_enum_variant().is_field_list_non_exhaustive();
                        if is_non_exhaustive && !def.did.is_local() {
                            return FfiUnsafe {
                                ty,
912
                                reason: format!("this {} is non-exhaustive", kind),
913 914 915 916
                                help: None,
                            };
                        }

917
                        if def.non_enum_variant().fields.is_empty() {
918
                            return FfiUnsafe {
919
                                ty,
920 921
                                reason: format!("this {} has no fields", kind),
                                help: Some(format!("consider adding a member to this {}", kind)),
922
                            };
923 924
                        }

925
                        self.check_variant_for_ffi(cache, ty, def, def.non_enum_variant(), substs)
V
Vadim Petrochenkov 已提交
926
                    }
927 928 929 930 931
                    AdtKind::Enum => {
                        if def.variants.is_empty() {
                            // Empty enums are okay... although sort of useless.
                            return FfiSafe;
                        }
932

933 934
                        // Check for a repr() attribute to specify the size of the
                        // discriminant.
935
                        if !def.repr.c() && !def.repr.transparent() && def.repr.int.is_none() {
936
                            // Special-case types like `Option<extern fn()>`.
J
jumbatm 已提交
937
                            if repr_nullable_ptr(self.cx, ty, self.mode).is_none() {
938
                                return FfiUnsafe {
939
                                    ty,
940
                                    reason: "enum has no representation hint".into(),
M
Mark Rousskov 已提交
941 942
                                    help: Some(
                                        "consider adding a `#[repr(C)]`, \
943
                                                `#[repr(transparent)]`, or integer `#[repr(...)]` \
944 945
                                                attribute to this enum"
                                            .into(),
M
Mark Rousskov 已提交
946
                                    ),
947
                                };
948
                            }
949
                        }
950

951 952 953
                        if def.is_variant_list_non_exhaustive() && !def.did.is_local() {
                            return FfiUnsafe {
                                ty,
954
                                reason: "this enum is non-exhaustive".into(),
955 956 957 958
                                help: None,
                            };
                        }

959 960
                        // Check the contained variants.
                        for variant in &def.variants {
961 962 963 964
                            let is_non_exhaustive = variant.is_field_list_non_exhaustive();
                            if is_non_exhaustive && !variant.def_id.is_local() {
                                return FfiUnsafe {
                                    ty,
965
                                    reason: "this enum has non-exhaustive variants".into(),
966 967 968 969
                                    help: None,
                                };
                            }

970 971 972
                            match self.check_variant_for_ffi(cache, ty, def, variant, substs) {
                                FfiSafe => (),
                                r => return r,
973
                            }
974
                        }
975

976
                        FfiSafe
977 978
                    }
                }
979
            }
980

981
            ty::Char => FfiUnsafe {
982
                ty,
983 984
                reason: "the `char` type has no C equivalent".into(),
                help: Some("consider using `u32` or `libc::wchar_t` instead".into()),
985
            },
986

987
            ty::Int(ast::IntTy::I128) | ty::Uint(ast::UintTy::U128) => FfiUnsafe {
988
                ty,
989
                reason: "128-bit integers don't currently have a known stable ABI".into(),
990 991
                help: None,
            },
992

993
            // Primitive types with a stable representation.
994
            ty::Bool | ty::Int(..) | ty::Uint(..) | ty::Float(..) | ty::Never => FfiSafe,
995

V
varkor 已提交
996
            ty::Slice(_) => FfiUnsafe {
997
                ty,
998 999
                reason: "slices have no C equivalent".into(),
                help: Some("consider using a raw pointer instead".into()),
1000 1001
            },

M
Mark Rousskov 已提交
1002
            ty::Dynamic(..) => {
1003
                FfiUnsafe { ty, reason: "trait objects have no C equivalent".into(), help: None }
M
Mark Rousskov 已提交
1004
            }
1005

1006
            ty::Str => FfiUnsafe {
1007
                ty,
1008 1009
                reason: "string slices have no C equivalent".into(),
                help: Some("consider using `*const u8` and a length instead".into()),
1010 1011
            },

V
varkor 已提交
1012
            ty::Tuple(..) => FfiUnsafe {
1013
                ty,
1014 1015
                reason: "tuples have unspecified layout".into(),
                help: Some("consider using a struct instead".into()),
1016
            },
1017

1018 1019
            ty::RawPtr(ty::TypeAndMut { ty, .. }) | ty::Ref(_, ty, _)
                if {
J
jumbatm 已提交
1020
                    matches!(self.mode, CItemKind::Definition)
1021 1022 1023 1024 1025 1026
                        && ty.is_sized(self.cx.tcx.at(DUMMY_SP), self.cx.param_env)
                } =>
            {
                FfiSafe
            }

M
Mark Rousskov 已提交
1027 1028 1029
            ty::RawPtr(ty::TypeAndMut { ty, .. }) | ty::Ref(_, ty, _) => {
                self.check_type_for_ffi(cache, ty)
            }
1030

E
Elichai Turkel 已提交
1031
            ty::Array(inner_ty, _) => self.check_type_for_ffi(cache, inner_ty),
1032

V
varkor 已提交
1033
            ty::FnPtr(sig) => {
1034 1035 1036 1037 1038 1039 1040
                if self.is_internal_abi(sig.abi()) {
                    return FfiUnsafe {
                        ty,
                        reason: "this function pointer has Rust-specific calling convention".into(),
                        help: Some(
                            "consider using an `extern fn(...) -> ...` \
                                    function pointer instead"
1041
                                .into(),
1042 1043
                        ),
                    };
1044 1045
                }

J
jumbatm 已提交
1046
                let sig = tcx.erase_late_bound_regions(&sig);
K
kenta7777 已提交
1047
                if !sig.output().is_unit() {
1048
                    let r = self.check_type_for_ffi(cache, sig.output());
1049 1050
                    match r {
                        FfiSafe => {}
1051 1052 1053
                        _ => {
                            return r;
                        }
1054 1055
                    }
                }
1056
                for arg in sig.inputs() {
1057 1058 1059
                    let r = self.check_type_for_ffi(cache, arg);
                    match r {
                        FfiSafe => {}
1060 1061 1062
                        _ => {
                            return r;
                        }
1063 1064 1065 1066 1067
                    }
                }
                FfiSafe
            }

V
varkor 已提交
1068
            ty::Foreign(..) => FfiSafe,
P
Paul Lietar 已提交
1069

1070 1071 1072
            // While opaque types are checked for earlier, if a projection in a struct field
            // normalizes to an opaque type, then it will reach this branch.
            ty::Opaque(..) => {
1073
                FfiUnsafe { ty, reason: "opaque types have no C equivalent".into(), help: None }
1074 1075
            }

1076 1077
            // `extern "C" fn` functions can have type parameters, which may or may not be FFI-safe,
            //  so they are currently ignored for the purposes of this lint.
J
jumbatm 已提交
1078
            ty::Param(..) | ty::Projection(..) if matches!(self.mode, CItemKind::Definition) => {
1079 1080
                FfiSafe
            }
1081

1082 1083 1084
            ty::Param(..)
            | ty::Projection(..)
            | ty::Infer(..)
M
Mark Rousskov 已提交
1085
            | ty::Bound(..)
1086
            | ty::Error(_)
M
Mark Rousskov 已提交
1087 1088 1089 1090 1091
            | ty::Closure(..)
            | ty::Generator(..)
            | ty::GeneratorWitness(..)
            | ty::Placeholder(..)
            | ty::FnDef(..) => bug!("unexpected type in foreign function: {:?}", ty),
1092 1093 1094
        }
    }

V
varkor 已提交
1095 1096 1097 1098 1099 1100 1101
    fn emit_ffi_unsafe_type_lint(
        &mut self,
        ty: Ty<'tcx>,
        sp: Span,
        note: &str,
        help: Option<&str>,
    ) {
1102
        let lint = match self.mode {
J
jumbatm 已提交
1103 1104
            CItemKind::Declaration => IMPROPER_CTYPES,
            CItemKind::Definition => IMPROPER_CTYPES_DEFINITIONS,
1105 1106 1107 1108
        };

        self.cx.struct_span_lint(lint, sp, |lint| {
            let item_description = match self.mode {
J
jumbatm 已提交
1109 1110
                CItemKind::Declaration => "block",
                CItemKind::Definition => "fn",
1111 1112 1113 1114 1115
            };
            let mut diag = lint.build(&format!(
                "`extern` {} uses type `{}`, which is not FFI-safe",
                item_description, ty
            ));
J
jumbatm 已提交
1116 1117 1118
            diag.span_label(sp, "not FFI-safe");
            if let Some(help) = help {
                diag.help(help);
V
varkor 已提交
1119
            }
J
jumbatm 已提交
1120
            diag.note(note);
L
LeSeulArtichaut 已提交
1121
            if let ty::Adt(def, _) = ty.kind() {
J
jumbatm 已提交
1122 1123 1124 1125 1126 1127
                if let Some(sp) = self.cx.tcx.hir().span_if_local(def.did) {
                    diag.span_note(sp, "the type is defined here");
                }
            }
            diag.emit();
        });
V
varkor 已提交
1128 1129
    }

1130
    fn check_for_opaque_ty(&mut self, sp: Span, ty: Ty<'tcx>) -> bool {
1131
        struct ProhibitOpaqueTypes<'a, 'tcx> {
1132
            cx: &'a LateContext<'tcx>,
V
varkor 已提交
1133
            ty: Option<Ty<'tcx>>,
V
varkor 已提交
1134 1135
        };

1136
        impl<'a, 'tcx> ty::fold::TypeVisitor<'tcx> for ProhibitOpaqueTypes<'a, 'tcx> {
V
varkor 已提交
1137
            fn visit_ty(&mut self, ty: Ty<'tcx>) -> bool {
L
LeSeulArtichaut 已提交
1138
                match ty.kind() {
1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152
                    ty::Opaque(..) => {
                        self.ty = Some(ty);
                        true
                    }
                    // Consider opaque types within projections FFI-safe if they do not normalize
                    // to more opaque types.
                    ty::Projection(..) => {
                        let ty = self.cx.tcx.normalize_erasing_regions(self.cx.param_env, ty);

                        // If `ty` is a opaque type directly then `super_visit_with` won't invoke
                        // this function again.
                        if ty.has_opaque_types() { self.visit_ty(ty) } else { false }
                    }
                    _ => ty.super_visit_with(self),
V
varkor 已提交
1153 1154 1155 1156
                }
            }
        }

1157
        let mut visitor = ProhibitOpaqueTypes { cx: self.cx, ty: None };
V
varkor 已提交
1158 1159
        ty.visit_with(&mut visitor);
        if let Some(ty) = visitor.ty {
M
Mark Rousskov 已提交
1160
            self.emit_ffi_unsafe_type_lint(ty, sp, "opaque types have no C equivalent", None);
V
varkor 已提交
1161 1162 1163 1164
            true
        } else {
            false
        }
V
varkor 已提交
1165 1166
    }

1167 1168 1169 1170 1171 1172 1173
    fn check_type_for_ffi_and_report_errors(
        &mut self,
        sp: Span,
        ty: Ty<'tcx>,
        is_static: bool,
        is_return_type: bool,
    ) {
V
varkor 已提交
1174 1175
        // We have to check for opaque types before `normalize_erasing_regions`,
        // which will replace opaque types with their underlying concrete type.
1176
        if self.check_for_opaque_ty(sp, ty) {
V
varkor 已提交
1177 1178 1179 1180
            // We've already emitted an error due to an opaque type.
            return;
        }

1181 1182
        // it is only OK to use this function because extern fns cannot have
        // any generic types right now:
1183
        let ty = self.cx.tcx.normalize_erasing_regions(self.cx.param_env, ty);
1184 1185 1186 1187

        // C doesn't really support passing arrays by value - the only way to pass an array by value
        // is through a struct. So, first test that the top level isn't an array, and then
        // recursively check the types inside.
E
Elichai Turkel 已提交
1188 1189 1190
        if !is_static && self.check_for_array_ty(sp, ty) {
            return;
        }
1191

1192 1193 1194 1195 1196 1197 1198
        // Don't report FFI errors for unit return types. This check exists here, and not in
        // `check_foreign_fn` (where it would make more sense) so that normalization has definitely
        // happened.
        if is_return_type && ty.is_unit() {
            return;
        }

1199
        match self.check_type_for_ffi(&mut FxHashSet::default(), ty) {
1200
            FfiResult::FfiSafe => {}
1201
            FfiResult::FfiPhantom(ty) => {
1202
                self.emit_ffi_unsafe_type_lint(ty, sp, "composed only of `PhantomData`", None);
V
Vadim Petrochenkov 已提交
1203
            }
1204 1205
            // If `ty` is a `repr(transparent)` newtype, and the non-zero-sized type is a generic
            // argument, which after substitution, is `()`, then this branch can be hit.
1206
            FfiResult::FfiUnsafe { ty, .. } if is_return_type && ty.is_unit() => {}
V
varkor 已提交
1207
            FfiResult::FfiUnsafe { ty, reason, help } => {
1208
                self.emit_ffi_unsafe_type_lint(ty, sp, &reason, help.as_deref());
1209 1210 1211 1212
            }
        }
    }

C
Camille GILLOT 已提交
1213
    fn check_foreign_fn(&mut self, id: hir::HirId, decl: &hir::FnDecl<'_>) {
1214
        let def_id = self.cx.tcx.hir().local_def_id(id);
1215
        let sig = self.cx.tcx.fn_sig(def_id);
1216 1217
        let sig = self.cx.tcx.erase_late_bound_regions(&sig);

C
Camille GILLOT 已提交
1218
        for (input_ty, input_hir) in sig.inputs().iter().zip(decl.inputs) {
1219
            self.check_type_for_ffi_and_report_errors(input_hir.span, input_ty, false, false);
1220 1221
        }

1222
        if let hir::FnRetTy::Return(ref ret_hir) = decl.output {
1223
            let ret_ty = sig.output();
1224
            self.check_type_for_ffi_and_report_errors(ret_hir.span, ret_ty, false, true);
1225 1226
        }
    }
1227

L
ljedrz 已提交
1228
    fn check_foreign_static(&mut self, id: hir::HirId, span: Span) {
1229
        let def_id = self.cx.tcx.hir().local_def_id(id);
1230
        let ty = self.cx.tcx.type_of(def_id);
1231
        self.check_type_for_ffi_and_report_errors(span, ty, true, false);
D
David Wood 已提交
1232
    }
1233

J
jumbatm 已提交
1234 1235 1236 1237 1238 1239
    fn is_internal_abi(&self, abi: SpecAbi) -> bool {
        if let SpecAbi::Rust
        | SpecAbi::RustCall
        | SpecAbi::RustIntrinsic
        | SpecAbi::PlatformIntrinsic = abi
        {
1240 1241 1242 1243 1244
            true
        } else {
            false
        }
    }
1245 1246
}

1247 1248
impl<'tcx> LateLintPass<'tcx> for ImproperCTypesDeclarations {
    fn check_foreign_item(&mut self, cx: &LateContext<'_>, it: &hir::ForeignItem<'_>) {
J
jumbatm 已提交
1249
        let mut vis = ImproperCTypesVisitor { cx, mode: CItemKind::Declaration };
1250
        let abi = cx.tcx.hir().get_foreign_abi(it.hir_id);
1251 1252

        if !vis.is_internal_abi(abi) {
1253
            match it.kind {
1254
                hir::ForeignItemKind::Fn(ref decl, _, _) => {
1255
                    vis.check_foreign_fn(it.hir_id, decl);
1256 1257
                }
                hir::ForeignItemKind::Static(ref ty, _) => {
L
ljedrz 已提交
1258
                    vis.check_foreign_static(it.hir_id, ty.span);
1259
                }
M
Mark Rousskov 已提交
1260
                hir::ForeignItemKind::Type => (),
1261 1262 1263 1264
            }
        }
    }
}
1265

1266
impl<'tcx> LateLintPass<'tcx> for ImproperCTypesDefinitions {
1267 1268
    fn check_fn(
        &mut self,
1269
        cx: &LateContext<'tcx>,
1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283
        kind: hir::intravisit::FnKind<'tcx>,
        decl: &'tcx hir::FnDecl<'_>,
        _: &'tcx hir::Body<'_>,
        _: Span,
        hir_id: hir::HirId,
    ) {
        use hir::intravisit::FnKind;

        let abi = match kind {
            FnKind::ItemFn(_, _, header, ..) => header.abi,
            FnKind::Method(_, sig, ..) => sig.header.abi,
            _ => return,
        };

J
jumbatm 已提交
1284
        let mut vis = ImproperCTypesVisitor { cx, mode: CItemKind::Definition };
1285 1286 1287 1288 1289 1290
        if !vis.is_internal_abi(abi) {
            vis.check_foreign_fn(hir_id, decl);
        }
    }
}

1291
declare_lint_pass!(VariantSizeDifferences => [VARIANT_SIZE_DIFFERENCES]);
1292

1293 1294
impl<'tcx> LateLintPass<'tcx> for VariantSizeDifferences {
    fn check_item(&mut self, cx: &LateContext<'_>, it: &hir::Item<'_>) {
V
varkor 已提交
1295
        if let hir::ItemKind::Enum(ref enum_definition, _) = it.kind {
1296
            let item_def_id = cx.tcx.hir().local_def_id(it.hir_id);
V
varkor 已提交
1297 1298
            let t = cx.tcx.type_of(item_def_id);
            let ty = cx.tcx.erase_regions(&t);
1299 1300
            let layout = match cx.layout_of(ty) {
                Ok(layout) => layout,
1301 1302 1303
                Err(
                    ty::layout::LayoutError::Unknown(_) | ty::layout::LayoutError::SizeOverflow(_),
                ) => return,
1304 1305
            };
            let (variants, tag) = match layout.variants {
1306
                Variants::Multiple {
R
Ralf Jung 已提交
1307 1308
                    tag_encoding: TagEncoding::Direct,
                    ref tag,
1309
                    ref variants,
1310
                    ..
R
Ralf Jung 已提交
1311
                } => (variants, tag),
1312 1313 1314
                _ => return,
            };

R
Ralf Jung 已提交
1315
            let tag_size = tag.value.size(&cx.tcx).bytes();
1316

M
Mark Rousskov 已提交
1317 1318 1319 1320 1321 1322
            debug!(
                "enum `{}` is {} bytes large with layout:\n{:#?}",
                t,
                layout.size.bytes(),
                layout
            );
1323

M
Mark Rousskov 已提交
1324 1325
            let (largest, slargest, largest_index) = enum_definition
                .variants
1326 1327 1328
                .iter()
                .zip(variants)
                .map(|(variant, variant_layout)| {
R
Ralf Jung 已提交
1329 1330
                    // Subtract the size of the enum tag.
                    let bytes = variant_layout.size.bytes().saturating_sub(tag_size);
1331

M
Mark Rousskov 已提交
1332
                    debug!("- variant `{}` is {} bytes large", variant.ident, bytes);
1333 1334 1335
                    bytes
                })
                .enumerate()
M
Mark Rousskov 已提交
1336 1337 1338 1339 1340 1341 1342 1343
                .fold((0, 0, 0), |(l, s, li), (idx, size)| {
                    if size > l {
                        (size, l, idx)
                    } else if size > s {
                        (l, size, li)
                    } else {
                        (l, s, li)
                    }
1344 1345 1346 1347 1348
                });

            // We only warn if the largest variant is at least thrice as large as
            // the second-largest.
            if largest > slargest * 3 && slargest > 0 {
J
jumbatm 已提交
1349
                cx.struct_span_lint(
M
Mark Rousskov 已提交
1350 1351
                    VARIANT_SIZE_DIFFERENCES,
                    enum_definition.variants[largest_index].span,
J
jumbatm 已提交
1352 1353 1354
                    |lint| {
                        lint.build(&format!(
                            "enum variant is more than three times \
1355
                                          larger ({} bytes) than the next largest",
J
jumbatm 已提交
1356 1357 1358 1359
                            largest
                        ))
                        .emit()
                    },
M
Mark Rousskov 已提交
1360
                );
1361 1362 1363 1364
            }
        }
    }
}