| use std::any::Any; |
| |
| /// This module provides types and traits for buffering lints until later in compilation. |
| use rustc_ast::node_id::NodeId; |
| use rustc_data_structures::fx::FxIndexMap; |
| use rustc_data_structures::sync::{DynSend, DynSync}; |
| use rustc_error_messages::MultiSpan; |
| use rustc_lint_defs::{AttributeLintKind, Lint, LintId}; |
| |
| use crate::{Diag, DiagCtxtHandle, Diagnostic, Level}; |
| |
| /// We can't implement `Diagnostic` for `AttributeLintKind`, because decorating some of its |
| /// variants requires types we don't have yet. So, handle that case separately. |
| pub enum DecorateDiagCompat { |
| /// The third argument of the closure is a `Session`. However, due to the dependency tree, |
| /// we don't have access to `rustc_session` here, so we downcast it when needed. |
| Dynamic( |
| Box< |
| dyn for<'a> FnOnce(DiagCtxtHandle<'a>, Level, &dyn Any) -> Diag<'a, ()> |
| + DynSync |
| + DynSend |
| + 'static, |
| >, |
| ), |
| Builtin(AttributeLintKind), |
| } |
| |
| impl std::fmt::Debug for DecorateDiagCompat { |
| fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |
| f.debug_struct("DecorateDiagCompat").finish() |
| } |
| } |
| |
| impl<D: for<'a> Diagnostic<'a, ()> + DynSync + DynSend + 'static> From<D> for DecorateDiagCompat { |
| #[inline] |
| fn from(d: D) -> Self { |
| Self::Dynamic(Box::new(|dcx, level, _| d.into_diag(dcx, level))) |
| } |
| } |
| |
| impl From<AttributeLintKind> for DecorateDiagCompat { |
| #[inline] |
| fn from(b: AttributeLintKind) -> Self { |
| Self::Builtin(b) |
| } |
| } |
| |
| /// Lints that are buffered up early on in the `Session` before the |
| /// `LintLevels` is calculated. |
| #[derive(Debug)] |
| pub struct BufferedEarlyLint { |
| /// The span of code that we are linting on. |
| pub span: Option<MultiSpan>, |
| |
| /// The `NodeId` of the AST node that generated the lint. |
| pub node_id: NodeId, |
| |
| /// A lint Id that can be passed to |
| /// `rustc_lint::early::EarlyContextAndPass::check_id`. |
| pub lint_id: LintId, |
| |
| /// Customization of the `Diag<'_>` for the lint. |
| pub diagnostic: DecorateDiagCompat, |
| } |
| |
| #[derive(Default, Debug)] |
| pub struct LintBuffer { |
| pub map: FxIndexMap<NodeId, Vec<BufferedEarlyLint>>, |
| } |
| |
| impl LintBuffer { |
| pub fn add_early_lint(&mut self, early_lint: BufferedEarlyLint) { |
| self.map.entry(early_lint.node_id).or_default().push(early_lint); |
| } |
| |
| pub fn take(&mut self, id: NodeId) -> Vec<BufferedEarlyLint> { |
| // FIXME(#120456) - is `swap_remove` correct? |
| self.map.swap_remove(&id).unwrap_or_default() |
| } |
| |
| pub fn buffer_lint( |
| &mut self, |
| lint: &'static Lint, |
| node_id: NodeId, |
| span: impl Into<MultiSpan>, |
| decorate: impl Into<DecorateDiagCompat>, |
| ) { |
| self.add_early_lint(BufferedEarlyLint { |
| lint_id: LintId::of(lint), |
| node_id, |
| span: Some(span.into()), |
| diagnostic: decorate.into(), |
| }); |
| } |
| |
| pub fn dyn_buffer_lint< |
| F: for<'a> FnOnce(DiagCtxtHandle<'a>, Level) -> Diag<'a, ()> + DynSync + DynSend + 'static, |
| >( |
| &mut self, |
| lint: &'static Lint, |
| node_id: NodeId, |
| span: impl Into<MultiSpan>, |
| callback: F, |
| ) { |
| self.add_early_lint(BufferedEarlyLint { |
| lint_id: LintId::of(lint), |
| node_id, |
| span: Some(span.into()), |
| diagnostic: DecorateDiagCompat::Dynamic(Box::new(|dcx, level, _| callback(dcx, level))), |
| }); |
| } |
| |
| pub fn dyn_buffer_lint_any< |
| F: for<'a> FnOnce(DiagCtxtHandle<'a>, Level, &dyn Any) -> Diag<'a, ()> |
| + DynSend |
| + DynSync |
| + 'static, |
| >( |
| &mut self, |
| lint: &'static Lint, |
| node_id: NodeId, |
| span: impl Into<MultiSpan>, |
| callback: F, |
| ) { |
| self.add_early_lint(BufferedEarlyLint { |
| lint_id: LintId::of(lint), |
| node_id, |
| span: Some(span.into()), |
| diagnostic: DecorateDiagCompat::Dynamic(Box::new(callback)), |
| }); |
| } |
| } |