Skip to content

Commit 9236a02

Browse files
committed
Add interactive option to UnstableOptions
If `interactive` is set to true, codegen will special case that to get faster and for the resulting LLVM IR to be executable from external programs such as REPL. This commit also has the following changes: - Added LLVM JIT-related API. - Stopped distributing rustc crates as rmeta files. See added code comment and rust-lang#102065 for detailed info. - Some visibility changes on existing APIs so they could be used by external programs.
1 parent 992d154 commit 9236a02

File tree

13 files changed

+192
-17
lines changed

13 files changed

+192
-17
lines changed

Cargo.lock

+13
Original file line numberDiff line numberDiff line change
@@ -4155,6 +4155,7 @@ dependencies = [
41554155
"rustc_codegen_ssa",
41564156
"rustc_driver",
41574157
"rustc_driver_impl",
4158+
"rustc_interactive",
41584159
"rustc_smir",
41594160
]
41604161

@@ -4807,6 +4808,18 @@ dependencies = [
48074808
"tracing",
48084809
]
48094810

4811+
[[package]]
4812+
name = "rustc_interactive"
4813+
version = "0.0.0"
4814+
dependencies = [
4815+
"rustc_codegen_llvm",
4816+
"rustc_codegen_ssa",
4817+
"rustc_hir",
4818+
"rustc_llvm",
4819+
"rustc_middle",
4820+
"rustc_span",
4821+
]
4822+
48104823
[[package]]
48114824
name = "rustc_interface"
48124825
version = "0.0.0"

compiler/rustc/Cargo.toml

+3
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ rustc_codegen_ssa = { path = "../rustc_codegen_ssa" }
1313
# Make sure rustc_smir ends up in the sysroot, because this
1414
# crate is intended to be used by stable MIR consumers, which are not in-tree
1515
rustc_smir = { path = "../rustc_smir" }
16+
# Make sure rustc_interactive ends up in the sysroot, because this
17+
# crate is intended to be used by programs that use the compiler as a library, which are not in-tree.
18+
rustc_interactive = { path = "../rustc_interactive" }
1619

1720
[dependencies.jemalloc-sys]
1821
version = "0.5.0"

compiler/rustc_codegen_llvm/src/callee.rs

+15
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,17 @@
44
//! and methods are represented as just a fn ptr and not a full
55
//! closure.
66
7+
use std::ffi::CString;
8+
79
use crate::abi::FnAbiLlvmExt;
810
use crate::attributes;
11+
use crate::builder::Builder;
912
use crate::common;
1013
use crate::context::CodegenCx;
1114
use crate::llvm;
1215
use crate::value::Value;
16+
use llvm::LLVMSearchForAddressOfSymbol;
17+
use rustc_codegen_ssa::base::codegen_instance;
1318
use rustc_codegen_ssa::traits::*;
1419

1520
use rustc_middle::ty::layout::{FnAbiOf, HasTyCtxt};
@@ -204,6 +209,16 @@ pub fn get_fn<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, instance: Instance<'tcx>) ->
204209
}
205210
}
206211

212+
if cx.sess().opts.unstable_opts.interactive {
213+
let addr = unsafe {
214+
let name = CString::new(sym).unwrap();
215+
LLVMSearchForAddressOfSymbol(name.as_ptr())
216+
};
217+
if addr.is_null() {
218+
codegen_instance::<Builder<'_, '_, '_>>(cx, instance);
219+
}
220+
}
221+
207222
llfn
208223
};
209224

compiler/rustc_codegen_llvm/src/context.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -342,7 +342,7 @@ pub unsafe fn create_module<'ll>(
342342
}
343343

