diff options
| author | Sebastian Humenda <shumenda@gmx.de> | 2018-04-03 14:53:13 +0200 |
|---|---|---|
| committer | Benjamin Lamowski <benjamin.lamowski@kernkonzept.com> | 2022-01-21 16:28:33 +0100 |
| commit | d98428711e6fe1b2a7f6d963d4b337beaa3dc285 (patch) | |
| tree | 45ec87733933a866c778507b6c7a4ce73afc9090 /compiler/rustc_codegen_ssa/src/back | |
| parent | 84e918971d643c6a33067d5125214ab800ce5307 (diff) | |
| download | rust-d98428711e6fe1b2a7f6d963d4b337beaa3dc285.tar.gz rust-d98428711e6fe1b2a7f6d963d4b337beaa3dc285.zip | |
Add L4Bender as linker variant
Diffstat (limited to 'compiler/rustc_codegen_ssa/src/back')
| -rw-r--r-- | compiler/rustc_codegen_ssa/src/back/link.rs | 1 | ||||
| -rw-r--r-- | compiler/rustc_codegen_ssa/src/back/linker.rs | 172 |
2 files changed, 172 insertions, 1 deletions
diff --git a/compiler/rustc_codegen_ssa/src/back/link.rs b/compiler/rustc_codegen_ssa/src/back/link.rs index acf65259f61..ec9fc22bc4d 100644 --- a/compiler/rustc_codegen_ssa/src/back/link.rs +++ b/compiler/rustc_codegen_ssa/src/back/link.rs @@ -1159,6 +1159,7 @@ pub fn linker_and_flavor(sess: &Session) -> (PathBuf, LinkerFlavor) { LinkerFlavor::Lld(_) => "lld", LinkerFlavor::PtxLinker => "rust-ptx-linker", LinkerFlavor::BpfLinker => "bpf-linker", + LinkerFlavor::L4Bender => "l4-bender", }), flavor, )), diff --git a/compiler/rustc_codegen_ssa/src/back/linker.rs b/compiler/rustc_codegen_ssa/src/back/linker.rs index 15d16e7d3d6..e2357eae7c1 100644 --- a/compiler/rustc_codegen_ssa/src/back/linker.rs +++ b/compiler/rustc_codegen_ssa/src/back/linker.rs @@ -3,6 +3,7 @@ use super::command::Command; use super::symbol_export; use rustc_span::symbol::sym; +use std::env; use std::ffi::{OsStr, OsString}; use std::fs::{self, File}; use std::io::prelude::*; @@ -126,7 +127,6 @@ pub fn get_linker<'a>( // FIXME: Move `/LIBPATH` addition for uwp targets from the linker construction // to the linker args construction. assert!(cmd.get_args().is_empty() || sess.target.vendor == "uwp"); - match flavor { LinkerFlavor::Lld(LldFlavor::Link) | LinkerFlavor::Msvc => { Box::new(MsvcLinker { cmd, sess }) as Box<dyn Linker> @@ -149,6 +149,10 @@ pub fn get_linker<'a>( LinkerFlavor::PtxLinker => Box::new(PtxLinker { cmd, sess }) as Box<dyn Linker>, LinkerFlavor::BpfLinker => Box::new(BpfLinker { cmd, sess }) as Box<dyn Linker>, + + LinkerFlavor::L4Bender => { + Box::new(L4Bender::new(cmd, sess)) as Box<dyn Linker> + }, } } @@ -1355,6 +1359,172 @@ impl<'a> Linker for WasmLd<'a> { } } +/// Linker shepherd script for L4Re (Fiasco) +pub struct L4Bender<'a> { + cmd: Command, + sess: &'a Session, + hinted_static: bool, +} + +impl<'a> Linker for L4Bender<'a> { + fn link_dylib(&mut self, _lib: Symbol) { + panic!("dylibs not supported yet") + } + fn link_staticlib(&mut self, lib: Symbol) { + self.hint_static(); + self.cmd.arg(format!("-PC{}", lib)); + } + fn link_rlib(&mut self, lib: &Path) { + self.hint_static(); + self.cmd.arg(lib); + } + fn include_path(&mut self, path: &Path) { + self.cmd.arg("-L").arg(path); + } + fn framework_path(&mut self, _: &Path) { + bug!("Frameworks are not supported on L4Re!"); + } + fn output_filename(&mut self, path: &Path) { self.cmd.arg("-o").arg(path); } + fn add_object(&mut self, path: &Path) { self.cmd.arg(path); } + // not sure about pie on L4Re + fn position_independent_executable(&mut self) { } + fn no_position_independent_executable(&mut self) { } + fn full_relro(&mut self) { self.cmd.arg("-z,relro,-z,now"); } + fn partial_relro(&mut self) { self.cmd.arg("-z,relro"); } + fn no_relro(&mut self) { self.cmd.arg("-z,norelro"); } + fn build_static_executable(&mut self) { self.cmd.arg("-static"); } + fn cmd(&mut self) -> &mut Command { + &mut self.cmd + } + + fn link_rust_dylib(&mut self, _: Symbol, _: &Path) { + panic!("Rust dylibs not supported"); + } + + fn link_framework(&mut self, _: Symbol) { + bug!("Frameworks not supported on L4Re."); + } + + // Here we explicitly ask that the entire archive is included into the + // result artifact. For more details see #15460, but the gist is that + // the linker will strip away any unused objects in the archive if we + // don't otherwise explicitly reference them. This can occur for + // libraries which are just providing bindings, libraries with generic + // functions, etc. + fn link_whole_staticlib(&mut self, lib: Symbol, _: &[PathBuf]) { + self.hint_static(); + self.cmd.arg("--whole-archive").arg(format!("-l{}", lib)); + self.cmd.arg("--no-whole-archive"); + } + + fn link_whole_rlib(&mut self, lib: &Path) { + self.hint_static(); + self.cmd.arg("--whole-archive").arg(lib).arg("--no-whole-archive"); + } + + fn gc_sections(&mut self, keep_metadata: bool) { + if !keep_metadata { + self.cmd.arg("--gc-sections"); + } + } + + fn optimize(&mut self) { + self.cmd.arg("-O2"); + } + + fn pgo_gen(&mut self) { } + + fn debuginfo(&mut self, strip: Strip) { + match strip { + Strip::None => {} + Strip::Debuginfo => { + self.cmd().arg("--strip-debug"); } + Strip::Symbols => { + self.cmd().arg("--strip-all"); + } + } + } + + fn no_default_libraries(&mut self) { + self.cmd.arg("-nostdlib"); + } + + fn build_dylib(&mut self, _: &Path) { + bug!("not implemented"); + } + + fn export_symbols(&mut self, _: &Path, _: CrateType) { + // ToDo, not implemented, copy from GCC + return; + } + + fn subsystem(&mut self, subsystem: &str) { + self.cmd.arg(&format!("--subsystem,{}", subsystem)); + } + + fn finalize(&mut self) { + self.hint_static(); // Reset to default before returning the composed command line. + } + + fn group_start(&mut self) { self.cmd.arg("--start-group"); } + fn group_end(&mut self) { self.cmd.arg("--end-group"); } + fn linker_plugin_lto(&mut self) { + // do nothing + } + fn control_flow_guard(&mut self) { + self.sess.warn("Windows Control Flow Guard is not supported by this linker."); + } + + fn no_crt_objects(&mut self) { } +} + +impl<'a> L4Bender<'a> { + pub fn new(mut cmd: Command, sess: &'a Session) -> L4Bender<'a> { + if let Ok(l4bender_args) = env::var("L4_BENDER_ARGS") { + L4Bender::split_cmd_args(&mut cmd, &l4bender_args); + } + + cmd.arg("--"); // separate direct l4-bender args from linker args + + L4Bender { + cmd: cmd, + sess: sess, + hinted_static: false, + } + } + + /// This parses a shell-escaped string and unquotes the arguments. It doesn't attempt to + /// completely understand shell, but should instead allow passing arguments like + /// `-Dlinker="ld -m x86_64"`, and a copy without quotes, but spaces preserved, is added as an + /// argument to the given Command. This means that constructs as \" are not understood, so + /// quote wisely. + fn split_cmd_args(cmd: &mut Command, shell_args: &str) { + let mut arg = String::new(); + let mut quoted = false; + for character in shell_args.chars() { + match character { + ' ' if !quoted => { + cmd.arg(&arg); + arg.clear(); + }, + '"' | '\'' => quoted = !quoted, + _ => arg.push(character), + }; + } + if arg.len() > 0 { + cmd.arg(&arg); + arg.clear(); + } + } + + fn hint_static(&mut self) { + if !self.hinted_static { + self.cmd.arg("-static"); + self.hinted_static = true; + } + } +} + pub(crate) fn exported_symbols(tcx: TyCtxt<'_>, crate_type: CrateType) -> Vec<String> { if let Some(ref exports) = tcx.sess.target.override_export_symbols { return exports.clone(); |
