diff options
| author | Erik Desjardins <erikdesjardins@users.noreply.github.com> | 2022-02-17 00:58:13 -0500 |
|---|---|---|
| committer | Erik Desjardins <erikdesjardins@users.noreply.github.com> | 2022-02-18 14:36:12 -0500 |
| commit | dcbdc8c19b4b1e581b8b83a513b11c4c4248d0fc (patch) | |
| tree | be9fb4a21141869d43e66686586e6d22d36dd19f /compiler/rustc_codegen_llvm | |
| parent | 73a7423e77b49a99e270531fbadda5b8899df3f6 (diff) | |
| download | rust-dcbdc8c19b4b1e581b8b83a513b11c4c4248d0fc.tar.gz rust-dcbdc8c19b4b1e581b8b83a513b11c4c4248d0fc.zip | |
At opt-level=0, apply only ABI-affecting attributes to functions
This should provide a small perf improvement for debug builds, and should more than cancel out the regression from adding noundef, which was only significant in debug builds.
Diffstat (limited to 'compiler/rustc_codegen_llvm')
| -rw-r--r-- | compiler/rustc_codegen_llvm/src/abi.rs | 93 |
1 files changed, 52 insertions, 41 deletions
diff --git a/compiler/rustc_codegen_llvm/src/abi.rs b/compiler/rustc_codegen_llvm/src/abi.rs index 8a11e3e71bc..a20801914ad 100644 --- a/compiler/rustc_codegen_llvm/src/abi.rs +++ b/compiler/rustc_codegen_llvm/src/abi.rs @@ -13,6 +13,7 @@ use rustc_middle::bug; use rustc_middle::ty::layout::LayoutOf; pub use rustc_middle::ty::layout::{FAT_PTR_ADDR, FAT_PTR_EXTRA}; use rustc_middle::ty::Ty; +use rustc_session::config; use rustc_target::abi::call::ArgAbi; pub use rustc_target::abi::call::*; use rustc_target::abi::{self, HasDataLayout, Int}; @@ -20,27 +21,6 @@ pub use rustc_target::spec::abi::Abi; use libc::c_uint; -macro_rules! for_each_kind { - ($flags: ident, $f: ident, $($kind: ident),+) => ({ - $(if $flags.contains(ArgAttribute::$kind) { $f(llvm::Attribute::$kind) })+ - }) -} - -trait ArgAttributeExt { - fn for_each_kind<F>(&self, f: F) - where - F: FnMut(llvm::Attribute); -} - -impl ArgAttributeExt for ArgAttribute { - fn for_each_kind<F>(&self, mut f: F) - where - F: FnMut(llvm::Attribute), - { - for_each_kind!(self, f, NoAlias, NoCapture, NonNull, ReadOnly, InReg, NoUndef) - } -} - pub trait ArgAttributesExt { fn apply_attrs_to_llfn(&self, idx: AttributePlace, cx: &CodegenCx<'_, '_>, llfn: &Value); fn apply_attrs_to_callsite( @@ -58,10 +38,36 @@ fn should_use_mutable_noalias(cx: &CodegenCx<'_, '_>) -> bool { cx.tcx.sess.opts.debugging_opts.mutable_noalias.unwrap_or(true) } +const ABI_AFFECTING_ATTRIBUTES: [(ArgAttribute, llvm::Attribute); 1] = + [(ArgAttribute::InReg, llvm::Attribute::InReg)]; + +const OPTIMIZATION_ATTRIBUTES: [(ArgAttribute, llvm::Attribute); 5] = [ + (ArgAttribute::NoAlias, llvm::Attribute::NoAlias), + (ArgAttribute::NoCapture, llvm::Attribute::NoCapture), + (ArgAttribute::NonNull, llvm::Attribute::NonNull), + (ArgAttribute::ReadOnly, llvm::Attribute::ReadOnly), + (ArgAttribute::NoUndef, llvm::Attribute::NoUndef), +]; + impl ArgAttributesExt for ArgAttributes { fn apply_attrs_to_llfn(&self, idx: AttributePlace, cx: &CodegenCx<'_, '_>, llfn: &Value) { let mut regular = self.regular; unsafe { + // ABI-affecting attributes must always be applied + for (attr, llattr) in ABI_AFFECTING_ATTRIBUTES { + if regular.contains(attr) { + llattr.apply_llfn(idx, llfn); + } + } + match self.arg_ext { + ArgExtension::None => {} + ArgExtension::Zext => llvm::Attribute::ZExt.apply_llfn(idx, llfn), + ArgExtension::Sext => llvm::Attribute::SExt.apply_llfn(idx, llfn), + } + // Only apply remaining attributes when optimizing + if cx.sess().opts.optimize == config::OptLevel::No { + return; + } let deref = self.pointee_size.bytes(); if deref != 0 { if regular.contains(ArgAttribute::NonNull) { @@ -74,19 +80,14 @@ impl ArgAttributesExt for ArgAttributes { if let Some(align) = self.pointee_align { llvm::LLVMRustAddAlignmentAttr(llfn, idx.as_uint(), align.bytes() as u32); } - regular.for_each_kind(|attr| attr.apply_llfn(idx, llfn)); + for (attr, llattr) in OPTIMIZATION_ATTRIBUTES { + if regular.contains(attr) { + llattr.apply_llfn(idx, llfn); + } + } if regular.contains(ArgAttribute::NoAliasMutRef) && should_use_mutable_noalias(cx) { llvm::Attribute::NoAlias.apply_llfn(idx, llfn); } - match self.arg_ext { - ArgExtension::None => {} - ArgExtension::Zext => { - llvm::Attribute::ZExt.apply_llfn(idx, llfn); - } - ArgExtension::Sext => { - llvm::Attribute::SExt.apply_llfn(idx, llfn); - } - } } } @@ -98,6 +99,21 @@ impl ArgAttributesExt for ArgAttributes { ) { let mut regular = self.regular; unsafe { + // ABI-affecting attributes must always be applied + for (attr, llattr) in ABI_AFFECTING_ATTRIBUTES { + if regular.contains(attr) { + llattr.apply_callsite(idx, callsite); + } + } + match self.arg_ext { + ArgExtension::None => {} + ArgExtension::Zext => llvm::Attribute::ZExt.apply_callsite(idx, callsite), + ArgExtension::Sext => llvm::Attribute::SExt.apply_callsite(idx, callsite), + } + // Only apply remaining attributes when optimizing + if cx.sess().opts.optimize == config::OptLevel::No { + return; + } let deref = self.pointee_size.bytes(); if deref != 0 { if regular.contains(ArgAttribute::NonNull) { @@ -118,19 +134,14 @@ impl ArgAttributesExt for ArgAttributes { align.bytes() as u32, ); } - regular.for_each_kind(|attr| attr.apply_callsite(idx, callsite)); + for (attr, llattr) in OPTIMIZATION_ATTRIBUTES { + if regular.contains(attr) { + llattr.apply_callsite(idx, callsite); + } + } if regular.contains(ArgAttribute::NoAliasMutRef) && should_use_mutable_noalias(cx) { llvm::Attribute::NoAlias.apply_callsite(idx, callsite); } - match self.arg_ext { - ArgExtension::None => {} - ArgExtension::Zext => { - llvm::Attribute::ZExt.apply_callsite(idx, callsite); - } - ArgExtension::Sext => { - llvm::Attribute::SExt.apply_callsite(idx, callsite); - } - } } } } |
