Auto merge of #83302 - camsteffen:write-piece-unchecked, r=dtolnay
Get piece unchecked in `write` We already use specialized `zip`, but it seems like we can do a little better by not checking `pieces` length at all. `Arguments` constructors are now unsafe. So the `format_args!` expansion now includes an `unsafe` block. <details> <summary>Local Bench Diff</summary> ```text name before ns/iter after ns/iter diff ns/iter diff % speedup fmt::write_str_macro1 22,967 19,718 -3,249 -14.15% x 1.16 fmt::write_str_macro2 35,527 32,654 -2,873 -8.09% x 1.09 fmt::write_str_macro_debug 571,953 575,973 4,020 0.70% x 0.99 fmt::write_str_ref 9,579 9,459 -120 -1.25% x 1.01 fmt::write_str_value 9,573 9,572 -1 -0.01% x 1.00 fmt::write_u128_max 176 173 -3 -1.70% x 1.02 fmt::write_u128_min 138 134 -4 -2.90% x 1.03 fmt::write_u64_max 139 136 -3 -2.16% x 1.02 fmt::write_u64_min 129 135 6 4.65% x 0.96 fmt::write_vec_macro1 24,401 22,273 -2,128 -8.72% x 1.10 fmt::write_vec_macro2 37,096 35,602 -1,494 -4.03% x 1.04 fmt::write_vec_macro_debug 588,291 589,575 1,284 0.22% x 1.00 fmt::write_vec_ref 9,568 9,732 164 1.71% x 0.98 fmt::write_vec_value 9,516 9,625 109 1.15% x 0.99 ``` </details>
This commit is contained in:
commit
de42550d0a
14 changed files with 165 additions and 65 deletions
|
@ -3,8 +3,8 @@ use Position::*;
|
|||
|
||||
use rustc_ast as ast;
|
||||
use rustc_ast::ptr::P;
|
||||
use rustc_ast::token;
|
||||
use rustc_ast::tokenstream::TokenStream;
|
||||
use rustc_ast::{token, BlockCheckMode, UnsafeSource};
|
||||
use rustc_data_structures::fx::{FxHashMap, FxHashSet};
|
||||
use rustc_errors::{pluralize, Applicability, DiagnosticBuilder};
|
||||
use rustc_expand::base::{self, *};
|
||||
|
@ -838,12 +838,15 @@ impl<'a, 'b> Context<'a, 'b> {
|
|||
//
|
||||
// But the nested match expression is proved to perform not as well
|
||||
// as series of let's; the first approach does.
|
||||
let pat = self.ecx.pat_tuple(self.macsp, pats);
|
||||
let arm = self.ecx.arm(self.macsp, pat, args_array);
|
||||
let head = self.ecx.expr(self.macsp, ast::ExprKind::Tup(heads));
|
||||
let result = self.ecx.expr_match(self.macsp, head, vec![arm]);
|
||||
let args_match = {
|
||||
let pat = self.ecx.pat_tuple(self.macsp, pats);
|
||||
let arm = self.ecx.arm(self.macsp, pat, args_array);
|
||||
let head = self.ecx.expr(self.macsp, ast::ExprKind::Tup(heads));
|
||||
self.ecx.expr_match(self.macsp, head, vec![arm])
|
||||
};
|
||||
|
||||
let args_slice = self.ecx.expr_addr_of(self.macsp, result);
|
||||
let ident = Ident::from_str_and_span("args", self.macsp);
|
||||
let args_slice = self.ecx.expr_ident(self.macsp, ident);
|
||||
|
||||
// Now create the fmt::Arguments struct with all our locals we created.
|
||||
let (fn_name, fn_args) = if self.all_pieces_simple {
|
||||
|
@ -857,7 +860,20 @@ impl<'a, 'b> Context<'a, 'b> {
|
|||
};
|
||||
|
||||
let path = self.ecx.std_path(&[sym::fmt, sym::Arguments, Symbol::intern(fn_name)]);
|
||||
self.ecx.expr_call_global(self.macsp, path, fn_args)
|
||||
let arguments = self.ecx.expr_call_global(self.macsp, path, fn_args);
|
||||
let body = self.ecx.expr_block(P(ast::Block {
|
||||
stmts: vec![self.ecx.stmt_expr(arguments)],
|
||||
id: ast::DUMMY_NODE_ID,
|
||||
rules: BlockCheckMode::Unsafe(UnsafeSource::CompilerGenerated),
|
||||
span: self.macsp,
|
||||
tokens: None,
|
||||
}));
|
||||
|
||||
let ident = Ident::from_str_and_span("args", self.macsp);
|
||||
let binding_mode = ast::BindingMode::ByRef(ast::Mutability::Not);
|
||||
let pat = self.ecx.pat_ident_binding_mode(self.macsp, ident, binding_mode);
|
||||
let arm = self.ecx.arm(self.macsp, pat, body);
|
||||
self.ecx.expr_match(self.macsp, args_match, vec![arm])
|
||||
}
|
||||
|
||||
fn format_arg(
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue