Skip to content

Commit 814ddfe

Browse files
committed
Auto merge of rust-lang#131663 - veera-sivarajan:fix-128709-final, r=<try>
Evaluate `std::fmt::Arguments::new_const()` during Compile Time Fixes rust-lang#128709 This PR aims to optimize calls to string formating macros without any arguments by evaluating `std::fmt::Arguments::new_const()` in a const context. Currently, `println!("hola")` compiles to `std::io::_print(std::fmt::Arguments::new_const(&["hola\n"]))`. With this PR, `println!("hola")` compiles to `std::io::_print(const { std::fmt::Arguments::new_const(&["hola\n"]) })`. This is accomplished in two steps: 1. Const stabilize `std::fmt::Arguments::new_const()`. 2. Wrap calls to `std::fmt::Arguments::new_const()` in an inline const block when lowering the AST to HIR. This reduces the generated code to a `memcpy` instead of multiple `getelementptr` and `store` instructions even with `-C no-prepopulate-passes -C opt-level=0`. Godbolt for code comparison: https://rust.godbolt.org/z/P7Px7de6c This is a safe and sound transformation because `std::fmt::Arguments::new_const()` is a trivial constructor function taking a slice containing a `'static` string literal as input. CC rust-lang#99012
2 parents 17a19e6 + 95b1d81 commit 814ddfe

File tree

22 files changed

+152
-124
lines changed

22 files changed

+152
-124
lines changed

compiler/rustc_ast_lowering/src/format.rs

+25-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ use rustc_ast::visit::Visitor;
55
use rustc_ast::*;
66
use rustc_data_structures::fx::FxIndexMap;
77
use rustc_hir as hir;
8+
use rustc_hir::def::DefKind;
89
use rustc_session::config::FmtDebug;
910
use rustc_span::symbol::{Ident, kw};
1011
use rustc_span::{Span, Symbol, sym};
@@ -24,6 +25,28 @@ impl<'hir> LoweringContext<'_, 'hir> {
2425
expand_format_args(self, sp, &fmt, allow_const)
2526
}
2627

