Skip to content

Commit 38030ff

Browse files
committed
Auto merge of rust-lang#78920 - jonas-schievink:rollup-w2mjsuh, r=jonas-schievink
Rollup of 14 pull requests Successful merges: - rust-lang#76765 (Make it more clear what an about async fn's returns when referring to what it returns) - rust-lang#78574 (Use check-pass instead of build-pass in regions ui test suite) - rust-lang#78669 (Use check-pass instead of build-pass in some consts ui test suits) - rust-lang#78847 (Assert that a return place is not used for indexing during integration) - rust-lang#78854 (Workaround for "could not fully normalize" ICE ) - rust-lang#78875 (rustc_target: Further cleanup use of target options) - rust-lang#78887 (Add comments to explain memory usage optimization) - rust-lang#78890 (comment attribution fix) - rust-lang#78896 (Clarified description of write! macro) - rust-lang#78897 (Add missing newline to error message of the default OOM hook) - rust-lang#78898 (add regression test for rust-lang#78892) - rust-lang#78908 ((rustdoc) [src] link for types defined by macros shows invocation, not defintion) - rust-lang#78910 (Fix links to stabilized versions of some intrinsics) - rust-lang#78912 (Add macro test for min-const-generics) Failed merges: r? `@ghost` `@rustbot` modify labels: rollup
2 parents cf9cf7c + fa4d0f2 commit 38030ff

File tree

204 files changed

+1089
-684
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

204 files changed

+1089
-684
lines changed

compiler/rustc_ast_passes/src/ast_validation.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -796,7 +796,7 @@ impl<'a> Visitor<'a> for AstValidator<'a> {
796796

