Skip to content

Commit 6f6d73b

Browse files
committed
Move report_lit_error from rustc_session to rustc_parse.
It's a more logical spot for it, and will be a big help for the next commit. Doing this creates a new dependency from `rustc_ast_lowering` on `rustc_parse`, but `rustc_ast_lowering` is clearly higher up the crate graph, so this isn't a big deal. One thing in favour of this change, is that two fluent labels were duplicated across `rustc_session` and `rustc_parse`: `invalid_literal_suffix` and `parse_not_supported`. This duplication is now gone, so that's nice evidence that this is a reasonable change.
1 parent 8769da6 commit 6f6d73b

File tree

12 files changed

+216
-214
lines changed

12 files changed

+216
-214
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -3457,6 +3457,7 @@ dependencies = [
34573457
"rustc_index",
34583458
"rustc_macros",
34593459
"rustc_middle",
3460+
"rustc_parse",
34603461
"rustc_session",
34613462
"rustc_span",
34623463
"rustc_target",

compiler/rustc_ast_lowering/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ rustc_hir = { path = "../rustc_hir" }
1717
rustc_index = { path = "../rustc_index" }
1818
rustc_macros = { path = "../rustc_macros" }
1919
rustc_middle = { path = "../rustc_middle" }
20+
rustc_parse = { path = "../rustc_parse" }
2021
rustc_session = { path = "../rustc_session" }
2122
rustc_span = { path = "../rustc_span" }
2223
rustc_target = { path = "../rustc_target" }

compiler/rustc_ast_lowering/src/expr.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use rustc_data_structures::stack::ensure_sufficient_stack;
1414
use rustc_hir as hir;
1515
use rustc_hir::def::{DefKind, Res};
1616
use rustc_middle::span_bug;
17-
use rustc_session::errors::report_lit_error;
17+
use rustc_parse::parser::report_lit_error;
1818
use rustc_span::source_map::{respan, Spanned};
1919
use rustc_span::symbol::{kw, sym, Ident, Symbol};
2020
use rustc_span::DUMMY_SP;

compiler/rustc_builtin_macros/src/concat.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use rustc_ast as ast;
22
use rustc_ast::tokenstream::TokenStream;
33
use rustc_expand::base::{self, DummyResult};
4-
use rustc_session::errors::report_lit_error;
4+
use rustc_parse::parser::report_lit_error;
55
use rustc_span::symbol::Symbol;
66

77
use crate::errors;

compiler/rustc_builtin_macros/src/concat_bytes.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use rustc_ast as ast;
22
use rustc_ast::{ptr::P, tokenstream::TokenStream};
33
use rustc_expand::base::{self, DummyResult};
4-
use rustc_session::errors::report_lit_error;
4+
use rustc_parse::parser::report_lit_error;
55
use rustc_span::Span;
66

77
use crate::errors;

compiler/rustc_expand/src/base.rs

+2-3
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,7 @@ use rustc_errors::{
2121
use rustc_feature::Features;
2222
use rustc_lint_defs::builtin::PROC_MACRO_BACK_COMPAT;
2323
use rustc_lint_defs::{BufferedEarlyLint, BuiltinLintDiagnostics, RegisteredTools};
24-
use rustc_parse::{parser, MACRO_ARGUMENTS};
25-
use rustc_session::errors::report_lit_error;
24+
use rustc_parse::{self, parser, MACRO_ARGUMENTS};
2625
use rustc_session::{parse::ParseSess, Limit, Session};
2726
use rustc_span::def_id::{CrateNum, DefId, LocalDefId};
2827
use rustc_span::edition::Edition;
@@ -1251,7 +1250,7 @@ pub fn expr_to_spanned_string<'a>(
12511250
}
12521251
Ok(ast::LitKind::Err) => None,
12531252
Err(err) => {
1254-
report_lit_error(&cx.sess.parse_sess, err, token_lit, expr.span);
1253+
parser::report_lit_error(&cx.sess.parse_sess, err, token_lit, expr.span);
12551254
None
12561255
}
12571256
_ => Some((cx.struct_span_err(expr.span, err_msg), false)),

compiler/rustc_parse/messages.ftl

+29
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,8 @@ parse_bare_cr = {$double_quotes ->
5959
6060
parse_bare_cr_in_raw_string = bare CR not allowed in raw string
6161
62+
parse_binary_float_literal_not_supported = binary float literal is not supported
63+
6264
parse_bounds_not_allowed_on_trait_aliases = bounds are not allowed on trait aliases
6365
6466
parse_box_not_pat = expected pattern, found {$descr}
@@ -292,7 +294,11 @@ parse_generic_parameters_without_angle_brackets = generic parameters without sur
292294
parse_generics_in_path = unexpected generic arguments in path
293295
294296
parse_help_set_edition_cargo = set `edition = "{$edition}"` in `Cargo.toml`
297+
295298
parse_help_set_edition_standalone = pass `--edition {$edition}` to `rustc`
299+
300+
parse_hexadecimal_float_literal_not_supported = hexadecimal float literal is not supported
301+
296302
parse_if_expression_missing_condition = missing condition for `if` expression
297303
.condition_label = expected condition here
298304
.block_label = if this block is the condition of the `if` expression, then it must be followed by another block
@@ -364,6 +370,9 @@ parse_inner_doc_comment_not_permitted = expected outer doc comment
364370
.label_does_not_annotate_this = the inner doc comment doesn't annotate this {$item}
365371
.sugg_change_inner_to_outer = to annotate the {$item}, change the doc comment from inner to outer style
366372
373+
parse_int_literal_too_large = integer literal is too large
374+
.note = value exceeds limit of `{$limit}`
375+
367376
parse_invalid_block_macro_segment = cannot use a `block` macro fragment here
368377
.label = the `block` fragment is within this context
369378
.suggestion = wrap this in another block
@@ -388,8 +397,18 @@ parse_invalid_dyn_keyword = invalid `dyn` keyword
388397
.suggestion = remove this keyword
389398
390399
parse_invalid_expression_in_let_else = a `{$operator}` expression cannot be directly assigned in `let...else`
400+
parse_invalid_float_literal_suffix = invalid suffix `{$suffix}` for float literal
401+
.label = invalid suffix `{$suffix}`
402+
.help = valid suffixes are `f32` and `f64`
403+
404+
parse_invalid_float_literal_width = invalid width `{$width}` for float literal
405+
.help = valid widths are 32 and 64
406+
391407
parse_invalid_identifier_with_leading_number = identifiers cannot start with a number
392408
409+
parse_invalid_int_literal_width = invalid width `{$width}` for integer literal
410+
.help = valid widths are 8, 16, 32, 64 and 128
411+
393412
parse_invalid_interpolated_expression = invalid interpolated expression
394413
395414
parse_invalid_literal_suffix = suffixes on {$kind} literals are invalid
@@ -408,6 +427,14 @@ parse_invalid_logical_operator = `{$incorrect}` is not a logical operator
408427
409428
parse_invalid_meta_item = expected unsuffixed literal or identifier, found `{$token}`
410429
430+
parse_invalid_num_literal_base_prefix = invalid base prefix for number literal
431+
.note = base prefixes (`0xff`, `0b1010`, `0o755`) are lowercase
432+
.suggestion = try making the prefix lowercase
433+
434+
parse_invalid_num_literal_suffix = invalid suffix `{$suffix}` for number literal
435+
.label = invalid suffix `{$suffix}`
436+
.help = the suffix must be one of the numeric types (`u32`, `isize`, `f32`, etc.)
437+
411438
parse_invalid_unicode_escape = invalid unicode character escape
412439
.label = invalid escape
413440
.help = unicode escape must {$surrogate ->
@@ -614,6 +641,8 @@ parse_note_pattern_alternatives_use_single_vert = alternatives in or-patterns ar
614641
615642
parse_nul_in_c_str = null characters in C string literals are not supported
616643
644+
parse_octal_float_literal_not_supported = octal float literal is not supported
645+
617646
parse_or_pattern_not_allowed_in_fn_parameters = top-level or-patterns are not allowed in function parameters
618647
parse_or_pattern_not_allowed_in_let_binding = top-level or-patterns are not allowed in `let` bindings
619648
parse_out_of_range_hex_escape = out of range hex escape

compiler/rustc_parse/src/errors.rs

+92
Original file line numberDiff line numberDiff line change
@@ -2902,3 +2902,95 @@ pub(crate) struct TransposeDynOrImplSugg<'a> {
29022902
pub insertion_span: Span,
29032903
pub kw: &'a str,
29042904
}
2905+
2906+
#[derive(Diagnostic)]
2907+
#[diag(parse_invalid_literal_suffix)]
2908+
pub(crate) struct InvalidLiteralSuffix<'a> {
2909+
#[primary_span]
2910+
#[label]
2911+
pub span: Span,
2912+
// FIXME(#100717)
2913+
pub kind: &'a str,
2914+
pub suffix: Symbol,
2915+
}
2916+
2917+
#[derive(Diagnostic)]
2918+
#[diag(parse_invalid_int_literal_width)]
2919+
#[help]
2920+
pub(crate) struct InvalidIntLiteralWidth {
2921+
#[primary_span]
2922+
pub span: Span,
2923+
pub width: String,
2924+
}
2925+
2926+
#[derive(Diagnostic)]
2927+
#[diag(parse_invalid_num_literal_base_prefix)]
2928+
#[note]
2929+
pub(crate) struct InvalidNumLiteralBasePrefix {
2930+
#[primary_span]
2931+
#[suggestion(applicability = "maybe-incorrect", code = "{fixed}")]
2932+
pub span: Span,
2933+
pub fixed: String,
2934+
}
2935+
2936+
#[derive(Diagnostic)]
2937+
#[diag(parse_invalid_num_literal_suffix)]
2938+
#[help]
2939+
pub(crate) struct InvalidNumLiteralSuffix {
2940+
#[primary_span]
2941+
#[label]
2942+
pub span: Span,
2943+
pub suffix: String,
2944+
}
2945+
2946+
#[derive(Diagnostic)]
2947+
#[diag(parse_invalid_float_literal_width)]
2948+
#[help]
2949+
pub(crate) struct InvalidFloatLiteralWidth {
2950+
#[primary_span]
2951+
pub span: Span,
2952+
pub width: String,
2953+
}
2954+
2955+
#[derive(Diagnostic)]
2956+
#[diag(parse_invalid_float_literal_suffix)]
2957+
#[help]
2958+
pub(crate) struct InvalidFloatLiteralSuffix {
2959+
#[primary_span]
2960+
#[label]
2961+
pub span: Span,
2962+
pub suffix: String,
2963+
}
2964+
2965+
#[derive(Diagnostic)]
2966+
#[diag(parse_hexadecimal_float_literal_not_supported)]
2967+
pub(crate) struct HexadecimalFloatLiteralNotSupported {
2968+
#[primary_span]
2969+
#[label(parse_not_supported)]
2970+
pub span: Span,
2971+
}
2972+
2973+
#[derive(Diagnostic)]
2974+
#[diag(parse_octal_float_literal_not_supported)]
2975+
pub(crate) struct OctalFloatLiteralNotSupported {
2976+
#[primary_span]
2977+
#[label(parse_not_supported)]
2978+
pub span: Span,
2979+
}
2980+
2981+
#[derive(Diagnostic)]
2982+
#[diag(parse_binary_float_literal_not_supported)]
2983+
pub(crate) struct BinaryFloatLiteralNotSupported {
2984+
#[primary_span]
2985+
#[label(parse_not_supported)]
2986+
pub span: Span,
2987+
}
2988+
2989+
#[derive(Diagnostic)]
2990+
#[diag(parse_int_literal_too_large)]
2991+
#[note]
2992+
pub(crate) struct IntLiteralTooLarge {
2993+
#[primary_span]
2994+
pub span: Span,
2995+
pub limit: String,
2996+
}

compiler/rustc_parse/src/parser/expr.rs

+87-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ use rustc_ast::token::{self, Delimiter, Token, TokenKind};
1717
use rustc_ast::tokenstream::Spacing;
1818
use rustc_ast::util::case::Case;
1919
use rustc_ast::util::classify;
20+
use rustc_ast::util::literal::LitError;
2021
use rustc_ast::util::parser::{prec_let_scrutinee_needs_par, AssocOp, Fixity};
2122
use rustc_ast::visit::Visitor;
2223
use rustc_ast::{self as ast, AttrStyle, AttrVec, CaptureBy, ExprField, UnOp, DUMMY_NODE_ID};
@@ -30,9 +31,10 @@ use rustc_errors::{
3031
PResult, StashKey,
3132
};
3233
use rustc_macros::Subdiagnostic;
33-
use rustc_session::errors::{report_lit_error, ExprParenthesesNeeded};
34+
use rustc_session::errors::ExprParenthesesNeeded;
3435
use rustc_session::lint::builtin::BREAK_WITH_LABEL_AND_LOOP;
3536
use rustc_session::lint::BuiltinLintDiagnostics;
37+
use rustc_session::parse::ParseSess;
3638
use rustc_span::source_map::{self, Spanned};
3739
use rustc_span::symbol::kw::PathRoot;
3840
use rustc_span::symbol::{kw, sym, Ident, Symbol};
@@ -3657,6 +3659,90 @@ impl<'a> Parser<'a> {
36573659
}
36583660
}
36593661

