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::{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.as_ref() {
264 "hermit" => elf::ELFOSABI_STANDALONE,
265 "freebsd" => elf::ELFOSABI_FREEBSD,
266 "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 specifed 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.unstable_target_features.contains(&sym::c) {
335 e_flags |= elf::EF_RISCV_RVC;
336 }
337
338 match &*sess.target.llvm_abiname {
341 "ilp32" | "lp64" => (),
342 "ilp32f" | "lp64f" => e_flags |= elf::EF_RISCV_FLOAT_ABI_SINGLE,
343 "ilp32d" | "lp64d" => e_flags |= elf::EF_RISCV_FLOAT_ABI_DOUBLE,
344 "ilp32e" | "lp64e" => e_flags |= elf::EF_RISCV_RVE,
346 _ => bug!("unknown RISC-V ABI name"),
347 }
348
349 e_flags
350 }
351 Architecture::LoongArch64 => {
352 let mut e_flags: u32 = elf::EF_LARCH_OBJABI_V1;
354
355 match &*sess.target.llvm_abiname {
358 "ilp32s" | "lp64s" => e_flags |= elf::EF_LARCH_ABI_SOFT_FLOAT,
359 "ilp32f" | "lp64f" => e_flags |= elf::EF_LARCH_ABI_SINGLE_FLOAT,
360 "ilp32d" | "lp64d" => e_flags |= elf::EF_LARCH_ABI_DOUBLE_FLOAT,
361 _ => bug!("unknown LoongArch ABI name"),
362 }
363
364 e_flags
365 }
366 Architecture::Avr => {
367 if let Some(ref cpu) = sess.opts.cg.target_cpu {
370 ef_avr_arch(cpu)
371 } else {
372 bug!("AVR CPU not explicitly specified")
373 }
374 }
375 Architecture::Csky => {
376 let e_flags = match sess.target.options.abi.as_ref() {
377 "abiv2" => elf::EF_CSKY_ABIV2,
378 _ => elf::EF_CSKY_ABIV1,
379 };
380 e_flags
381 }
382 _ => 0,
383 }
384}
385
386fn macho_object_build_version_for_target(sess: &Session) -> object::write::MachOBuildVersion {
405 fn pack_version(apple::OSVersion { major, minor, patch }: apple::OSVersion) -> u32 {
408 let (major, minor, patch) = (major as u32, minor as u32, patch as u32);
409 (major << 16) | (minor << 8) | patch
410 }
411
412 let platform = apple::macho_platform(&sess.target);
413 let min_os = sess.apple_deployment_target();
414
415 let mut build_version = object::write::MachOBuildVersion::default();
416 build_version.platform = platform;
417 build_version.minos = pack_version(min_os);
418 build_version.sdk = 0;
424
425 build_version
426}
427
428fn macho_is_arm64e(target: &Target) -> bool {
430 target.llvm_target.starts_with("arm64e")
431}
432
433pub(crate) enum MetadataPosition {
434 First,
435 Last,
436}
437
438pub(crate) fn create_wrapper_file(
470 sess: &Session,
471 section_name: String,
472 data: &[u8],
473) -> (Vec<u8>, MetadataPosition) {
474 let Some(mut file) = create_object_file(sess) else {
475 if sess.target.is_like_wasm {
476 return (
477 create_metadata_file_for_wasm(sess, data, §ion_name),
478 MetadataPosition::First,
479 );
480 }
481
482 return (data.to_vec(), MetadataPosition::Last);
486 };
487 let section = if file.format() == BinaryFormat::Xcoff {
488 file.add_section(Vec::new(), b".info".to_vec(), SectionKind::Debug)
489 } else {
490 file.add_section(
491 file.segment_name(StandardSegment::Debug).to_vec(),
492 section_name.into_bytes(),
493 SectionKind::Debug,
494 )
495 };
496 match file.format() {
497 BinaryFormat::Coff => {
498 file.section_mut(section).flags =
499 SectionFlags::Coff { characteristics: pe::IMAGE_SCN_LNK_REMOVE };
500 }
501 BinaryFormat::Elf => {
502 file.section_mut(section).flags =
503 SectionFlags::Elf { sh_flags: elf::SHF_EXCLUDE as u64 };
504 }
505 BinaryFormat::Xcoff => {
506 file.add_section(Vec::new(), b".text".to_vec(), SectionKind::Text);
508 file.section_mut(section).flags =
509 SectionFlags::Xcoff { s_flags: xcoff::STYP_INFO as u32 };
510 let len: u32 = data.len().try_into().unwrap();
515 let offset = file.append_section_data(section, &len.to_be_bytes(), 1);
516 file.add_symbol(Symbol {
518 name: AIX_METADATA_SYMBOL_NAME.into(),
519 value: offset + 4,
520 size: 0,
521 kind: SymbolKind::Unknown,
522 scope: SymbolScope::Compilation,
523 weak: false,
524 section: SymbolSection::Section(section),
525 flags: SymbolFlags::Xcoff {
526 n_sclass: xcoff::C_INFO,
527 x_smtyp: xcoff::C_HIDEXT,
528 x_smclas: xcoff::C_HIDEXT,
529 containing_csect: None,
530 },
531 });
532 }
533 _ => {}
534 };
535 file.append_section_data(section, data, 1);
536 (file.write().unwrap(), MetadataPosition::First)
537}
538
539pub fn create_compressed_metadata_file(
554 sess: &Session,
555 metadata: &EncodedMetadata,
556 symbol_name: &str,
557) -> Vec<u8> {
558 let mut packed_metadata = rustc_metadata::METADATA_HEADER.to_vec();
559 packed_metadata.write_all(&(metadata.stub_or_full().len() as u64).to_le_bytes()).unwrap();
560 packed_metadata.extend(metadata.stub_or_full());
561
562 let Some(mut file) = create_object_file(sess) else {
563 if sess.target.is_like_wasm {
564 return create_metadata_file_for_wasm(sess, &packed_metadata, ".rustc");
565 }
566 return packed_metadata.to_vec();
567 };
568 if file.format() == BinaryFormat::Xcoff {
569 return create_compressed_metadata_file_for_xcoff(file, &packed_metadata, symbol_name);
570 }
571 let section = file.add_section(
572 file.segment_name(StandardSegment::Data).to_vec(),
573 b".rustc".to_vec(),
574 SectionKind::ReadOnlyData,
575 );
576 match file.format() {
577 BinaryFormat::Elf => {
578 file.section_mut(section).flags = SectionFlags::Elf { sh_flags: 0 };
580 }
581 _ => {}
582 };
583 let offset = file.append_section_data(section, &packed_metadata, 1);
584
585 file.add_symbol(Symbol {
588 name: symbol_name.as_bytes().to_vec(),
589 value: offset,
590 size: packed_metadata.len() as u64,
591 kind: SymbolKind::Data,
592 scope: SymbolScope::Dynamic,
593 weak: false,
594 section: SymbolSection::Section(section),
595 flags: SymbolFlags::None,
596 });
597
598 file.write().unwrap()
599}
600
601pub fn create_compressed_metadata_file_for_xcoff(
615 mut file: write::Object<'_>,
616 data: &[u8],
617 symbol_name: &str,
618) -> Vec<u8> {
619 assert!(file.format() == BinaryFormat::Xcoff);
620 file.add_section(Vec::new(), b".text".to_vec(), SectionKind::Text);
622 let data_section = file.add_section(Vec::new(), b".data".to_vec(), SectionKind::Data);
623 let section = file.add_section(Vec::new(), b".info".to_vec(), SectionKind::Debug);
624 file.add_file_symbol("lib.rmeta".into());
625 file.section_mut(section).flags = SectionFlags::Xcoff { s_flags: xcoff::STYP_INFO as u32 };
626 file.add_symbol(Symbol {
628 name: symbol_name.as_bytes().into(),
629 value: 0,
630 size: 0,
631 kind: SymbolKind::Data,
632 scope: SymbolScope::Dynamic,
633 weak: true,
634 section: SymbolSection::Section(data_section),
635 flags: SymbolFlags::None,
636 });
637 let len: u32 = data.len().try_into().unwrap();
638 let offset = file.append_section_data(section, &len.to_be_bytes(), 1);
639 file.add_symbol(Symbol {
641 name: AIX_METADATA_SYMBOL_NAME.into(),
642 value: offset + 4, size: 0,
644 kind: SymbolKind::Unknown,
645 scope: SymbolScope::Dynamic,
646 weak: false,
647 section: SymbolSection::Section(section),
648 flags: SymbolFlags::Xcoff {
649 n_sclass: xcoff::C_INFO,
650 x_smtyp: xcoff::C_HIDEXT,
651 x_smclas: xcoff::C_HIDEXT,
652 containing_csect: None,
653 },
654 });
655 file.append_section_data(section, data, 1);
656 file.write().unwrap()
657}
658
659pub fn create_metadata_file_for_wasm(sess: &Session, data: &[u8], section_name: &str) -> Vec<u8> {
680 assert!(sess.target.is_like_wasm);
681 let mut module = wasm_encoder::Module::new();
682 let mut imports = wasm_encoder::ImportSection::new();
683
684 if sess.target.pointer_width == 64 {
685 imports.import(
686 "env",
687 "__linear_memory",
688 wasm_encoder::MemoryType {
689 minimum: 0,
690 maximum: None,
691 memory64: true,
692 shared: false,
693 page_size_log2: None,
694 },
695 );
696 }
697
698 if imports.len() > 0 {
699 module.section(&imports);
700 }
701 module.section(&wasm_encoder::CustomSection {
702 name: "linking".into(),
703 data: Cow::Borrowed(&[2]),
704 });
705 module.section(&wasm_encoder::CustomSection { name: section_name.into(), data: data.into() });
706 module.finish()
707}