797797
fn visit_expr(&mut self, expr: &'a Expr) {
798798
match &expr.kind {
799-
ExprKind::LlvmInlineAsm(..) if !self.session.target.options.allow_asm => {
799+
ExprKind::LlvmInlineAsm(..) if !self.session.target.allow_asm => {
800800
struct_span_err!(
801801
self.session,
802802
expr.span,

compiler/rustc_builtin_macros/src/test_harness.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ struct TestCtxt<'a> {
3737
pub fn inject(sess: &Session, resolver: &mut dyn ResolverExpand, krate: &mut ast::Crate) {
3838
let span_diagnostic = sess.diagnostic();
3939
let panic_strategy = sess.panic_strategy();
40-
let platform_panic_strategy = sess.target.options.panic_strategy;
40+
let platform_panic_strategy = sess.target.panic_strategy;
4141

4242
// Check for #![reexport_test_harness_main = "some_name"] which gives the
4343
// main test function the name `some_name` without hygiene. This needs to be

compiler/rustc_codegen_cranelift/src/archive.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -63,9 +63,9 @@ impl<'a> ArchiveBuilder<'a> for ArArchiveBuilder<'a> {
6363
sess,
6464
dst: output.to_path_buf(),
6565
lib_search_paths: archive_search_paths(sess),
66-
use_gnu_style_archive: sess.target.options.archive_format == "gnu",
66+
use_gnu_style_archive: sess.target.archive_format == "gnu",
6767
// FIXME fix builtin ranlib on macOS
68-
no_builtin_ranlib: sess.target.options.is_like_osx,
68+
no_builtin_ranlib: sess.target.is_like_osx,
6969

7070
src_archives,
7171
entries,

compiler/rustc_codegen_cranelift/src/debuginfo/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ impl<'tcx> DebugContext<'tcx> {
5050
// TODO: this should be configurable
5151
// macOS doesn't seem to support DWARF > 3
5252
// 5 version is required for md5 file hash
53-
version: if tcx.sess.target.options.is_like_osx {
53+
version: if tcx.sess.target.is_like_osx {
5454
3
5555
} else {
5656
// FIXME change to version 5 once the gdb and lldb shipping with the latest debian

compiler/rustc_codegen_cranelift/src/driver/aot.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -320,8 +320,8 @@ fn codegen_global_asm(tcx: TyCtxt<'_>, cgu_name: &str, global_asm: &str) {
320320
}
321321

322322
if cfg!(not(feature = "inline_asm"))
323-
|| tcx.sess.target.options.is_like_osx
324-
|| tcx.sess.target.options.is_like_windows
323+
|| tcx.sess.target.is_like_osx
324+
|| tcx.sess.target.is_like_windows
325325
{
326326
if global_asm.contains("__rust_probestack") {
327327
return;

compiler/rustc_codegen_cranelift/src/metadata.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -101,7 +101,7 @@ pub(crate) fn write_metadata<P: WriteMetadata>(
101101
product.add_rustc_section(
102102
rustc_middle::middle::exported_symbols::metadata_symbol_name(tcx),
103103
compressed,
104-
tcx.sess.target.options.is_like_osx,
104+
tcx.sess.target.is_like_osx,
105105
);
106106

107107
metadata

compiler/rustc_codegen_cranelift/src/toolchain.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ fn linker_and_flavor(sess: &Session) -> (PathBuf, LinkerFlavor) {
9191
} else if stem == "link" || stem == "lld-link" {
9292
LinkerFlavor::Msvc
9393
} else if stem == "lld" || stem == "rust-lld" {
94-
LinkerFlavor::Lld(sess.target.options.lld_flavor)
94+
LinkerFlavor::Lld(sess.target.lld_flavor)
9595
} else {
9696
// fall back to the value in the target spec
9797
sess.target.linker_flavor
@@ -115,7 +115,7 @@ fn linker_and_flavor(sess: &Session) -> (PathBuf, LinkerFlavor) {
115115

116116
if let Some(ret) = infer_from(
117117
sess,
118-
sess.target.options.linker.clone().map(PathBuf::from),
118+
sess.target.linker.clone().map(PathBuf::from),
119119
Some(sess.target.linker_flavor),
120120
) {
121121
return ret;

compiler/rustc_codegen_llvm/src/allocator.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ pub(crate) unsafe fn codegen(
5757
let name = format!("__rust_{}", method.name);
5858
let llfn = llvm::LLVMRustGetOrInsertFunction(llmod, name.as_ptr().cast(), name.len(), ty);
5959

60-
if tcx.sess.target.options.default_hidden_visibility {
60+
if tcx.sess.target.default_hidden_visibility {
6161
llvm::LLVMRustSetVisibility(llfn, llvm::Visibility::Hidden);
6262
}
6363
if tcx.sess.must_emit_unwind_tables() {
@@ -98,7 +98,7 @@ pub(crate) unsafe fn codegen(
9898
// -> ! DIFlagNoReturn
9999
llvm::Attribute::NoReturn.apply_llfn(llvm::AttributePlace::Function, llfn);
100100

101-
if tcx.sess.target.options.default_hidden_visibility {
101+
if tcx.sess.target.default_hidden_visibility {
102102
llvm::LLVMRustSetVisibility(llfn, llvm::Visibility::Hidden);
103103
}
104104
if tcx.sess.must_emit_unwind_tables() {

compiler/rustc_codegen_llvm/src/attributes.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -90,8 +90,7 @@ fn set_instrument_function(cx: &CodegenCx<'ll, '_>, llfn: &'ll Value) {
9090

9191
// The function name varies on platforms.
9292
// See test/CodeGen/mcount.c in clang.
93-
let mcount_name =
94-
CString::new(cx.sess().target.options.target_mcount.as_str().as_bytes()).unwrap();
93+
let mcount_name = CString::new(cx.sess().target.mcount.as_str().as_bytes()).unwrap();
9594

9695
llvm::AddFunctionAttrStringValue(
9796
llfn,
@@ -105,7 +104,7 @@ fn set_instrument_function(cx: &CodegenCx<'ll, '_>, llfn: &'ll Value) {
105104
fn set_probestack(cx: &CodegenCx<'ll, '_>, llfn: &'ll Value) {
106105
// Only use stack probes if the target specification indicates that we
107106
// should be using stack probes
108-
if !cx.sess().target.options.stack_probes {
107+
if !cx.sess().target.stack_probes {
109108
return;
110109
}
111110

@@ -174,7 +173,6 @@ pub fn llvm_target_features(sess: &Session) -> impl Iterator<Item = &str> {
174173
.split(',')
175174
.filter(|f| !RUSTC_SPECIFIC_FEATURES.iter().any(|s| f.contains(s)));
176175
sess.target
177-
.options
178176
.features
179177
.split(',')
180178
.chain(cmdline)

compiler/rustc_codegen_llvm/src/back/archive.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -206,7 +206,7 @@ impl<'a> LlvmArchiveBuilder<'a> {
206206
}
207207

208208
fn llvm_archive_kind(&self) -> Result<ArchiveKind, &str> {
209-
let kind = &*self.config.sess.target.options.archive_format;
209+
let kind = &*self.config.sess.target.archive_format;
210210
kind.parse().map_err(|_| kind)
211211
}
212212

compiler/rustc_codegen_llvm/src/back/write.rs

+9-15
Original file line numberDiff line numberDiff line change
@@ -129,13 +129,13 @@ pub fn target_machine_factory(
129129
let use_softfp = sess.opts.cg.soft_float;
130130

131131
let ffunction_sections =
132-
sess.opts.debugging_opts.function_sections.unwrap_or(sess.target.options.function_sections);
132+
sess.opts.debugging_opts.function_sections.unwrap_or(sess.target.function_sections);
133133
let fdata_sections = ffunction_sections;
134134

135135
let code_model = to_llvm_code_model(sess.code_model());
136136

137137
let features = attributes::llvm_target_features(sess).collect::<Vec<_>>();
138-
let mut singlethread = sess.target.options.singlethread;
138+
let mut singlethread = sess.target.singlethread;
139139

140140
// On the wasm target once the `atomics` feature is enabled that means that
141141
// we're no longer single-threaded, or otherwise we don't want LLVM to
@@ -151,22 +151,16 @@ pub fn target_machine_factory(
151151
let cpu = SmallCStr::new(llvm_util::target_cpu(sess));
152152
let features = features.join(",");
153153
let features = CString::new(features).unwrap();
154-
let abi = SmallCStr::new(&sess.target.options.llvm_abiname);
155-
let trap_unreachable = sess.target.options.trap_unreachable;
154+
let abi = SmallCStr::new(&sess.target.llvm_abiname);
155+
let trap_unreachable = sess.target.trap_unreachable;
156156
let emit_stack_size_section = sess.opts.debugging_opts.emit_stack_sizes;
157157

158158
let asm_comments = sess.asm_comments();
159-
let relax_elf_relocations = sess
160-
.opts
161-
.debugging_opts
162-
.relax_elf_relocations
163-
.unwrap_or(sess.target.options.relax_elf_relocations);
164-
165-
let use_init_array = !sess
166-
.opts
167-
.debugging_opts
168-
.use_ctors_section
169-
.unwrap_or(sess.target.options.use_ctors_section);
159+
let relax_elf_relocations =
160+
sess.opts.debugging_opts.relax_elf_relocations.unwrap_or(sess.target.relax_elf_relocations);
161+
162+
let use_init_array =
163+
!sess.opts.debugging_opts.use_ctors_section.unwrap_or(sess.target.use_ctors_section);
170164

171165
Arc::new(move || {
172166
let tm = unsafe {

compiler/rustc_codegen_llvm/src/callee.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -176,7 +176,7 @@ pub fn get_fn(cx: &CodegenCx<'ll, 'tcx>, instance: Instance<'tcx>) -> &'ll Value
176176
// should use dllimport for functions.
177177
if cx.use_dll_storage_attrs
178178
&& tcx.is_dllimport_foreign_item(instance_def_id)
179-
&& tcx.sess.target.target_env != "gnu"
179+
&& tcx.sess.target.env != "gnu"
180180
{
181181
unsafe {
182182
llvm::LLVMSetDLLStorageClass(llfn, llvm::DLLStorageClass::DllImport);

compiler/rustc_codegen_llvm/src/consts.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ fn set_global_alignment(cx: &CodegenCx<'ll, '_>, gv: &'ll Value, mut align: Alig
9191
// The target may require greater alignment for globals than the type does.
9292
// Note: GCC and Clang also allow `__attribute__((aligned))` on variables,
9393
// which can force it to be smaller. Rust doesn't support this yet.
94-
if let Some(min) = cx.sess().target.options.min_global_align {
94+
if let Some(min) = cx.sess().target.min_global_align {
9595
match Align::from_bits(min) {
9696
Ok(min) => align = align.max(min),
9797
Err(err) => {
@@ -283,7 +283,7 @@ impl CodegenCx<'ll, 'tcx> {
283283
// argument validation.
284284
debug_assert!(
285285
!(self.tcx.sess.opts.cg.linker_plugin_lto.enabled()
286-
&& self.tcx.sess.target.options.is_like_windows
286+
&& self.tcx.sess.target.is_like_windows
287287
&& self.tcx.sess.opts.cg.prefer_dynamic)
288288
);
289289

@@ -435,7 +435,7 @@ impl StaticMethods for CodegenCx<'ll, 'tcx> {
435435
// will use load-unaligned instructions instead, and thus avoiding the crash.
436436
//
437437
// We could remove this hack whenever we decide to drop macOS 10.10 support.
438-
if self.tcx.sess.target.options.is_like_osx {
438+
if self.tcx.sess.target.is_like_osx {
439439
// The `inspect` method is okay here because we checked relocations, and
440440
// because we are doing this access to inspect the final interpreter state
441441
// (not as part of the interpreter execution).

compiler/rustc_codegen_llvm/src/context.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -129,7 +129,7 @@ pub unsafe fn create_module(
129129
}
130130

131131
// Ensure the data-layout values hardcoded remain the defaults.
132-
if sess.target.options.is_builtin {
132+
if sess.target.is_builtin {
133133
let tm = crate::back::write::create_informational_target_machine(tcx.sess);
134134
llvm::LLVMRustSetDataLayoutFromTargetMachine(llmod, tm);
135135
llvm::LLVMRustDisposeTargetMachine(tm);
@@ -190,7 +190,7 @@ pub unsafe fn create_module(
190190
}
191191

192192
// Control Flow Guard is currently only supported by the MSVC linker on Windows.
193-
if sess.target.options.is_like_msvc {
193+
if sess.target.is_like_msvc {
194194
match sess.opts.cg.control_flow_guard {
195195
CFGuard::Disabled => {}
196196
CFGuard::NoChecks => {
@@ -265,7 +265,7 @@ impl<'ll, 'tcx> CodegenCx<'ll, 'tcx> {
265265
// linker will take care of everything. Fixing this problem will likely
266266
// require adding a few attributes to Rust itself (feature gated at the
267267
// start) and then strongly recommending static linkage on Windows!
268-
let use_dll_storage_attrs = tcx.sess.target.options.is_like_windows;
268+
let use_dll_storage_attrs = tcx.sess.target.is_like_windows;
269269

270270
let check_overflow = tcx.sess.overflow_checks();
271271

@@ -839,7 +839,7 @@ impl CodegenCx<'b, 'tcx> {
839839
return eh_catch_typeinfo;
840840
}
841841
let tcx = self.tcx;
842-
assert!(self.sess().target.options.is_like_emscripten);
842+
assert!(self.sess().target.is_like_emscripten);
843843
let eh_catch_typeinfo = match tcx.lang_items().eh_catch_typeinfo() {
844844
Some(def_id) => self.get_static(def_id),
845845
_ => {

compiler/rustc_codegen_llvm/src/debuginfo/gdb.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -67,5 +67,5 @@ pub fn needs_gdb_debug_scripts_section(cx: &CodegenCx<'_, '_>) -> bool {
6767

6868
!omit_gdb_pretty_printer_section
6969
&& cx.sess().opts.debuginfo != DebugInfo::None
70-
&& cx.sess().target.options.emit_debug_gdb_scripts
70+
&& cx.sess().target.emit_debug_gdb_scripts
7171
}

compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -870,7 +870,7 @@ fn basic_type_metadata(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll DIType {
870870

871871
// When targeting MSVC, emit MSVC style type names for compatibility with
872872
// .natvis visualizers (and perhaps other existing native debuggers?)
873-
let msvc_like_names = cx.tcx.sess.target.options.is_like_msvc;
873+
let msvc_like_names = cx.tcx.sess.target.is_like_msvc;
874874

875875
let (name, encoding) = match t.kind() {
876876
ty::Never => ("!", DW_ATE_unsigned),
@@ -981,7 +981,7 @@ pub fn compile_unit_metadata(
981981
// if multiple object files with the same `DW_AT_name` are linked together.
982982
// As a workaround we generate unique names for each object file. Those do
983983
// not correspond to an actual source file but that should be harmless.
984-
if tcx.sess.target.options.is_like_osx {
984+
if tcx.sess.target.is_like_osx {
985985
name_in_debuginfo.push("@");
986986
name_in_debuginfo.push(codegen_unit_name);
987987
}
@@ -1397,7 +1397,7 @@ fn prepare_union_metadata(
13971397
/// on MSVC we have to use the fallback mode, because LLVM doesn't
13981398
/// lower variant parts to PDB.
13991399
fn use_enum_fallback(cx: &CodegenCx<'_, '_>) -> bool {
1400-
cx.sess().target.options.is_like_msvc
1400+
cx.sess().target.is_like_msvc
14011401
}
14021402

14031403
// FIXME(eddyb) maybe precompute this? Right now it's computed once

compiler/rustc_codegen_llvm/src/debuginfo/mod.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -122,12 +122,12 @@ pub fn finalize(cx: &CodegenCx<'_, '_>) {
122122
// for macOS to understand. For more info see #11352
123123
// This can be overridden using --llvm-opts -dwarf-version,N.
124124
// Android has the same issue (#22398)
125-
if let Some(version) = cx.sess().target.options.dwarf_version {
125+
if let Some(version) = cx.sess().target.dwarf_version {
126126
llvm::LLVMRustAddModuleFlag(cx.llmod, "Dwarf Version\0".as_ptr().cast(), version)
127127
}
128128

129129
// Indicate that we want CodeView debug information on MSVC
130-
if cx.sess().target.options.is_like_msvc {
130+
if cx.sess().target.is_like_msvc {
131131
llvm::LLVMRustAddModuleFlag(cx.llmod, "CodeView\0".as_ptr().cast(), 1)
132132
}
133133

@@ -251,7 +251,7 @@ impl CodegenCx<'ll, '_> {
251251
// For MSVC, omit the column number.
252252
// Otherwise, emit it. This mimics clang behaviour.
253253
// See discussion in https://github.com/rust-lang/rust/issues/42921
254-
if self.sess().target.options.is_like_msvc {
254+
if self.sess().target.is_like_msvc {
255255
DebugLoc { file, line, col: None }
256256
} else {
257257
DebugLoc { file, line, col }
@@ -387,7 +387,7 @@ impl DebugInfoMethods<'tcx> for CodegenCx<'ll, 'tcx> {
387387
});
388388

389389
// Arguments types
390-
if cx.sess().target.options.is_like_msvc {
390+
if cx.sess().target.is_like_msvc {
391391
// FIXME(#42800):
392392
// There is a bug in MSDIA that leads to a crash when it encounters
393393
// a fixed-size array of `u8` or something zero-sized in a

compiler/rustc_codegen_llvm/src/declare.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ fn declare_raw_fn(
4242
// be merged.
4343
llvm::SetUnnamedAddress(llfn, llvm::UnnamedAddr::Global);
4444

45-
if cx.tcx.sess.opts.cg.no_redzone.unwrap_or(cx.tcx.sess.target.options.disable_redzone) {
45+
if cx.tcx.sess.opts.cg.no_redzone.unwrap_or(cx.tcx.sess.target.disable_redzone) {
4646
llvm::Attribute::NoRedZone.apply_llfn(Function, llfn);
4747
}
4848

compiler/rustc_codegen_llvm/src/intrinsic.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -367,7 +367,7 @@ fn try_intrinsic(
367367
bx.store(bx.const_i32(0), dest, ret_align);
368368
} else if wants_msvc_seh(bx.sess()) {
369369
codegen_msvc_try(bx, try_func, data, catch_func, dest);
370-
} else if bx.sess().target.options.is_like_emscripten {
370+
} else if bx.sess().target.is_like_emscripten {
371371
codegen_emcc_try(bx, try_func, data, catch_func, dest);
372372
} else {
373373
codegen_gnu_try(bx, try_func, data, catch_func, dest);

compiler/rustc_codegen_llvm/src/llvm_util.rs

+5-10
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ fn require_inited() {
4646
}
4747

4848
unsafe fn configure_llvm(sess: &Session) {
49-
let n_args = sess.opts.cg.llvm_args.len() + sess.target.options.llvm_args.len();
49+
let n_args = sess.opts.cg.llvm_args.len() + sess.target.llvm_args.len();
5050
let mut llvm_c_strs = Vec::with_capacity(n_args + 1);
5151
let mut llvm_args = Vec::with_capacity(n_args + 1);
5252

@@ -57,7 +57,7 @@ unsafe fn configure_llvm(sess: &Session) {
5757
}
5858

5959
let cg_opts = sess.opts.cg.llvm_args.iter();
60-
let tg_opts = sess.target.options.llvm_args.iter();
60+
let tg_opts = sess.target.llvm_args.iter();
6161
let sess_args = cg_opts.chain(tg_opts);
6262

6363
let user_specified_args: FxHashSet<_> =
@@ -84,19 +84,14 @@ unsafe fn configure_llvm(sess: &Session) {
8484
if !sess.opts.debugging_opts.no_generate_arange_section {
8585
add("-generate-arange-section", false);
8686
}
87-
match sess
88-
.opts
89-
.debugging_opts
90-
.merge_functions
91-
.unwrap_or(sess.target.options.merge_functions)
92-
{
87+
match sess.opts.debugging_opts.merge_functions.unwrap_or(sess.target.merge_functions) {
9388
MergeFunctions::Disabled | MergeFunctions::Trampolines => {}
9489
MergeFunctions::Aliases => {
9590
add("-mergefunc-use-aliases", false);
9691
}
9792
}
9893

99-
if sess.target.target_os == "emscripten" && sess.panic_strategy() == PanicStrategy::Unwind {
94+
if sess.target.os == "emscripten" && sess.panic_strategy() == PanicStrategy::Unwind {
10095
add("-enable-emscripten-cxx-exceptions", false);
10196
}
10297

@@ -215,7 +210,7 @@ fn handle_native(name: &str) -> &str {
215210
pub fn target_cpu(sess: &Session) -> &str {
216211
let name = match sess.opts.cg.target_cpu {
217212
Some(ref s) => &**s,
218-
None => &*sess.target.options.cpu,
213+
None => &*sess.target.cpu,
219214
};
220215

221216
handle_native(name)

compiler/rustc_codegen_llvm/src/metadata.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ pub fn metadata_section_name(target: &Target) -> &'static str {
104104
// As a result, we choose a slightly shorter name! As to why
105105
// `.note.rustc` works on MinGW, that's another good question...
106106

107-
if target.options.is_like_osx { "__DATA,.rustc" } else { ".rustc" }
107+
if target.is_like_osx { "__DATA,.rustc" } else { ".rustc" }
108108
}
109109

110110
fn read_metadata_section_name(_target: &Target) -> &'static str {

0 commit comments

Comments
 (0)