344344
impl<'ll, 'tcx> CodegenCx<'ll, 'tcx> {
345-
pub(crate) fn new(
345+
pub fn new(
346346
tcx: TyCtxt<'tcx>,
347347
codegen_unit: &'tcx CodegenUnit<'tcx>,
348348
llvm_module: &'ll crate::ModuleLlvm,

compiler/rustc_codegen_llvm/src/lib.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -59,11 +59,11 @@ mod allocator;
5959
mod asm;
6060
mod attributes;
6161
mod base;
62-
mod builder;
62+
pub mod builder;
6363
mod callee;
6464
mod common;
6565
mod consts;
66-
mod context;
66+
pub mod context;
6767
mod coverageinfo;
6868
mod debuginfo;
6969
mod declare;
@@ -396,7 +396,7 @@ unsafe impl Send for ModuleLlvm {}
396396
unsafe impl Sync for ModuleLlvm {}
397397

398398
impl ModuleLlvm {
399-
fn new(tcx: TyCtxt<'_>, mod_name: &str) -> Self {
399+
pub fn new(tcx: TyCtxt<'_>, mod_name: &str) -> Self {
400400
unsafe {
401401
let llcx = llvm::LLVMRustContextCreate(tcx.sess.fewer_names());
402402
let llmod_raw = context::create_module(tcx, llcx, mod_name) as *const _;
@@ -433,7 +433,7 @@ impl ModuleLlvm {
433433
}
434434
}
435435

436-
fn llmod(&self) -> &llvm::Module {
436+
pub fn llmod(&self) -> &llvm::Module {
437437
unsafe { &*self.llmod_raw }
438438
}
439439
}

compiler/rustc_codegen_llvm/src/llvm/ffi.rs

+21
Original file line numberDiff line numberDiff line change
@@ -669,6 +669,15 @@ pub struct OperandBundleDef<'a>(InvariantOpaque<'a>);
669669
#[repr(C)]
670670
pub struct Linker<'a>(InvariantOpaque<'a>);
671671

672+
extern "C" {
673+
pub type LLJIT;
674+
}
675+
unsafe impl Sync for LLJIT {}
676+
677+
extern "C" {
678+
pub type JITDylib;
679+
}
680+
672681
extern "C" {
673682
pub type DiagnosticHandler;
674683
}
@@ -2491,4 +2500,16 @@ extern "C" {
24912500
callback: GetSymbolsCallback,
24922501
error_callback: GetSymbolsErrorCallback,
24932502
) -> *mut c_void;
2503+
2504+
pub fn LLVMRustCreateLLJIT() -> &'static mut LLJIT;
2505+
pub fn LLVMRustLLJITLoadDynamicLibrary(J: &LLJIT, FileName: *const c_char);
2506+
pub fn LLVMRustLLJITAddIRModule(J: &LLJIT, M: &Module);
2507+
pub fn LLVMRustLLJITLookup(J: &LLJIT, Name: *const c_char) -> u64;
2508+
pub fn LLVMRustSetLinkageForAllFunctions(M: &Module);
2509+
pub fn LLVMRustAddGlobalCtor(C: &Context, M: &Module, V: &Value);
2510+
pub fn LLVMRustRunCtors(J: &LLJIT);
2511+
2512+
pub fn LLVMDumpModule(M: &Module);
2513+
2514+
pub fn LLVMSearchForAddressOfSymbol(SymbolName: *const c_char) -> *mut c_void;
24942515
}

compiler/rustc_codegen_ssa/src/mir/block.rs

+10
Original file line numberDiff line numberDiff line change
@@ -747,6 +747,16 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
747747
};
748748
let def = instance.map(|i| i.def);
749749

750+
if self.cx.sess().opts.unstable_opts.interactive
751+
&& let Some(instance) = instance {
752+
match instance.def {
753+
ty::InstanceDef::Intrinsic(_) => {}
754+
_ => {
755+
self.cx.get_fn(instance);
756+
}
757+
}
758+
}
759+
750760
if let Some(ty::InstanceDef::DropGlue(_, None)) = def {
751761
// Empty drop glue; a no-op.
752762
let target = target.unwrap();

compiler/rustc_interactive/Cargo.toml

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
[package]
2+
name = "rustc_interactive"
3+
version = "0.0.0"
4+
edition = "2021"
5+
6+
[dependencies]
7+
rustc_codegen_llvm = { path = "../rustc_codegen_llvm" }
8+
rustc_codegen_ssa = { path = "../rustc_codegen_ssa" }
9+
rustc_hir = { path = "../rustc_hir" }
10+
rustc_llvm = { path = "../rustc_llvm" }
11+
rustc_middle = { path = "../rustc_middle" }
12+
rustc_span = { path = "../rustc_span" }

compiler/rustc_interactive/src/lib.rs

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
//! Utilities for interactively use the Rust compiler from outside of
2+
//! the source tree.
3+
4+
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/nightly-rustc")]

compiler/rustc_llvm/build.rs

+12-2
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,18 @@ const OPTIONAL_COMPONENTS: &[&str] = &[
2424
"bpf",
2525
];
2626

27-
const REQUIRED_COMPONENTS: &[&str] =
28-
&["ipo", "bitreader", "bitwriter", "linker", "asmparser", "lto", "coverage", "instrumentation"];
27+
const REQUIRED_COMPONENTS: &[&str] = &[
28+
"ipo",
29+
"bitreader",
30+
"bitwriter",
31+
"linker",
32+
"asmparser",
33+
"lto",
34+
"coverage",
35+
"instrumentation",
36+
"executionengine",
37+
"orcjit",
38+
];
2939

3040
fn detect_llvm_link() -> (&'static str, &'static str) {
3141
// Force the link mode we want, preferring static by default, but

compiler/rustc_llvm/llvm-wrapper/RustWrapper.cpp

+88
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@
1111
#if LLVM_VERSION_GE(16, 0)
1212
#include "llvm/Support/ModRef.h"
1313
#endif
14+
#include "llvm/ExecutionEngine/Orc/LLJIT.h"
15+
#include "llvm/ExecutionEngine/Orc/ThreadSafeModule.h"
1416
#include "llvm/Object/Archive.h"
1517
#include "llvm/Object/COFFImportFile.h"
1618
#include "llvm/Object/ObjectFile.h"
@@ -33,6 +35,7 @@
3335
using namespace llvm;
3436
using namespace llvm::sys;
3537
using namespace llvm::object;
38+
using namespace llvm::orc;
3639

3740
// LLVMAtomicOrdering is already an enum - don't create another
3841
// one.
@@ -1996,3 +1999,88 @@ extern "C" int32_t LLVMRustGetElementTypeArgIndex(LLVMValueRef CallSite) {
19961999
extern "C" bool LLVMRustIsBitcode(char *ptr, size_t len) {
19972000
return identify_magic(StringRef(ptr, len)) == file_magic::bitcode;
19982001
}
2002+
2003+
extern "C" LLJIT *LLVMRustCreateLLJIT() {
2004+
Expected<std::unique_ptr<LLJIT>> J = LLJITBuilder().create();
2005+
if (!J)
2006+
report_fatal_error(toString(J.takeError()).c_str());
2007+
2008+
return J->release();
2009+
}
2010+
2011+
extern "C" void LLVMRustLLJITLoadDynamicLibrary(LLJIT *J,
2012+
const char *FileName) {
2013+
auto Generator = DynamicLibrarySearchGenerator::Load(
2014+
FileName, J->getDataLayout().getGlobalPrefix());
2015+
if (!Generator)
2016+
report_fatal_error(toString(Generator.takeError()).c_str());
2017+
2018+
J->getMainJITDylib().addGenerator(std::move(*Generator));
2019+
}
2020+
2021+
extern "C" void LLVMRustLLJITAddIRModule(LLJIT *J, LLVMModuleRef ModuleRef) {
2022+
auto M = std::unique_ptr<Module>(unwrap(ModuleRef));
2023+
auto C = std::make_unique<LLVMContext>();
2024+
ThreadSafeModule TSM = ThreadSafeModule(std::move(M), std::move(C));
2025+
auto Error = J->addIRModule(std::move(TSM));
2026+
if (Error) {
2027+
std::string errorString;
2028+
llvm::raw_string_ostream stream(errorString);
2029+
stream << Error;
2030+
report_fatal_error(errorString.c_str());
2031+
}
2032+
}
2033+
2034+
extern "C" uint64_t LLVMRustLLJITLookup(LLJIT *J, const char *Name) {
2035+
auto Addr = J->lookup(Name);
2036+
if (!Addr)
2037+
report_fatal_error(toString(Addr.takeError()).c_str());
2038+
2039+
return Addr->getValue();
2040+
}
2041+
2042+
extern "C" void LLVMRustSetLinkageForAllFunctions(LLVMModuleRef M) {
2043+
auto Module = unwrap(M);
2044+
for (auto F = Module->getFunctionList().begin();
2045+
F != Module->getFunctionList().end(); F++) {
2046+
(*F).setLinkage(GlobalValue::LinkageTypes::LinkOnceODRLinkage);
2047+
}
2048+
}
2049+
2050+
extern "C" void LLVMRustAddGlobalCtor(LLVMContextRef C, LLVMModuleRef M,
2051+
LLVMValueRef Fn) {
2052+
auto AddrSpace = unwrap(M)->getDataLayout().getProgramAddressSpace();
2053+
auto CtorFnPtrTy =
2054+
PointerType::get(unwrap<FunctionType>(LLVMTypeOf(Fn)), AddrSpace);
2055+
2056+
auto VoidPtrTy = PointerType::get(Type::getVoidTy(*unwrap(C)), AddrSpace);
2057+
2058+
auto CtorTy =
2059+
StructType::get(Type::getInt32Ty(*unwrap(C)), CtorFnPtrTy, VoidPtrTy);
2060+
2061+
auto Ctor = ConstantStruct::get(
2062+
CtorTy, {
2063+
ConstantInt::get(Type::getInt32Ty(*unwrap(C)), 65535),
2064+
ConstantExpr::getBitCast(unwrap<Constant>(Fn), CtorFnPtrTy),
2065+
ConstantPointerNull::get(VoidPtrTy),
2066+
});
2067+
2068+
auto CtorsTy = ArrayType::get(CtorTy, 1);
2069+
auto Ctors = ConstantArray::get(CtorsTy, {Ctor});
2070+
2071+
auto GlobalCtors = unwrap(M)->getOrInsertGlobal("llvm.global_ctors", CtorsTy);
2072+
unwrap(M)
2073+
->getGlobalVariable("llvm.global_ctors")
2074+
->setLinkage(GlobalValue::LinkageTypes::AppendingLinkage);
2075+
unwrap(M)->getGlobalVariable("llvm.global_ctors")->setInitializer(Ctors);
2076+
}
2077+
2078+
extern "C" void LLVMRustRunCtors(LLJIT *J) {
2079+
auto Error = J->initialize(J->getMainJITDylib());
2080+
if (Error) {
2081+
std::string errorString;
2082+
raw_string_ostream stream(errorString);
2083+
stream << Error;
2084+
report_fatal_error(errorString.c_str());
2085+
}
2086+
}

compiler/rustc_session/src/options.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1485,6 +1485,8 @@ options! {
14851485
`=skip-entry`
14861486
`=skip-exit`
14871487
Multiple options can be combined with commas."),
1488+
interactive: bool = (false, parse_bool, [TRACKED],
1489+
"enable interactive mode (default: no)"),
14881490
keep_hygiene_data: bool = (false, parse_bool, [UNTRACKED],
14891491
"keep hygiene data after analysis (default: no)"),
14901492
layout_seed: Option<u64> = (None, parse_opt_number, [TRACKED],

src/bootstrap/compile.rs

+7-10
Original file line numberDiff line numberDiff line change
@@ -1535,16 +1535,13 @@ pub fn run_cargo(
15351535
// During check builds we need to keep crate metadata
15361536
keep = true;
15371537
} else if rlib_only_metadata {
1538-
if filename.contains("jemalloc_sys") || filename.contains("rustc_smir") {
1539-
// jemalloc_sys and rustc_smir are not linked into librustc_driver.so,
1540-
// so we need to distribute them as rlib to be able to use them.
1541-
keep |= filename.ends_with(".rlib");
1542-
} else {
1543-
// Distribute the rest of the rustc crates as rmeta files only to reduce
1544-
// the tarball sizes by about 50%. The object files are linked into
1545-
// librustc_driver.so, so it is still possible to link against them.
1546-
keep |= filename.ends_with(".rmeta");
1547-
}
1538+
// Don't distribute rustc crates as rmeta files because doing
1539+
// so will require rustc_driver to be imported, which causes
1540+
// a linking error.
1541+
//
1542+
// See <https://github.com/rust-lang/rust/issues/102065> for
1543+
// more info.
1544+
keep |= filename.ends_with(".rlib");
15481545
} else {
15491546
// In all other cases keep all rlibs
15501547
keep |= filename.ends_with(".rlib");

0 commit comments

Comments
 (0)