3662+
pub fn report_lit_error(sess: &ParseSess, err: LitError, lit: token::Lit, span: Span) {
3663+
// Checks if `s` looks like i32 or u1234 etc.
3664+
fn looks_like_width_suffix(first_chars: &[char], s: &str) -> bool {
3665+
s.len() > 1 && s.starts_with(first_chars) && s[1..].chars().all(|c| c.is_ascii_digit())
3666+
}
3667+
3668+
// Try to lowercase the prefix if the prefix and suffix are valid.
3669+
fn fix_base_capitalisation(prefix: &str, suffix: &str) -> Option<String> {
3670+
let mut chars = suffix.chars();
3671+
3672+
let base_char = chars.next().unwrap();
3673+
let base = match base_char {
3674+
'B' => 2,
3675+
'O' => 8,
3676+
'X' => 16,
3677+
_ => return None,
3678+
};
3679+
3680+
// check that the suffix contains only base-appropriate characters
3681+
let valid = prefix == "0"
3682+
&& chars
3683+
.filter(|c| *c != '_')
3684+
.take_while(|c| *c != 'i' && *c != 'u')
3685+
.all(|c| c.to_digit(base).is_some());
3686+
3687+
valid.then(|| format!("0{}{}", base_char.to_ascii_lowercase(), &suffix[1..]))
3688+
}
3689+
3690+
let token::Lit { kind, symbol, suffix, .. } = lit;
3691+
match err {
3692+
// `LexerError` is an error, but it was already reported
3693+
// by lexer, so here we don't report it the second time.
3694+
LitError::LexerError => {}
3695+
LitError::InvalidSuffix => {
3696+
if let Some(suffix) = suffix {
3697+
sess.emit_err(errors::InvalidLiteralSuffix { span, kind: kind.descr(), suffix });
3698+
}
3699+
}
3700+
LitError::InvalidIntSuffix => {
3701+
let suf = suffix.expect("suffix error with no suffix");
3702+
let suf = suf.as_str();
3703+
if looks_like_width_suffix(&['i', 'u'], suf) {
3704+
// If it looks like a width, try to be helpful.
3705+
sess.emit_err(errors::InvalidIntLiteralWidth { span, width: suf[1..].into() });
3706+
} else if let Some(fixed) = fix_base_capitalisation(symbol.as_str(), suf) {
3707+
sess.emit_err(errors::InvalidNumLiteralBasePrefix { span, fixed });
3708+
} else {
3709+
sess.emit_err(errors::InvalidNumLiteralSuffix { span, suffix: suf.to_string() });
3710+
}
3711+
}
3712+
LitError::InvalidFloatSuffix => {
3713+
let suf = suffix.expect("suffix error with no suffix");
3714+
let suf = suf.as_str();
3715+
if looks_like_width_suffix(&['f'], suf) {
3716+
// If it looks like a width, try to be helpful.
3717+
sess.emit_err(errors::InvalidFloatLiteralWidth {
3718+
span,
3719+
width: suf[1..].to_string(),
3720+
});
3721+
} else {
3722+
sess.emit_err(errors::InvalidFloatLiteralSuffix { span, suffix: suf.to_string() });
3723+
}
3724+
}
3725+
LitError::NonDecimalFloat(base) => {
3726+
match base {
3727+
16 => sess.emit_err(errors::HexadecimalFloatLiteralNotSupported { span }),
3728+
8 => sess.emit_err(errors::OctalFloatLiteralNotSupported { span }),
3729+
2 => sess.emit_err(errors::BinaryFloatLiteralNotSupported { span }),
3730+
_ => unreachable!(),
3731+
};
3732+
}
3733+
LitError::IntTooLarge(base) => {
3734+
let max = u128::MAX;
3735+
let limit = match base {
3736+
2 => format!("{max:#b}"),
3737+
8 => format!("{max:#o}"),
3738+
16 => format!("{max:#x}"),
3739+
_ => format!("{max}"),
3740+
};
3741+
sess.emit_err(errors::IntLiteralTooLarge { span, limit });
3742+
}
3743+
}
3744+
}
3745+
36603746
/// Used to forbid `let` expressions in certain syntactic locations.
36613747
#[derive(Clone, Copy, Subdiagnostic)]
36623748
pub(crate) enum ForbiddenLetReason {

compiler/rustc_parse/src/parser/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ mod stmt;
1111
mod ty;
1212

1313
use crate::lexer::UnmatchedDelim;
14+
pub use crate::parser::expr::report_lit_error;
1415
pub use attr_wrapper::AttrWrapper;
1516
pub use diagnostics::AttemptLocalParseRecovery;
1617
pub(crate) use expr::ForbiddenLetReason;

0 commit comments

Comments
 (0)