1use std::borrow::Cow;
4use std::fs::File;
5use std::io::Write;
6use std::path::Path;
7
8use itertools::Itertools;
9use object::write::{self, StandardSegment, Symbol, SymbolSection};
10use object::{
11 Architecture, BinaryFormat, Endianness, FileFlags, Object, ObjectSection, ObjectSymbol,
12 SectionFlags, SectionKind, SymbolFlags, SymbolKind, SymbolScope, elf, pe, xcoff,
13};
14use rustc_abi::Endian;
15use rustc_data_structures::memmap::Mmap;
16use rustc_data_structures::owned_slice::{OwnedSlice, try_slice_owned};
17use rustc_metadata::EncodedMetadata;
18use rustc_metadata::creader::MetadataLoader;
19use rustc_metadata::fs::METADATA_FILENAME;
20use rustc_middle::bug;
21use rustc_session::Session;
22use rustc_span::sym;
23use rustc_target::spec::{Abi, Os, RelocModel, Target, ef_avr_arch};
24use tracing::debug;
25
26use super::apple;
27
28#[derive(Debug)]
39pub(crate) struct DefaultMetadataLoader;
40
41static AIX_METADATA_SYMBOL_NAME: &'static str = "__aix_rust_metadata";
42
43fn load_metadata_with(
44 path: &Path,
45 f: impl for<'a> FnOnce(&'a [u8]) -> Result<&'a [u8], String>,
46) -> Result<OwnedSlice, String> {
47 let file =
48 File::open(path).map_err(|e| format!("failed to open file '{}': {}", path.display(), e))?;
49
50 unsafe { Mmap::map(file) }
51 .map_err(|e| format!("failed to mmap file '{}': {}", path.display(), e))
52 .and_then(|mmap| try_slice_owned(mmap, |mmap| f(mmap)))
53}
54
55impl MetadataLoader for DefaultMetadataLoader {
56 fn get_rlib_metadata(&self, target: &Target, path: &Path) -> Result<OwnedSlice, String> {
57 debug!("getting rlib metadata for {}", path.display());
58 load_metadata_with(path, |data| {
59 let archive = object::read::archive::ArchiveFile::parse(&*data)
60 .map_err(|e| format!("failed to parse rlib '{}': {}", path.display(), e))?;
61
62 for entry_result in archive.members() {
63 let entry = entry_result
64 .map_err(|e| format!("failed to parse rlib '{}': {}", path.display(), e))?;
65 if entry.name() == METADATA_FILENAME.as_bytes() {
66 let data = entry
67 .data(data)
68 .map_err(|e| format!("failed to parse rlib '{}': {}", path.display(), e))?;
69 if target.is_like_aix {
70 return get_metadata_xcoff(path, data);
71 } else {
72 return search_for_section(path, data, ".rmeta");
73 }
74 }
75 }
76
77 Err(format!("metadata not found in rlib '{}'", path.display()))
78 })
79 }
80
81 fn get_dylib_metadata(&self, target: &Target, path: &Path) -> Result<OwnedSlice, String> {
82 debug!("getting dylib metadata for {}", path.display());
83 if target.is_like_aix {
84 load_metadata_with(path, |data| {
85 let archive = object::read::archive::ArchiveFile::parse(&*data).map_err(|e| {
86 format!("failed to parse aix dylib '{}': {}", path.display(), e)
87 })?;
88
89 match archive.members().exactly_one() {
90 Ok(lib) => {
91 let lib = lib.map_err(|e| {
92 format!("failed to parse aix dylib '{}': {}", path.display(), e)
93 })?;
94 let data = lib.data(data).map_err(|e| {
95 format!("failed to parse aix dylib '{}': {}", path.display(), e)
96 })?;
97 get_metadata_xcoff(path, data)
98 }
99 Err(e) => Err(format!("failed to parse aix dylib '{}': {}", path.display(), e)),
100 }
101 })
102 } else {
103 load_metadata_with(path, |data| search_for_section(path, data, ".rustc"))
104 }
105 }
106}
107
108pub(super) fn search_for_section<'a>(
109 path: &Path,
110 bytes: &'a [u8],
111 section: &str,
112) -> Result<&'a [u8], String> {
113 let Ok(file) = object::File::parse(bytes) else {
114 return Ok(bytes);
121 };
122 file.section_by_name(section)
123 .ok_or_else(|| format!("no `{}` section in '{}'", section, path.display()))?
124 .data()
125 .map_err(|e| format!("failed to read {} section in '{}': {}", section, path.display(), e))
126}
127
128fn add_gnu_property_note(
129 file: &mut write::Object<'static>,
130 architecture: Architecture,
131 binary_format: BinaryFormat,
132 endianness: Endianness,
133) {
134 if binary_format != BinaryFormat::Elf
136 || !matches!(architecture, Architecture::X86_64 | Architecture::Aarch64)
137 {
138 return;
139 }
140
141 let section = file.add_section(
142 file.segment_name(StandardSegment::Data).to_vec(),
143 b".note.gnu.property".to_vec(),
144 SectionKind::Note,
145 );
146 let mut data: Vec<u8> = Vec::new();
147 let n_namsz: u32 = 4; let n_descsz: u32 = 16; let n_type: u32 = object::elf::NT_GNU_PROPERTY_TYPE_0; let header_values = [n_namsz, n_descsz, n_type];
151 header_values.iter().for_each(|v| {
152 data.extend_from_slice(&match endianness {
153 Endianness::Little => v.to_le_bytes(),
154 Endianness::Big => v.to_be_bytes(),
155 })
156 });
157 data.extend_from_slice(b"GNU\0"); let pr_type: u32 = match architecture {
159 Architecture::X86_64 => object::elf::GNU_PROPERTY_X86_FEATURE_1_AND,
160 Architecture::Aarch64 => object::elf::GNU_PROPERTY_AARCH64_FEATURE_1_AND,
161 _ => unreachable!(),
162 };
163 let pr_datasz: u32 = 4; let pr_data: u32 = 3; let pr_padding: u32 = 0;
166 let property_values = [pr_type, pr_datasz, pr_data, pr_padding];
167 property_values.iter().for_each(|v| {
168 data.extend_from_slice(&match endianness {
169 Endianness::Little => v.to_le_bytes(),
170 Endianness::Big => v.to_be_bytes(),
171 })
172 });
173 file.append_section_data(section, &data, 8);
174}
175
176pub(super) fn get_metadata_xcoff<'a>(path: &Path, data: &'a [u8]) -> Result<&'a [u8], String> {
177 let Ok(file) = object::File::parse(data) else {
178 return Ok(data);
179 };
180 let info_data = search_for_section(path, data, ".info")?;
181 if let Some(metadata_symbol) =
182 file.symbols().find(|sym| sym.name() == Ok(AIX_METADATA_SYMBOL_NAME))
183 {
184 let offset = metadata_symbol.address() as usize;
185 if offset < 4 {
188 return Err(format!("Invalid metadata symbol offset: {offset}"));
189 }
190 let len = u32::from_be_bytes(info_data[(offset - 4)..offset].try_into().unwrap()) as usize;
192 if offset + len > (info_data.len() as usize) {
193 return Err(format!(
194 "Metadata at offset {offset} with size {len} is beyond .info section"
195 ));
196 }
197 Ok(&info_data[offset..(offset + len)])
198 } else {
199 Err(format!("Unable to find symbol {AIX_METADATA_SYMBOL_NAME}"))
200 }
201}
202
203pub(crate) fn create_object_file(sess: &Session) -> Option<write::Object<'static>> {
204 let endianness = match sess.target.options.endian {
205 Endian::Little => Endianness::Little,
206 Endian::Big => Endianness::Big,
207 };
208 let Some((architecture, sub_architecture)) =
209 sess.target.object_architecture(&sess.unstable_target_features)
210 else {
211 return None;
212 };
213 let binary_format = sess.target.binary_format.to_object();
214
215 let mut file = write::Object::new(binary_format, architecture, endianness);
216 file.set_sub_architecture(sub_architecture);
217 if sess.target.is_like_darwin {
218 if macho_is_arm64e(&sess.target) {
219 file.set_macho_cpu_subtype(object::macho::CPU_SUBTYPE_ARM64E);
220 }
221
222 file.set_macho_build_version(macho_object_build_version_for_target(sess))
223 }
224 if binary_format == BinaryFormat::Coff {
225 let original_mangling = file.mangling();
227 file.set_mangling(object::write::Mangling::None);
228
229 let mut feature = 0;
230
231 if file.architecture() == object::Architecture::I386 {
232 feature |= 1;
238 }
239
240 file.add_symbol(object::write::Symbol {
241 name: "@feat.00".into(),
242 value: feature,
243 size: 0,
244 kind: object::SymbolKind::Data,
245 scope: object::SymbolScope::Compilation,
246 weak: false,
247 section: object::write::SymbolSection::Absolute,
248 flags: object::SymbolFlags::None,
249 });
250
251 file.set_mangling(original_mangling);
252 }
253 let e_flags = elf_e_flags(architecture, sess);
254 let os_abi = elf_os_abi(sess);
256 let abi_version = 0;
257 add_gnu_property_note(&mut file, architecture, binary_format, endianness);
258 file.flags = FileFlags::Elf { os_abi, abi_version, e_flags };
259 Some(file)
260}
261
262pub(super) fn elf_os_abi(sess: &Session) -> u8 {
263 match sess.target.options.os {
264 Os::Hermit => elf::ELFOSABI_STANDALONE,
265 Os::FreeBsd => elf::ELFOSABI_FREEBSD,
266 Os::Solaris => elf::ELFOSABI_SOLARIS,
267 _ => elf::ELFOSABI_NONE,
268 }
269}
270
271pub(super) fn elf_e_flags(architecture: Architecture, sess: &Session) -> u32 {
272 match architecture {
273 Architecture::Mips | Architecture::Mips64 | Architecture::Mips64_N32 => {
274 let is_32bit = architecture == Architecture::Mips;
277 let mut e_flags = match sess.target.options.cpu.as_ref() {
278 "mips1" if is_32bit => elf::EF_MIPS_ARCH_1,
279 "mips2" if is_32bit => elf::EF_MIPS_ARCH_2,
280 "mips3" => elf::EF_MIPS_ARCH_3,
281 "mips4" => elf::EF_MIPS_ARCH_4,
282 "mips5" => elf::EF_MIPS_ARCH_5,
283 "mips32r2" if is_32bit => elf::EF_MIPS_ARCH_32R2,
284 "mips32r6" if is_32bit => elf::EF_MIPS_ARCH_32R6,
285 "mips64r2" if !is_32bit => elf::EF_MIPS_ARCH_64R2,
286 "mips64r6" if !is_32bit => elf::EF_MIPS_ARCH_64R6,
287 s if s.starts_with("mips32") && !is_32bit => {
288 sess.dcx().fatal(format!("invalid CPU `{}` for 64-bit MIPS target", s))
289 }
290 s if s.starts_with("mips64") && is_32bit => {
291 sess.dcx().fatal(format!("invalid CPU `{}` for 32-bit MIPS target", s))
292 }
293 _ if is_32bit => elf::EF_MIPS_ARCH_32R2,
294 _ => elf::EF_MIPS_ARCH_64R2,
295 };
296
297 match sess.target.options.llvm_abiname.as_ref() {
300 "o32" if is_32bit => e_flags |= elf::EF_MIPS_ABI_O32,
301 "n32" if !is_32bit => e_flags |= elf::EF_MIPS_ABI2,
302 "n64" if !is_32bit => {}
303 "" if is_32bit => e_flags |= elf::EF_MIPS_ABI_O32,
304 "" => sess.dcx().fatal("LLVM ABI must be specified for 64-bit MIPS targets"),
305 s if is_32bit => {
306 sess.dcx().fatal(format!("invalid LLVM ABI `{}` for 32-bit MIPS target", s))
307 }
308 s => sess.dcx().fatal(format!("invalid LLVM ABI `{}` for 64-bit MIPS target", s)),
309 };
310
311 if sess.target.options.relocation_model != RelocModel::Static {
312 e_flags |= elf::EF_MIPS_PIC | elf::EF_MIPS_CPIC;
322 }
323 if sess.target.options.cpu.contains("r6") {
324 e_flags |= elf::EF_MIPS_NAN2008;
325 }
326 e_flags
327 }
328 Architecture::Riscv32 | Architecture::Riscv64 => {
329 let mut e_flags: u32 = 0x0;
331
332 if sess.target_features.contains(&sym::zca) {
334 e_flags |= elf::EF_RISCV_RVC;
335 }
336
337 if sess.target_features.contains(&sym::ztso) {
339 e_flags |= elf::EF_RISCV_TSO;
340 }
341
342 match &*sess.target.llvm_abiname {
345 "ilp32" | "lp64" => (),
346 "ilp32f" | "lp64f" => e_flags |= elf::EF_RISCV_FLOAT_ABI_SINGLE,
347 "ilp32d" | "lp64d" => e_flags |= elf::EF_RISCV_FLOAT_ABI_DOUBLE,
348 "ilp32e" | "lp64e" => e_flags |= elf::EF_RISCV_RVE,
350 _ => bug!("unknown RISC-V ABI name"),
351 }
352
353 e_flags
354 }
355 Architecture::LoongArch32 | Architecture::LoongArch64 => {
356 let mut e_flags: u32 = elf::EF_LARCH_OBJABI_V1;
358
359 match &*sess.target.llvm_abiname {
362 "ilp32s" | "lp64s" => e_flags |= elf::EF_LARCH_ABI_SOFT_FLOAT,
363 "ilp32f" | "lp64f" => e_flags |= elf::EF_LARCH_ABI_SINGLE_FLOAT,
364 "ilp32d" | "lp64d" => e_flags |= elf::EF_LARCH_ABI_DOUBLE_FLOAT,
365 _ => bug!("unknown LoongArch ABI name"),
366 }
367
368 e_flags
369 }
370 Architecture::Avr => {
371 if let Some(ref cpu) = sess.opts.cg.target_cpu {
374 ef_avr_arch(cpu)
375 } else {
376 bug!("AVR CPU not explicitly specified")
377 }
378 }
379 Architecture::Csky => {
380 if matches!(sess.target.options.abi, Abi::AbiV2) {
381 elf::EF_CSKY_ABIV2
382 } else {
383 elf::EF_CSKY_ABIV1
384 }
385 }
386 Architecture::PowerPc64 => {
387 const EF_PPC64_ABI_UNKNOWN: u32 = 0;
388 const EF_PPC64_ABI_ELF_V1: u32 = 1;
389 const EF_PPC64_ABI_ELF_V2: u32 = 2;
390
391 match sess.target.options.llvm_abiname.as_ref() {
392 "elfv1" => EF_PPC64_ABI_ELF_V1,
396 "elfv2" => EF_PPC64_ABI_ELF_V2,
397 "" if sess.target.options.binary_format.to_object() == BinaryFormat::Elf => {
398 bug!("No ABI specified for this PPC64 ELF target");
399 }
400 _ => EF_PPC64_ABI_UNKNOWN,
402 }
403 }
404 _ => 0,
405 }
406}
407
408fn macho_object_build_version_for_target(sess: &Session) -> object::write::MachOBuildVersion {
427 fn pack_version(apple::OSVersion { major, minor, patch }: apple::OSVersion) -> u32 {
430 let (major, minor, patch) = (major as u32, minor as u32, patch as u32);
431 (major << 16) | (minor << 8) | patch
432 }
433
434 let platform = apple::macho_platform(&sess.target);
435 let min_os = sess.apple_deployment_target();
436
437 let mut build_version = object::write::MachOBuildVersion::default();
438 build_version.platform = platform;
439 build_version.minos = pack_version(min_os);
440 build_version.sdk = 0;
446
447 build_version
448}
449
450fn macho_is_arm64e(target: &Target) -> bool {
452 target.llvm_target.starts_with("arm64e")
453}
454
455pub(crate) enum MetadataPosition {
456 First,
457 Last,
458}
459
460pub(crate) fn create_wrapper_file(
492 sess: &Session,
493 section_name: String,
494 data: &[u8],
495) -> (Vec<u8>, MetadataPosition) {
496 let Some(mut file) = create_object_file(sess) else {
497 if sess.target.is_like_wasm {
498 return (
499 create_metadata_file_for_wasm(sess, data, §ion_name),
500 MetadataPosition::First,
501 );
502 }
503
504 return (data.to_vec(), MetadataPosition::Last);
508 };
509 let section = if file.format() == BinaryFormat::Xcoff {
510 file.add_section(Vec::new(), b".info".to_vec(), SectionKind::Debug)
511 } else {
512 file.add_section(
513 file.segment_name(StandardSegment::Debug).to_vec(),
514 section_name.into_bytes(),
515 SectionKind::Debug,
516 )
517 };
518 match file.format() {
519 BinaryFormat::Coff => {
520 file.section_mut(section).flags =
521 SectionFlags::Coff { characteristics: pe::IMAGE_SCN_LNK_REMOVE };
522 }
523 BinaryFormat::Elf => {
524 file.section_mut(section).flags =
525 SectionFlags::Elf { sh_flags: elf::SHF_EXCLUDE as u64 };
526 }
527 BinaryFormat::Xcoff => {
528 file.add_section(Vec::new(), b".text".to_vec(), SectionKind::Text);
530 file.section_mut(section).flags =
531 SectionFlags::Xcoff { s_flags: xcoff::STYP_INFO as u32 };
532 let len: u32 = data.len().try_into().unwrap();
537 let offset = file.append_section_data(section, &len.to_be_bytes(), 1);
538 file.add_symbol(Symbol {
540 name: AIX_METADATA_SYMBOL_NAME.into(),
541 value: offset + 4,
542 size: 0,
543 kind: SymbolKind::Unknown,
544 scope: SymbolScope::Compilation,
545 weak: false,
546 section: SymbolSection::Section(section),
547 flags: SymbolFlags::Xcoff {
548 n_sclass: xcoff::C_INFO,
549 x_smtyp: xcoff::C_HIDEXT,
550 x_smclas: xcoff::C_HIDEXT,
551 containing_csect: None,
552 },
553 });
554 }
555 _ => {}
556 };
557 file.append_section_data(section, data, 1);
558 (file.write().unwrap(), MetadataPosition::First)
559}
560
561pub fn create_compressed_metadata_file(
576 sess: &Session,
577 metadata: &EncodedMetadata,
578 symbol_name: &str,
579) -> Vec<u8> {
580 let mut packed_metadata = rustc_metadata::METADATA_HEADER.to_vec();
581 packed_metadata.write_all(&(metadata.stub_or_full().len() as u64).to_le_bytes()).unwrap();
582 packed_metadata.extend(metadata.stub_or_full());
583
584 let Some(mut file) = create_object_file(sess) else {
585 if sess.target.is_like_wasm {
586 return create_metadata_file_for_wasm(sess, &packed_metadata, ".rustc");
587 }
588 return packed_metadata.to_vec();
589 };
590 if file.format() == BinaryFormat::Xcoff {
591 return create_compressed_metadata_file_for_xcoff(file, &packed_metadata, symbol_name);
592 }
593 let section = file.add_section(
594 file.segment_name(StandardSegment::Data).to_vec(),
595 b".rustc".to_vec(),
596 SectionKind::ReadOnlyData,
597 );
598 match file.format() {
599 BinaryFormat::Elf => {
600 file.section_mut(section).flags = SectionFlags::Elf { sh_flags: 0 };
602 }
603 _ => {}
604 };
605 let offset = file.append_section_data(section, &packed_metadata, 1);
606
607 file.add_symbol(Symbol {
610 name: symbol_name.as_bytes().to_vec(),
611 value: offset,
612 size: packed_metadata.len() as u64,
613 kind: SymbolKind::Data,
614 scope: SymbolScope::Dynamic,
615 weak: false,
616 section: SymbolSection::Section(section),
617 flags: SymbolFlags::None,
618 });
619
620 file.write().unwrap()
621}
622
623pub fn create_compressed_metadata_file_for_xcoff(
637 mut file: write::Object<'_>,
638 data: &[u8],
639 symbol_name: &str,
640) -> Vec<u8> {
641 assert!(file.format() == BinaryFormat::Xcoff);
642 file.add_section(Vec::new(), b".text".to_vec(), SectionKind::Text);
644 let data_section = file.add_section(Vec::new(), b".data".to_vec(), SectionKind::Data);
645 let section = file.add_section(Vec::new(), b".info".to_vec(), SectionKind::Debug);
646 file.add_file_symbol("lib.rmeta".into());
647 file.section_mut(section).flags = SectionFlags::Xcoff { s_flags: xcoff::STYP_INFO as u32 };
648 file.add_symbol(Symbol {
650 name: symbol_name.as_bytes().into(),
651 value: 0,
652 size: 0,
653 kind: SymbolKind::Data,
654 scope: SymbolScope::Dynamic,
655 weak: true,
656 section: SymbolSection::Section(data_section),
657 flags: SymbolFlags::None,
658 });
659 let len: u32 = data.len().try_into().unwrap();
660 let offset = file.append_section_data(section, &len.to_be_bytes(), 1);
661 file.add_symbol(Symbol {
663 name: AIX_METADATA_SYMBOL_NAME.into(),
664 value: offset + 4, size: 0,
666 kind: SymbolKind::Unknown,
667 scope: SymbolScope::Dynamic,
668 weak: false,
669 section: SymbolSection::Section(section),
670 flags: SymbolFlags::Xcoff {
671 n_sclass: xcoff::C_INFO,
672 x_smtyp: xcoff::C_HIDEXT,
673 x_smclas: xcoff::C_HIDEXT,
674 containing_csect: None,
675 },
676 });
677 file.append_section_data(section, data, 1);
678 file.write().unwrap()
679}
680
681pub fn create_metadata_file_for_wasm(sess: &Session, data: &[u8], section_name: &str) -> Vec<u8> {
702 assert!(sess.target.is_like_wasm);
703 let mut module = wasm_encoder::Module::new();
704 let mut imports = wasm_encoder::ImportSection::new();
705
706 if sess.target.pointer_width == 64 {
707 imports.import(
708 "env",
709 "__linear_memory",
710 wasm_encoder::MemoryType {
711 minimum: 0,
712 maximum: None,
713 memory64: true,
714 shared: false,
715 page_size_log2: None,
716 },
717 );
718 }
719
720 if imports.len() > 0 {
721 module.section(&imports);
722 }
723 module.section(&wasm_encoder::CustomSection {
724 name: "linking".into(),
725 data: Cow::Borrowed(&[2]),
726 });
727 module.section(&wasm_encoder::CustomSection { name: section_name.into(), data: data.into() });
728 module.finish()
729}