Skip to content

Commit 3b6412b

Browse files
committed
Auto merge of #48896 - alexcrichton:bitcode-in-object, r=michaelwoerister
rustc: Enable embedding LLVM bitcode for iOS This commit updates rustc to embed bitcode in each object file generated by default when compiling for iOS. This was determined in #35968 as a step towards better compatibility with the iOS toolchain, so let's give it a spin and see how it turns out! Note that this also updates the `cc` dependency which should propagate this change of embedding bitcode for C dependencies as well.
2 parents cc34ca1 + 0e0f74b commit 3b6412b

File tree

5 files changed

+89
-4
lines changed

5 files changed

+89
-4
lines changed

src/bootstrap/cc_detect.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ pub fn find(build: &mut Build) {
7777
.collect::<HashSet<_>>();
7878
for target in targets.into_iter() {
7979
let mut cfg = cc::Build::new();
80-
cfg.cargo_metadata(false).opt_level(0).warnings(false).debug(false)
80+
cfg.cargo_metadata(false).opt_level(2).warnings(false).debug(false)
8181
.target(&target).host(&build.build);
8282
if target.contains("msvc") {
8383
cfg.static_crt(true);
@@ -109,7 +109,7 @@ pub fn find(build: &mut Build) {
109109
let hosts = build.hosts.iter().cloned().chain(iter::once(build.build)).collect::<HashSet<_>>();
110110
for host in hosts.into_iter() {
111111
let mut cfg = cc::Build::new();
112-
cfg.cargo_metadata(false).opt_level(0).warnings(false).debug(false).cpp(true)
112+
cfg.cargo_metadata(false).opt_level(2).warnings(false).debug(false).cpp(true)
113113
.target(&host).host(&build.build);
114114
let config = build.config.target_config.get(&host);
115115
if let Some(cxx) = config.and_then(|c| c.cxx.as_ref()) {

src/librustc/session/config.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1288,6 +1288,8 @@ options! {DebuggingOptions, DebuggingSetter, basic_debugging_options,
12881288
"run `dsymutil` and delete intermediate object files"),
12891289
ui_testing: bool = (false, parse_bool, [UNTRACKED],
12901290
"format compiler diagnostics in a way that's better suitable for UI testing"),
1291+
embed_bitcode: bool = (false, parse_bool, [TRACKED],
1292+
"embed LLVM bitcode in object files"),
12911293
}
12921294

12931295
pub fn default_lib_output() -> CrateType {

src/librustc_back/target/mod.rs

+6
Original file line numberDiff line numberDiff line change
@@ -473,6 +473,9 @@ pub struct TargetOptions {
473473
/// The default visibility for symbols in this target should be "hidden"
474474
/// rather than "default"
475475
pub default_hidden_visibility: bool,
476+
477+
/// Whether or not bitcode is embedded in object files
478+
pub embed_bitcode: bool,
476479
}
477480

478481
impl Default for TargetOptions {
@@ -544,6 +547,7 @@ impl Default for TargetOptions {
544547
i128_lowering: false,
545548
codegen_backend: "llvm".to_string(),
546549
default_hidden_visibility: false,
550+
embed_bitcode: false,
547551
}
548552
}
549553
}
@@ -792,6 +796,7 @@ impl Target {
792796
key!(no_builtins, bool);
793797
key!(codegen_backend);
794798
key!(default_hidden_visibility, bool);
799+
key!(embed_bitcode, bool);
795800

796801
if let Some(array) = obj.find("abi-blacklist").and_then(Json::as_array) {
797802
for name in array.iter().filter_map(|abi| abi.as_string()) {
@@ -990,6 +995,7 @@ impl ToJson for Target {
990995
target_option_val!(no_builtins);
991996
target_option_val!(codegen_backend);
992997
target_option_val!(default_hidden_visibility);
998+
target_option_val!(embed_bitcode);
993999

9941000
if default.abi_blacklist != self.options.abi_blacklist {
9951001
d.insert("abi-blacklist".to_string(), self.options.abi_blacklist.iter()

src/librustc_trans/back/write.rs

+77-1
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ use syntax_pos::MultiSpan;
4242
use syntax_pos::symbol::Symbol;
4343
use type_::Type;
4444
use context::{is_pie_binary, get_reloc_model};
45+
use common::{C_bytes_in_context, val_ty};
4546
use jobserver::{Client, Acquired};
4647
use rustc_demangle;
4748

@@ -262,6 +263,8 @@ pub struct ModuleConfig {
262263
// emscripten's ecc compiler, when used as the linker.
263264
obj_is_bitcode: bool,
264265
no_integrated_as: bool,
266+
embed_bitcode: bool,
267+
embed_bitcode_marker: bool,
265268
}
266269

267270
impl ModuleConfig {
@@ -279,6 +282,8 @@ impl ModuleConfig {
279282
emit_asm: false,
280283
emit_obj: false,
281284
obj_is_bitcode: false,
285+
embed_bitcode: false,
286+
embed_bitcode_marker: false,
282287
no_integrated_as: false,
283288

284289
no_verify: false,
@@ -299,6 +304,17 @@ impl ModuleConfig {
299304
self.time_passes = sess.time_passes();
300305
self.inline_threshold = sess.opts.cg.inline_threshold;
301306
self.obj_is_bitcode = sess.target.target.options.obj_is_bitcode;
307+
let embed_bitcode = sess.target.target.options.embed_bitcode ||
308+
sess.opts.debugging_opts.embed_bitcode;
309+
if embed_bitcode {
310+
match sess.opts.optimize {
311+
config::OptLevel::No |
312+
config::OptLevel::Less => {
313+
self.embed_bitcode_marker = embed_bitcode;
314+
}
315+
_ => self.embed_bitcode = embed_bitcode,
316+
}
317+
}
302318

303319
// Copy what clang does by turning on loop vectorization at O2 and
304320
// slp vectorization at O3. Otherwise configure other optimization aspects
@@ -662,7 +678,7 @@ unsafe fn codegen(cgcx: &CodegenContext,
662678
let obj_out = cgcx.output_filenames.temp_path(OutputType::Object, module_name);
663679

664680

665-
if write_bc || config.emit_bc_compressed {
681+
if write_bc || config.emit_bc_compressed || config.embed_bitcode {
666682
let thin;
667683
let old;
668684
let data = if llvm::LLVMRustThinLTOAvailable() {
@@ -681,6 +697,11 @@ unsafe fn codegen(cgcx: &CodegenContext,
681697
timeline.record("write-bc");
682698
}
683699

700+
if config.embed_bitcode {
701+
embed_bitcode(cgcx, llcx, llmod, Some(data));
702+
timeline.record("embed-bc");
703+
}
704+
684705
if config.emit_bc_compressed {
685706
let dst = bc_out.with_extension(RLIB_BYTECODE_EXTENSION);
686707
let data = bytecode::encode(&mtrans.llmod_id, data);
@@ -689,6 +710,8 @@ unsafe fn codegen(cgcx: &CodegenContext,
689710
}
690711
timeline.record("compress-bc");
691712
}
713+
} else if config.embed_bitcode_marker {
714+
embed_bitcode(cgcx, llcx, llmod, None);
692715
}
693716

694717
time_ext(config.time_passes, None, &format!("codegen passes [{}]", module_name.unwrap()),
@@ -796,6 +819,59 @@ unsafe fn codegen(cgcx: &CodegenContext,
796819
&cgcx.output_filenames))
797820
}
798821

822+
/// Embed the bitcode of an LLVM module in the LLVM module itself.
823+
///
824+
/// This is done primarily for iOS where it appears to be standard to compile C
825+
/// code at least with `-fembed-bitcode` which creates two sections in the
826+
/// executable:
827+
///
828+
/// * __LLVM,__bitcode
829+
/// * __LLVM,__cmdline
830+
///
831+
/// It appears *both* of these sections are necessary to get the linker to
832+
/// recognize what's going on. For us though we just always throw in an empty
833+
/// cmdline section.
834+
///
835+
/// Furthermore debug/O1 builds don't actually embed bitcode but rather just
836+
/// embed an empty section.
837+
///
838+
/// Basically all of this is us attempting to follow in the footsteps of clang
839+
/// on iOS. See #35968 for lots more info.
840+
unsafe fn embed_bitcode(cgcx: &CodegenContext,
841+
llcx: ContextRef,
842+
llmod: ModuleRef,
843+
bitcode: Option<&[u8]>) {
844+
let llconst = C_bytes_in_context(llcx, bitcode.unwrap_or(&[]));
845+
let llglobal = llvm::LLVMAddGlobal(
846+
llmod,
847+
val_ty(llconst).to_ref(),
848+
"rustc.embedded.module\0".as_ptr() as *const _,
849+
);
850+
llvm::LLVMSetInitializer(llglobal, llconst);
851+
let section = if cgcx.opts.target_triple.contains("-ios") {
852+
"__LLVM,__bitcode\0"
853+
} else {
854+
".llvmbc\0"
855+
};
856+
llvm::LLVMSetSection(llglobal, section.as_ptr() as *const _);
857+
llvm::LLVMRustSetLinkage(llglobal, llvm::Linkage::PrivateLinkage);
858+
859+
let llconst = C_bytes_in_context(llcx, &[]);
860+
let llglobal = llvm::LLVMAddGlobal(
861+
llmod,
862+
val_ty(llconst).to_ref(),
863+
"rustc.embedded.cmdline\0".as_ptr() as *const _,
864+
);
865+
llvm::LLVMSetInitializer(llglobal, llconst);
866+
let section = if cgcx.opts.target_triple.contains("-ios") {
867+
"__LLVM,__cmdline\0"
868+
} else {
869+
".llvmcmd\0"
870+
};
871+
llvm::LLVMSetSection(llglobal, section.as_ptr() as *const _);
872+
llvm::LLVMRustSetLinkage(llglobal, llvm::Linkage::PrivateLinkage);
873+
}
874+
799875
pub(crate) struct CompiledModules {
800876
pub modules: Vec<CompiledModule>,
801877
pub metadata_module: CompiledModule,

src/libstd/build.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,7 @@ fn main() {
8686

8787
fn build_libbacktrace(host: &str, target: &str) -> Result<(), ()> {
8888
let native = native_lib_boilerplate("libbacktrace", "libbacktrace", "backtrace", ".libs")?;
89+
let cflags = env::var("CFLAGS").unwrap_or_default() + " -fvisibility=hidden -O2";
8990

9091
run(Command::new("sh")
9192
.current_dir(&native.out_dir)
@@ -98,7 +99,7 @@ fn build_libbacktrace(host: &str, target: &str) -> Result<(), ()> {
9899
.arg("--disable-host-shared")
99100
.arg(format!("--host={}", build_helper::gnu_target(target)))
100101
.arg(format!("--build={}", build_helper::gnu_target(host)))
101-
.env("CFLAGS", env::var("CFLAGS").unwrap_or_default() + " -fvisibility=hidden"));
102+
.env("CFLAGS", cflags));
102103

103104
run(Command::new(build_helper::make(host))
104105
.current_dir(&native.out_dir)

0 commit comments

Comments
 (0)