28+
/// Wraps given `ExprKind` in an inline const block.
29+
///
30+
/// Caller must ensure it's safe and sound to do so.
31+
fn wrap_in_const_context(
32+
&mut self,
33+
sp: Span,
34+
kind: hir::ExprKind<'hir>,
35+
) -> hir::ExprKind<'hir> {
36+
let expr = hir::Expr { hir_id: self.next_id(), kind, span: self.lower_span(sp) };
37+
let const_node_id = self.next_node_id();
38+
let parent_def_id = self.current_def_id_parent;
39+
let def_id =
40+
self.create_def(parent_def_id, const_node_id, kw::Empty, DefKind::InlineConst, sp);
41+
let hir_id = self.lower_node_id(const_node_id);
42+
let const_block = self.with_new_scopes(sp, |this| hir::ConstBlock {
43+
def_id,
44+
hir_id,
45+
body: this.with_def_id_parent(def_id, |this| this.lower_body(|_| (&[], expr))),
46+
});
47+
hir::ExprKind::ConstBlock(const_block)
48+
}
49+
2750
/// Try to convert a literal into an interned string
2851
fn try_inline_lit(&self, lit: token::Lit) -> Option<Symbol> {
2952
match LitKind::from_token_lit(lit) {
@@ -464,14 +487,14 @@ fn expand_format_args<'hir>(
464487

465488
if allow_const && arguments.is_empty() && argmap.is_empty() {
466489
// Generate:
467-
// <core::fmt::Arguments>::new_const(lit_pieces)
490+
// const { <core::fmt::Arguments>::new_const(lit_pieces) }
468491
let new = ctx.arena.alloc(ctx.expr_lang_item_type_relative(
469492
macsp,
470493
hir::LangItem::FormatArguments,
471494
sym::new_const,
472495
));
473496
let new_args = ctx.arena.alloc_from_iter([lit_pieces]);
474-
return hir::ExprKind::Call(new, new_args);
497+
return ctx.wrap_in_const_context(macsp, hir::ExprKind::Call(new, new_args));
475498
}
476499

477500
// If the args array contains exactly all the original arguments once,

library/core/src/fmt/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -333,7 +333,7 @@ pub struct Arguments<'a> {
333333
#[unstable(feature = "fmt_internals", issue = "none")]
334334
impl<'a> Arguments<'a> {
335335
#[inline]
336-
#[rustc_const_unstable(feature = "const_fmt_arguments_new", issue = "none")]
336+
#[rustc_const_stable(feature = "const_fmt_arguments_new", since = "CURRENT_RUSTC_VERSION")]
337337
pub const fn new_const<const N: usize>(pieces: &'a [&'static str; N]) -> Self {
338338
const { assert!(N <= 1) };
339339
Arguments { pieces, fmt: None, args: &[] }

library/core/src/lib.rs

-1
Original file line numberDiff line numberDiff line change
@@ -120,7 +120,6 @@
120120
#![feature(const_char_encode_utf16)]
121121
#![feature(const_eval_select)]
122122
#![feature(const_exact_div)]
123-
#![feature(const_fmt_arguments_new)]
124123
#![feature(const_hash)]
125124
#![feature(const_heap)]
126125
#![feature(const_index_range_slice_index)]

library/core/src/macros/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1038,7 +1038,7 @@ pub(crate) mod builtin {
10381038
///
10391039
/// This macro will be removed once `format_args` is allowed in const contexts.
10401040
#[unstable(feature = "const_format_args", issue = "none")]
1041-
#[allow_internal_unstable(fmt_internals, const_fmt_arguments_new)]
1041+
#[allow_internal_unstable(fmt_internals)]
10421042
#[rustc_builtin_macro]
10431043
#[macro_export]
10441044
macro_rules! const_format_args {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
//@ compile-flags: -C no-prepopulate-passes -C opt-level=0
2+
3+
#![crate_type = "lib"]
4+
5+
// String formating macros without any arguments should compile
6+
// to a `memcpy` followed by a call to a library function.
7+
8+
#[no_mangle]
9+
pub fn code() {
10+
// CHECK-LABEL: @code
11+
// CHECK-NOT: getelementptr
12+
// CHECK-NOT: store
13+
// CHECK-NOT: ; call core::fmt::Arguments::new_const
14+
// CHECK: call void @llvm.memcpy
15+
// CHECK-NEXT: ; call std::io::stdio::_print
16+
println!("hello world");
17+
}

tests/coverage/assert.cov-map

+5-8
Original file line numberDiff line numberDiff line change
@@ -28,15 +28,12 @@ Number of file 0 mappings: 9
2828
Highest counter ID seen: c4
2929

3030
Function name: assert::might_fail_assert
31-
Raw bytes (21): 0x[01, 01, 01, 01, 05, 03, 01, 04, 01, 02, 0f, 02, 02, 25, 00, 3d, 05, 01, 01, 00, 02]
31+
Raw bytes (14): 0x[01, 01, 00, 02, 01, 04, 01, 02, 3e, 05, 03, 01, 00, 02]
3232
Number of files: 1
3333
- file 0 => global file 1
34-
Number of expressions: 1
35-
- expression 0 operands: lhs = Counter(0), rhs = Counter(1)
36-
Number of file 0 mappings: 3
37-
- Code(Counter(0)) at (prev + 4, 1) to (start + 2, 15)
38-
- Code(Expression(0, Sub)) at (prev + 2, 37) to (start + 0, 61)
39-
= (c0 - c1)
40-
- Code(Counter(1)) at (prev + 1, 1) to (start + 0, 2)
34+
Number of expressions: 0
35+
Number of file 0 mappings: 2
36+
- Code(Counter(0)) at (prev + 4, 1) to (start + 2, 62)
37+
- Code(Counter(1)) at (prev + 3, 1) to (start + 0, 2)
4138
Highest counter ID seen: c1
4239

tests/coverage/bad_counter_ids.cov-map

+18-24
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,13 @@ Number of file 0 mappings: 2
99
Highest counter ID seen: c0
1010

1111
Function name: bad_counter_ids::eq_bad_message
12-
Raw bytes (21): 0x[01, 01, 01, 01, 00, 03, 01, 29, 01, 02, 0f, 02, 02, 20, 00, 2b, 00, 01, 01, 00, 02]
12+
Raw bytes (14): 0x[01, 01, 00, 02, 01, 29, 01, 02, 2c, 00, 03, 01, 00, 02]
1313
Number of files: 1
1414
- file 0 => global file 1
15-
Number of expressions: 1
16-
- expression 0 operands: lhs = Counter(0), rhs = Zero
17-
Number of file 0 mappings: 3
18-
- Code(Counter(0)) at (prev + 41, 1) to (start + 2, 15)
19-
- Code(Expression(0, Sub)) at (prev + 2, 32) to (start + 0, 43)
20-
= (c0 - Zero)
21-
- Code(Zero) at (prev + 1, 1) to (start + 0, 2)
15+
Number of expressions: 0
16+
Number of file 0 mappings: 2
17+
- Code(Counter(0)) at (prev + 41, 1) to (start + 2, 44)
18+
- Code(Zero) at (prev + 3, 1) to (start + 0, 2)
2219
Highest counter ID seen: c0
2320

2421
Function name: bad_counter_ids::eq_good
@@ -32,14 +29,13 @@ Number of file 0 mappings: 2
3229
Highest counter ID seen: c1
3330

3431
Function name: bad_counter_ids::eq_good_message
35-
Raw bytes (19): 0x[01, 01, 00, 03, 01, 15, 01, 02, 0f, 00, 02, 20, 00, 2b, 05, 01, 01, 00, 02]
32+
Raw bytes (14): 0x[01, 01, 00, 02, 01, 15, 01, 02, 2c, 05, 03, 01, 00, 02]
3633
Number of files: 1
3734
- file 0 => global file 1
3835
Number of expressions: 0
39-
Number of file 0 mappings: 3
40-
- Code(Counter(0)) at (prev + 21, 1) to (start + 2, 15)
41-
- Code(Zero) at (prev + 2, 32) to (start + 0, 43)
42-
- Code(Counter(1)) at (prev + 1, 1) to (start + 0, 2)
36+
Number of file 0 mappings: 2
37+
- Code(Counter(0)) at (prev + 21, 1) to (start + 2, 44)
38+
- Code(Counter(1)) at (prev + 3, 1) to (start + 0, 2)
4339
Highest counter ID seen: c1
4440

4541
Function name: bad_counter_ids::ne_bad
@@ -53,15 +49,14 @@ Number of file 0 mappings: 2
5349
Highest counter ID seen: c0
5450

5551
Function name: bad_counter_ids::ne_bad_message
56-
Raw bytes (19): 0x[01, 01, 00, 03, 01, 33, 01, 02, 0f, 05, 02, 20, 00, 2b, 00, 01, 01, 00, 02]
52+
Raw bytes (14): 0x[01, 01, 00, 02, 01, 33, 01, 02, 2c, 00, 03, 01, 00, 02]
5753
Number of files: 1
5854
- file 0 => global file 1
5955
Number of expressions: 0
60-
Number of file 0 mappings: 3
61-
- Code(Counter(0)) at (prev + 51, 1) to (start + 2, 15)
62-
- Code(Counter(1)) at (prev + 2, 32) to (start + 0, 43)
63-
- Code(Zero) at (prev + 1, 1) to (start + 0, 2)
64-
Highest counter ID seen: c1
56+
Number of file 0 mappings: 2
57+
- Code(Counter(0)) at (prev + 51, 1) to (start + 2, 44)
58+
- Code(Zero) at (prev + 3, 1) to (start + 0, 2)
59+
Highest counter ID seen: c0
6560

6661
Function name: bad_counter_ids::ne_good
6762
Raw bytes (16): 0x[01, 01, 01, 01, 00, 02, 01, 1a, 01, 02, 1f, 02, 03, 01, 00, 02]
@@ -76,15 +71,14 @@ Number of file 0 mappings: 2
7671
Highest counter ID seen: c0
7772

7873
Function name: bad_counter_ids::ne_good_message
79-
Raw bytes (21): 0x[01, 01, 01, 01, 00, 03, 01, 1f, 01, 02, 0f, 00, 02, 20, 00, 2b, 02, 01, 01, 00, 02]
74+
Raw bytes (16): 0x[01, 01, 01, 01, 00, 02, 01, 1f, 01, 02, 2c, 02, 03, 01, 00, 02]
8075
Number of files: 1
8176
- file 0 => global file 1
8277
Number of expressions: 1
8378
- expression 0 operands: lhs = Counter(0), rhs = Zero
84-
Number of file 0 mappings: 3
85-
- Code(Counter(0)) at (prev + 31, 1) to (start + 2, 15)
86-
- Code(Zero) at (prev + 2, 32) to (start + 0, 43)
87-
- Code(Expression(0, Sub)) at (prev + 1, 1) to (start + 0, 2)
79+
Number of file 0 mappings: 2
80+
- Code(Counter(0)) at (prev + 31, 1) to (start + 2, 44)
81+
- Code(Expression(0, Sub)) at (prev + 3, 1) to (start + 0, 2)
8882
= (c0 - Zero)
8983
Highest counter ID seen: c0
9084

tests/coverage/closure_macro.cov-map

+4-5
Original file line numberDiff line numberDiff line change
@@ -25,20 +25,19 @@ Number of file 0 mappings: 6
2525
Highest counter ID seen: c1
2626

2727
Function name: closure_macro::main::{closure#0}
28-
Raw bytes (35): 0x[01, 01, 03, 01, 05, 05, 0b, 09, 0d, 05, 01, 10, 1c, 03, 21, 05, 04, 11, 01, 27, 02, 03, 11, 00, 16, 0d, 00, 17, 00, 1e, 07, 02, 09, 00, 0a]
28+
Raw bytes (30): 0x[01, 01, 03, 01, 05, 05, 0b, 09, 0d, 04, 01, 10, 1c, 03, 21, 05, 04, 11, 01, 27, 02, 03, 11, 00, 1f, 07, 02, 09, 00, 0a]
2929
Number of files: 1
3030
- file 0 => global file 1
3131
Number of expressions: 3
3232
- expression 0 operands: lhs = Counter(0), rhs = Counter(1)
3333
- expression 1 operands: lhs = Counter(1), rhs = Expression(2, Add)
3434
- expression 2 operands: lhs = Counter(2), rhs = Counter(3)
35-
Number of file 0 mappings: 5
35+
Number of file 0 mappings: 4
3636
- Code(Counter(0)) at (prev + 16, 28) to (start + 3, 33)
3737
- Code(Counter(1)) at (prev + 4, 17) to (start + 1, 39)
38-
- Code(Expression(0, Sub)) at (prev + 3, 17) to (start + 0, 22)
38+
- Code(Expression(0, Sub)) at (prev + 3, 17) to (start + 0, 31)
3939
= (c0 - c1)
40-
- Code(Counter(3)) at (prev + 0, 23) to (start + 0, 30)
4140
- Code(Expression(1, Add)) at (prev + 2, 9) to (start + 0, 10)
4241
= (c1 + (c2 + c3))
43-
Highest counter ID seen: c3
42+
Highest counter ID seen: c1
4443

tests/coverage/closure_macro_async.cov-map

+4-5
Original file line numberDiff line numberDiff line change
@@ -34,20 +34,19 @@ Number of file 0 mappings: 6
3434
Highest counter ID seen: c1
3535

3636
Function name: closure_macro_async::test::{closure#0}::{closure#0}
37-
Raw bytes (35): 0x[01, 01, 03, 01, 05, 05, 0b, 09, 0d, 05, 01, 15, 1c, 03, 21, 05, 04, 11, 01, 27, 02, 03, 11, 00, 16, 0d, 00, 17, 00, 1e, 07, 02, 09, 00, 0a]
37+
Raw bytes (30): 0x[01, 01, 03, 01, 05, 05, 0b, 09, 0d, 04, 01, 15, 1c, 03, 21, 05, 04, 11, 01, 27, 02, 03, 11, 00, 1f, 07, 02, 09, 00, 0a]
3838
Number of files: 1
3939
- file 0 => global file 1
4040
Number of expressions: 3
4141
- expression 0 operands: lhs = Counter(0), rhs = Counter(1)
4242
- expression 1 operands: lhs = Counter(1), rhs = Expression(2, Add)
4343
- expression 2 operands: lhs = Counter(2), rhs = Counter(3)
44-
Number of file 0 mappings: 5
44+
Number of file 0 mappings: 4
4545
- Code(Counter(0)) at (prev + 21, 28) to (start + 3, 33)
4646
- Code(Counter(1)) at (prev + 4, 17) to (start + 1, 39)
47-
- Code(Expression(0, Sub)) at (prev + 3, 17) to (start + 0, 22)
47+
- Code(Expression(0, Sub)) at (prev + 3, 17) to (start + 0, 31)
4848
= (c0 - c1)
49-
- Code(Counter(3)) at (prev + 0, 23) to (start + 0, 30)
5049
- Code(Expression(1, Add)) at (prev + 2, 9) to (start + 0, 10)
5150
= (c1 + (c2 + c3))
52-
Highest counter ID seen: c3
51+
Highest counter ID seen: c1
5352

0 commit comments

Comments
 (0)