ink_ir/
error.rs

1// Copyright (C) Use Ink (UK) Ltd.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15//! Provide macros to simplify error reporting in procedural macros.
16
17pub trait ExtError {
18    /// Returns `self` combined with the other error.
19    fn into_combine(self, another: syn::Error) -> Self;
20}
21
22impl ExtError for syn::Error {
23    fn into_combine(mut self, another: syn::Error) -> Self {
24        self.combine(another);
25        self
26    }
27}
28
29/// Creates a [`syn::Error`] with the format message and infers the
30/// [`Span`](`proc_macro2::Span`) using [`ToTokens`](`quote::ToTokens`).
31///
32/// # Parameters
33///
34/// - The first argument must implement [`quote::ToTokens`] in order to infer a
35///   [`Span`](`proc_macro2::Span`).
36/// - The second argument is a format string.
37/// - The rest are format string arguments.
38///
39/// # Note
40///
41/// On stable Rust this might yield higher quality error span information to the user
42/// than [`format_err`](`crate::format_err`).
43/// - Source: [`syn::Error::new_spanned`](https://docs.rs/syn/1.0.33/syn/struct.Error.html#method.new_spanned)
44/// - Tracking issue: [`#54725`](https://github.com/rust-lang/rust/issues/54725)
45#[macro_export]
46macro_rules! format_err_spanned {
47    ($tokens:expr, $($msg:tt)*) => {
48        ::syn::Error::new_spanned(
49            &$tokens,
50            format_args!($($msg)*)
51        )
52    }
53}
54
55/// Creates a [`syn::Error`] with the format message and infers the
56/// [`Span`](`proc_macro2::Span`) using the [`ToTokens`](`quote::ToTokens`) implementation
57/// for the [`MetaValue`][crate::ast::MetaValue] (if possible).
58///
59/// See [`format_err_spanned`] for more details.
60macro_rules! format_err_spanned_value {
61    ($arg:expr, $($msg:tt)*) => {
62        if let Some(value) = $arg.value() {
63            format_err_spanned!(value, $($msg)*)
64        } else {
65            format_err_spanned!($arg, $($msg)*)
66        }
67    };
68}
69
70/// Creates a [`syn::Error`] with the format message and infers the
71/// [`Span`](`proc_macro2::Span`) using [`Spanned`](`syn::spanned::Spanned`).
72///
73/// # Parameters
74///
75/// - The first argument must be a type that implements [`syn::spanned::Spanned`].
76/// - The second argument is a format string.
77/// - The rest are format string arguments.
78///
79/// # Note
80///
81/// On stable Rust this might yield worse error span information to the user
82/// than [`format_err_spanned`].
83/// - Source: [`syn::Error::new_spanned`](https://docs.rs/syn/1.0.33/syn/struct.Error.html#method.new_spanned)
84/// - Tracking issue: [`#54725`](https://github.com/rust-lang/rust/issues/54725)
85#[macro_export]
86macro_rules! format_err {
87    ($spanned:expr, $($msg:tt)*) => {
88        ::syn::Error::new(
89            <_ as ::syn::spanned::Spanned>::span(&$spanned),
90            format_args!($($msg)*)
91        )
92    }
93}