2014-11-25 21:17:11 -05:00
|
|
|
//! The Rust compiler.
|
|
|
|
//!
|
|
|
|
//! # Note
|
|
|
|
//!
|
|
|
|
//! This API is completely unstable and subject to change.
|
2014-11-15 20:30:33 -05:00
|
|
|
|
2020-09-23 21:51:56 +02:00
|
|
|
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/nightly-rustc/")]
|
2019-10-08 01:14:42 +01:00
|
|
|
#![feature(bool_to_option)]
|
2019-05-29 04:46:55 +03:00
|
|
|
#![feature(const_cstr_unchecked)]
|
2018-06-27 17:57:25 +03:00
|
|
|
#![feature(crate_visibility_modifier)]
|
Stabilize extended_key_value_attributes
# Stabilization report
## Summary
This stabilizes using macro expansion in key-value attributes, like so:
```rust
#[doc = include_str!("my_doc.md")]
struct S;
#[path = concat!(env!("OUT_DIR"), "/generated.rs")]
mod m;
```
See the changes to the reference for details on what macros are allowed;
see Petrochenkov's excellent blog post [on internals](https://internals.rust-lang.org/t/macro-expansion-points-in-attributes/11455)
for alternatives that were considered and rejected ("why accept no more
and no less?")
This has been available on nightly since 1.50 with no major issues.
## Notes
### Accepted syntax
The parser accepts arbitrary Rust expressions in this position, but any expression other than a macro invocation will ultimately lead to an error because it is not expected by the built-in expression forms (e.g., `#[doc]`). Note that decorators and the like may be able to observe other expression forms.
### Expansion ordering
Expansion of macro expressions in "inert" attributes occurs after decorators have executed, analogously to macro expressions appearing in the function body or other parts of decorator input.
There is currently no way for decorators to accept macros in key-value position if macro expansion must be performed before the decorator executes (if the macro can simply be copied into the output for later expansion, that can work).
## Test cases
- https://github.com/rust-lang/rust/blob/master/src/test/ui/attributes/key-value-expansion-on-mac.rs
- https://github.com/rust-lang/rust/blob/master/src/test/rustdoc/external-doc.rs
The feature has also been dogfooded extensively in the compiler and
standard library:
- https://github.com/rust-lang/rust/pull/83329
- https://github.com/rust-lang/rust/pull/83230
- https://github.com/rust-lang/rust/pull/82641
- https://github.com/rust-lang/rust/pull/80534
## Implementation history
- Initial proposal: https://github.com/rust-lang/rust/issues/55414#issuecomment-554005412
- Experiment to see how much code it would break: https://github.com/rust-lang/rust/pull/67121
- Preliminary work to restrict expansion that would conflict with this
feature: https://github.com/rust-lang/rust/pull/77271
- Initial implementation: https://github.com/rust-lang/rust/pull/78837
- Fix for an ICE: https://github.com/rust-lang/rust/pull/80563
## Unresolved Questions
~~https://github.com/rust-lang/rust/pull/83366#issuecomment-805180738 listed some concerns, but they have been resolved as of this final report.~~
## Additional Information
There are two workarounds that have a similar effect for `#[doc]`
attributes on nightly. One is to emulate this behavior by using a limited version of this feature that was stabilized for historical reasons:
```rust
macro_rules! forward_inner_docs {
($e:expr => $i:item) => {
#[doc = $e]
$i
};
}
forward_inner_docs!(include_str!("lib.rs") => struct S {});
```
This also works for other attributes (like `#[path = concat!(...)]`).
The other is to use `doc(include)`:
```rust
#![feature(external_doc)]
#[doc(include = "lib.rs")]
struct S {}
```
The first works, but is non-trivial for people to discover, and
difficult to read and maintain. The second is a strange special-case for
a particular use of the macro. This generalizes it to work for any use
case, not just including files.
I plan to remove `doc(include)` when this is stabilized. The
`forward_inner_docs` workaround will still compile without warnings, but
I expect it to be used less once it's no longer necessary.
2021-03-22 01:10:10 -04:00
|
|
|
#![cfg_attr(bootstrap, feature(extended_key_value_attributes))]
|
2018-06-27 13:12:47 +03:00
|
|
|
#![feature(extern_types)]
|
2018-06-27 17:57:25 +03:00
|
|
|
#![feature(in_band_lifetimes)]
|
2021-03-08 15:32:41 -08:00
|
|
|
#![feature(iter_zip)]
|
2018-09-26 14:26:46 -07:00
|
|
|
#![feature(nll)]
|
2020-01-05 02:10:23 +01:00
|
|
|
#![recursion_limit = "256"]
|
2014-11-15 20:30:33 -05:00
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
use back::write::{create_informational_target_machine, create_target_machine};
|
2016-07-21 12:49:59 -04:00
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
pub use llvm_util::target_features;
|
2020-02-29 20:37:32 +03:00
|
|
|
use rustc_ast::expand::allocator::AllocatorKind;
|
2019-12-22 17:42:04 -05:00
|
|
|
use rustc_codegen_ssa::back::lto::{LtoModuleCodegen, SerializedModule, ThinModule};
|
2020-09-23 16:57:50 +01:00
|
|
|
use rustc_codegen_ssa::back::write::{
|
2020-09-23 17:33:54 +01:00
|
|
|
CodegenContext, FatLTOInput, ModuleConfig, TargetMachineFactoryConfig, TargetMachineFactoryFn,
|
2020-09-23 16:57:50 +01:00
|
|
|
};
|
2018-11-16 13:45:28 +02:00
|
|
|
use rustc_codegen_ssa::traits::*;
|
2020-03-11 12:49:08 +01:00
|
|
|
use rustc_codegen_ssa::ModuleCodegen;
|
2020-01-29 00:16:14 +11:00
|
|
|
use rustc_codegen_ssa::{CodegenResults, CompiledModule};
|
2020-10-10 15:14:58 +02:00
|
|
|
use rustc_data_structures::fx::FxHashMap;
|
2020-03-31 21:18:30 +02:00
|
|
|
use rustc_errors::{ErrorReported, FatalError, Handler};
|
2020-10-10 15:14:58 +02:00
|
|
|
use rustc_middle::dep_graph::{WorkProduct, WorkProductId};
|
2020-03-29 17:19:48 +02:00
|
|
|
use rustc_middle::middle::cstore::{EncodedMetadata, MetadataLoaderDyn};
|
|
|
|
use rustc_middle::ty::{self, TyCtxt};
|
2020-10-10 16:18:36 +02:00
|
|
|
use rustc_session::config::{OptLevel, OutputFilenames, PrintRequest};
|
2020-03-11 12:49:08 +01:00
|
|
|
use rustc_session::Session;
|
|
|
|
use rustc_span::symbol::Symbol;
|
|
|
|
|
2017-07-23 08:14:38 -07:00
|
|
|
use std::any::Any;
|
2019-08-27 12:25:35 -07:00
|
|
|
use std::ffi::CStr;
|
2017-08-28 17:06:03 -07:00
|
|
|
|
2017-12-27 19:03:48 +01:00
|
|
|
mod back {
|
2019-03-30 15:30:07 +01:00
|
|
|
pub mod archive;
|
2018-08-17 16:07:23 +02:00
|
|
|
pub mod lto;
|
2020-02-11 22:37:16 +01:00
|
|
|
mod profiling;
|
2014-11-15 20:30:33 -05:00
|
|
|
pub mod write;
|
|
|
|
}
|
|
|
|
|
2016-03-22 19:23:36 +02:00
|
|
|
mod abi;
|
2017-06-03 14:54:08 -07:00
|
|
|
mod allocator;
|
2016-03-22 19:23:36 +02:00
|
|
|
mod asm;
|
|
|
|
mod attributes;
|
|
|
|
mod base;
|
|
|
|
mod builder;
|
|
|
|
mod callee;
|
|
|
|
mod common;
|
|
|
|
mod consts;
|
|
|
|
mod context;
|
2020-06-21 23:29:08 -07:00
|
|
|
mod coverageinfo;
|
2016-03-22 19:23:36 +02:00
|
|
|
mod debuginfo;
|
|
|
|
mod declare;
|
|
|
|
mod intrinsic;
|
2018-09-06 10:23:42 -07:00
|
|
|
|
|
|
|
// The following is a work around that replaces `pub mod llvm;` and that fixes issue 53912.
|
2019-12-22 17:42:04 -05:00
|
|
|
#[path = "llvm/mod.rs"]
|
|
|
|
mod llvm_;
|
|
|
|
pub mod llvm {
|
|
|
|
pub use super::llvm_::*;
|
|
|
|
}
|
2018-09-06 10:23:42 -07:00
|
|
|
|
2017-04-30 20:33:25 +02:00
|
|
|
mod llvm_util;
|
2018-05-08 16:10:16 +03:00
|
|
|
mod mono_item;
|
2016-03-22 19:23:36 +02:00
|
|
|
mod type_;
|
|
|
|
mod type_of;
|
2018-10-23 23:13:33 +00:00
|
|
|
mod va_arg;
|
2019-12-22 17:42:04 -05:00
|
|
|
mod value;
|
2016-03-22 19:23:36 +02:00
|
|
|
|
2018-10-23 17:01:35 +02:00
|
|
|
#[derive(Clone)]
|
2018-05-08 16:10:16 +03:00
|
|
|
pub struct LlvmCodegenBackend(());
|
2017-09-16 17:27:29 +02:00
|
|
|
|
2018-10-04 15:23:10 +02:00
|
|
|
impl ExtraBackendMethods for LlvmCodegenBackend {
|
2019-06-14 00:48:52 +03:00
|
|
|
fn new_metadata(&self, tcx: TyCtxt<'_>, mod_name: &str) -> ModuleLlvm {
|
2019-02-20 22:27:00 +02:00
|
|
|
ModuleLlvm::new_metadata(tcx, mod_name)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2019-02-20 22:27:00 +02:00
|
|
|
|
2019-06-14 00:48:52 +03:00
|
|
|
fn write_compressed_metadata<'tcx>(
|
2018-09-25 17:52:03 +02:00
|
|
|
&self,
|
2019-06-14 00:48:52 +03:00
|
|
|
tcx: TyCtxt<'tcx>,
|
2019-04-26 17:22:36 +10:00
|
|
|
metadata: &EncodedMetadata,
|
2019-06-12 00:11:55 +03:00
|
|
|
llvm_module: &mut ModuleLlvm,
|
2019-04-26 17:22:36 +10:00
|
|
|
) {
|
|
|
|
base::write_compressed_metadata(tcx, metadata, llvm_module)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2019-06-14 00:48:52 +03:00
|
|
|
fn codegen_allocator<'tcx>(
|
2019-02-25 08:40:18 +01:00
|
|
|
&self,
|
2019-06-14 00:48:52 +03:00
|
|
|
tcx: TyCtxt<'tcx>,
|
2019-02-25 08:40:18 +01:00
|
|
|
mods: &mut ModuleLlvm,
|
2019-06-12 00:11:55 +03:00
|
|
|
kind: AllocatorKind,
|
2020-09-07 10:45:20 +02:00
|
|
|
has_alloc_error_handler: bool,
|
2019-02-25 08:40:18 +01:00
|
|
|
) {
|
2020-09-07 10:45:20 +02:00
|
|
|
unsafe { allocator::codegen(tcx, mods, kind, has_alloc_error_handler) }
|
2018-10-23 17:01:35 +02:00
|
|
|
}
|
2019-09-25 13:14:43 -04:00
|
|
|
fn compile_codegen_unit(
|
2019-12-22 17:42:04 -05:00
|
|
|
&self,
|
|
|
|
tcx: TyCtxt<'_>,
|
2019-10-21 17:14:03 +11:00
|
|
|
cgu_name: Symbol,
|
2020-01-05 02:10:23 +01:00
|
|
|
) -> (ModuleCodegen<ModuleLlvm>, u64) {
|
|
|
|
base::compile_codegen_unit(tcx, cgu_name)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
fn target_machine_factory(
|
2018-09-25 17:52:03 +02:00
|
|
|
&self,
|
2018-10-23 17:01:35 +02:00
|
|
|
sess: &Session,
|
2018-10-27 15:29:06 +03:00
|
|
|
optlvl: OptLevel,
|
2020-09-23 16:57:50 +01:00
|
|
|
) -> TargetMachineFactoryFn<Self> {
|
2020-04-23 21:10:01 +03:00
|
|
|
back::write::target_machine_factory(sess, optlvl)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
fn target_cpu<'b>(&self, sess: &'b Session) -> &'b str {
|
|
|
|
llvm_util::target_cpu(sess)
|
2018-10-03 13:49:57 +02:00
|
|
|
}
|
2020-09-17 17:39:26 +08:00
|
|
|
fn tune_cpu<'b>(&self, sess: &'b Session) -> Option<&'b str> {
|
|
|
|
llvm_util::tune_cpu(sess)
|
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl WriteBackendMethods for LlvmCodegenBackend {
|
|
|
|
type Module = ModuleLlvm;
|
|
|
|
type ModuleBuffer = back::lto::ModuleBuffer;
|
|
|
|
type Context = llvm::Context;
|
|
|
|
type TargetMachine = &'static mut llvm::TargetMachine;
|
|
|
|
type ThinData = back::lto::ThinData;
|
|
|
|
type ThinBuffer = back::lto::ThinBuffer;
|
|
|
|
fn print_pass_timings(&self) {
|
2019-12-22 17:42:04 -05:00
|
|
|
unsafe {
|
|
|
|
llvm::LLVMRustPrintPassTimings();
|
|
|
|
}
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2020-09-09 14:51:16 +10:00
|
|
|
fn run_link(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
|
|
|
diag_handler: &Handler,
|
|
|
|
modules: Vec<ModuleCodegen<Self::Module>>,
|
|
|
|
) -> Result<ModuleCodegen<Self::Module>, FatalError> {
|
|
|
|
back::write::link(cgcx, diag_handler, modules)
|
|
|
|
}
|
2018-12-03 20:45:03 +01:00
|
|
|
fn run_fat_lto(
|
2018-10-23 17:01:35 +02:00
|
|
|
cgcx: &CodegenContext<Self>,
|
2019-02-11 07:46:04 -08:00
|
|
|
modules: Vec<FatLTOInput<Self>>,
|
|
|
|
cached_modules: Vec<(SerializedModule<Self::ModuleBuffer>, WorkProduct)>,
|
2018-12-03 20:50:39 +01:00
|
|
|
) -> Result<LtoModuleCodegen<Self>, FatalError> {
|
2019-02-13 08:13:30 -05:00
|
|
|
back::lto::run_fat(cgcx, modules, cached_modules)
|
2018-12-03 20:45:03 +01:00
|
|
|
}
|
|
|
|
fn run_thin_lto(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
2018-12-04 16:24:20 +01:00
|
|
|
modules: Vec<(String, Self::ThinBuffer)>,
|
2018-12-03 20:45:03 +01:00
|
|
|
cached_modules: Vec<(SerializedModule<Self::ModuleBuffer>, WorkProduct)>,
|
|
|
|
) -> Result<(Vec<LtoModuleCodegen<Self>>, Vec<WorkProduct>), FatalError> {
|
2019-02-13 08:13:30 -05:00
|
|
|
back::lto::run_thin(cgcx, modules, cached_modules)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
unsafe fn optimize(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
|
|
|
diag_handler: &Handler,
|
|
|
|
module: &ModuleCodegen<Self::Module>,
|
|
|
|
config: &ModuleConfig,
|
|
|
|
) -> Result<(), FatalError> {
|
2021-04-05 15:37:11 +02:00
|
|
|
back::write::optimize(cgcx, diag_handler, module, config)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
unsafe fn optimize_thin(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
|
|
|
thin: &mut ThinModule<Self>,
|
|
|
|
) -> Result<ModuleCodegen<Self::Module>, FatalError> {
|
2019-02-13 08:13:30 -05:00
|
|
|
back::lto::optimize_thin_module(thin, cgcx)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
unsafe fn codegen(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
|
|
|
diag_handler: &Handler,
|
|
|
|
module: ModuleCodegen<Self::Module>,
|
|
|
|
config: &ModuleConfig,
|
|
|
|
) -> Result<CompiledModule, FatalError> {
|
2019-02-13 08:13:30 -05:00
|
|
|
back::write::codegen(cgcx, diag_handler, module, config)
|
2018-09-25 17:52:03 +02:00
|
|
|
}
|
2019-12-22 17:42:04 -05:00
|
|
|
fn prepare_thin(module: ModuleCodegen<Self::Module>) -> (String, Self::ThinBuffer) {
|
2019-02-11 07:46:04 -08:00
|
|
|
back::lto::prepare_thin(module)
|
|
|
|
}
|
2019-12-22 17:42:04 -05:00
|
|
|
fn serialize_module(module: ModuleCodegen<Self::Module>) -> (String, Self::ModuleBuffer) {
|
2019-02-11 07:46:04 -08:00
|
|
|
(module.name, back::lto::ModuleBuffer::new(module.module_llvm.llmod()))
|
2018-12-04 16:24:20 +01:00
|
|
|
}
|
2018-10-23 17:01:35 +02:00
|
|
|
fn run_lto_pass_manager(
|
|
|
|
cgcx: &CodegenContext<Self>,
|
|
|
|
module: &ModuleCodegen<Self::Module>,
|
|
|
|
config: &ModuleConfig,
|
2019-12-22 17:42:04 -05:00
|
|
|
thin: bool,
|
2021-04-05 15:37:11 +02:00
|
|
|
) -> Result<(), FatalError> {
|
|
|
|
let diag_handler = cgcx.create_diag_handler();
|
|
|
|
back::lto::run_pass_manager(cgcx, &diag_handler, module, config, thin)
|
2018-09-26 17:00:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-16 13:37:55 +02:00
|
|
|
unsafe impl Send for LlvmCodegenBackend {} // Llvm is on a per-thread basis
|
|
|
|
unsafe impl Sync for LlvmCodegenBackend {}
|
2017-10-30 18:42:21 +01:00
|
|
|
|
2018-05-08 16:10:16 +03:00
|
|
|
impl LlvmCodegenBackend {
|
2018-07-11 12:49:11 +02:00
|
|
|
pub fn new() -> Box<dyn CodegenBackend> {
|
2019-11-17 18:54:13 +01:00
|
|
|
Box::new(LlvmCodegenBackend(()))
|
2017-09-16 17:27:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-16 13:37:55 +02:00
|
|
|
impl CodegenBackend for LlvmCodegenBackend {
|
2018-01-22 07:29:24 -08:00
|
|
|
fn init(&self, sess: &Session) {
|
|
|
|
llvm_util::init(sess); // Make sure llvm is inited
|
|
|
|
}
|
|
|
|
|
2017-10-30 18:42:21 +01:00
|
|
|
fn print(&self, req: PrintRequest, sess: &Session) {
|
|
|
|
match req {
|
|
|
|
PrintRequest::RelocationModels => {
|
|
|
|
println!("Available relocation models:");
|
2020-04-23 00:46:45 +03:00
|
|
|
for name in
|
|
|
|
&["static", "pic", "dynamic-no-pic", "ropi", "rwpi", "ropi-rwpi", "default"]
|
|
|
|
{
|
2017-10-30 18:42:21 +01:00
|
|
|
println!(" {}", name);
|
|
|
|
}
|
2019-09-13 17:36:35 +08:00
|
|
|
println!();
|
2017-10-30 18:42:21 +01:00
|
|
|
}
|
|
|
|
PrintRequest::CodeModels => {
|
|
|
|
println!("Available code models:");
|
2020-05-20 23:09:19 +03:00
|
|
|
for name in &["tiny", "small", "kernel", "medium", "large"] {
|
2017-10-30 18:42:21 +01:00
|
|
|
println!(" {}", name);
|
|
|
|
}
|
2019-09-13 17:36:35 +08:00
|
|
|
println!();
|
2017-10-30 18:42:21 +01:00
|
|
|
}
|
|
|
|
PrintRequest::TlsModels => {
|
|
|
|
println!("Available TLS models:");
|
2020-04-25 21:45:21 +03:00
|
|
|
for name in &["global-dynamic", "local-dynamic", "initial-exec", "local-exec"] {
|
2017-10-30 18:42:21 +01:00
|
|
|
println!(" {}", name);
|
|
|
|
}
|
2019-09-13 17:36:35 +08:00
|
|
|
println!();
|
2017-10-30 18:42:21 +01:00
|
|
|
}
|
|
|
|
req => llvm_util::print(req, sess),
|
|
|
|
}
|
|
|
|
}
|
2017-09-16 17:27:29 +02:00
|
|
|
|
2018-01-22 07:29:24 -08:00
|
|
|
fn print_passes(&self) {
|
|
|
|
llvm_util::print_passes();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_version(&self) {
|
|
|
|
llvm_util::print_version();
|
|
|
|
}
|
|
|
|
|
2017-10-30 18:42:21 +01:00
|
|
|
fn target_features(&self, sess: &Session) -> Vec<Symbol> {
|
|
|
|
target_features(sess)
|
|
|
|
}
|
|
|
|
|
2019-06-12 15:59:10 +03:00
|
|
|
fn codegen_crate<'tcx>(
|
2017-10-30 18:42:21 +01:00
|
|
|
&self,
|
2019-06-14 00:48:52 +03:00
|
|
|
tcx: TyCtxt<'tcx>,
|
2019-04-26 17:22:36 +10:00
|
|
|
metadata: EncodedMetadata,
|
|
|
|
need_metadata_module: bool,
|
2018-07-11 12:49:11 +02:00
|
|
|
) -> Box<dyn Any> {
|
2019-11-17 18:54:13 +01:00
|
|
|
Box::new(rustc_codegen_ssa::base::codegen_crate(
|
2019-12-22 17:42:04 -05:00
|
|
|
LlvmCodegenBackend(()),
|
|
|
|
tcx,
|
2021-03-28 22:14:09 +02:00
|
|
|
crate::llvm_util::target_cpu(tcx.sess).to_string(),
|
2019-12-22 17:42:04 -05:00
|
|
|
metadata,
|
|
|
|
need_metadata_module,
|
2019-11-17 18:54:13 +01:00
|
|
|
))
|
2017-09-16 17:27:29 +02:00
|
|
|
}
|
|
|
|
|
2020-01-29 00:16:14 +11:00
|
|
|
fn join_codegen(
|
2017-10-30 18:42:21 +01:00
|
|
|
&self,
|
2018-07-11 12:49:11 +02:00
|
|
|
ongoing_codegen: Box<dyn Any>,
|
2017-09-16 17:27:29 +02:00
|
|
|
sess: &Session,
|
2020-10-10 16:18:36 +02:00
|
|
|
) -> Result<(CodegenResults, FxHashMap<WorkProductId, WorkProduct>), ErrorReported> {
|
2019-12-22 17:42:04 -05:00
|
|
|
let (codegen_results, work_products) = ongoing_codegen
|
|
|
|
.downcast::<rustc_codegen_ssa::back::write::OngoingCodegen<LlvmCodegenBackend>>()
|
|
|
|
.expect("Expected LlvmCodegenBackend's OngoingCodegen, found Box<Any>")
|
|
|
|
.join(sess);
|
2017-09-16 17:27:29 +02:00
|
|
|
|
2020-10-10 16:20:42 +02:00
|
|
|
sess.time("llvm_dump_timing_file", || {
|
|
|
|
if sess.opts.debugging_opts.llvm_time_trace {
|
|
|
|
llvm_util::time_trace_profiler_finish("llvm_timings.json");
|
|
|
|
}
|
2019-12-22 17:42:04 -05:00
|
|
|
});
|
2017-10-30 18:42:21 +01:00
|
|
|
|
2020-10-10 16:18:36 +02:00
|
|
|
Ok((codegen_results, work_products))
|
2020-01-29 00:16:14 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn link(
|
|
|
|
&self,
|
|
|
|
sess: &Session,
|
2020-10-10 16:18:36 +02:00
|
|
|
codegen_results: CodegenResults,
|
2020-01-29 00:16:14 +11:00
|
|
|
outputs: &OutputFilenames,
|
|
|
|
) -> Result<(), ErrorReported> {
|
2020-12-03 14:11:35 +01:00
|
|
|
use crate::back::archive::LlvmArchiveBuilder;
|
|
|
|
use rustc_codegen_ssa::back::link::link_binary;
|
|
|
|
|
2017-10-30 18:42:21 +01:00
|
|
|
// Run the linker on any artifacts that resulted from the LLVM run.
|
|
|
|
// This should produce either a finished executable or library.
|
2020-12-03 14:11:35 +01:00
|
|
|
link_binary::<LlvmArchiveBuilder<'_>>(
|
|
|
|
sess,
|
|
|
|
&codegen_results,
|
|
|
|
outputs,
|
|
|
|
&codegen_results.crate_name.as_str(),
|
|
|
|
);
|
2017-10-30 18:42:21 +01:00
|
|
|
|
|
|
|
Ok(())
|
2017-09-16 17:27:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-25 17:52:03 +02:00
|
|
|
pub struct ModuleLlvm {
|
2018-06-27 17:57:25 +03:00
|
|
|
llcx: &'static mut llvm::Context,
|
|
|
|
llmod_raw: *const llvm::Module,
|
|
|
|
tm: &'static mut llvm::TargetMachine,
|
2016-03-22 19:23:36 +02:00
|
|
|
}
|
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
unsafe impl Send for ModuleLlvm {}
|
|
|
|
unsafe impl Sync for ModuleLlvm {}
|
2017-07-23 08:14:38 -07:00
|
|
|
|
2018-09-06 16:04:20 -07:00
|
|
|
impl ModuleLlvm {
|
2019-06-14 00:48:52 +03:00
|
|
|
fn new(tcx: TyCtxt<'_>, mod_name: &str) -> Self {
|
2018-06-27 17:57:25 +03:00
|
|
|
unsafe {
|
2018-10-27 15:29:06 +03:00
|
|
|
let llcx = llvm::LLVMRustContextCreate(tcx.sess.fewer_names());
|
|
|
|
let llmod_raw = context::create_module(tcx, llcx, mod_name) as *const _;
|
2020-09-23 17:33:54 +01:00
|
|
|
ModuleLlvm { llmod_raw, llcx, tm: create_target_machine(tcx, mod_name) }
|
2018-06-27 17:57:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-14 00:48:52 +03:00
|
|
|
fn new_metadata(tcx: TyCtxt<'_>, mod_name: &str) -> Self {
|
2019-02-20 22:27:00 +02:00
|
|
|
unsafe {
|
|
|
|
let llcx = llvm::LLVMRustContextCreate(tcx.sess.fewer_names());
|
|
|
|
let llmod_raw = context::create_module(tcx, llcx, mod_name) as *const _;
|
2020-04-23 21:10:01 +03:00
|
|
|
ModuleLlvm { llmod_raw, llcx, tm: create_informational_target_machine(tcx.sess) }
|
2019-02-20 22:27:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-11 07:46:04 -08:00
|
|
|
fn parse(
|
|
|
|
cgcx: &CodegenContext<LlvmCodegenBackend>,
|
2019-08-27 12:25:35 -07:00
|
|
|
name: &CStr,
|
|
|
|
buffer: &[u8],
|
2019-02-11 07:46:04 -08:00
|
|
|
handler: &Handler,
|
|
|
|
) -> Result<Self, FatalError> {
|
|
|
|
unsafe {
|
|
|
|
let llcx = llvm::LLVMRustContextCreate(cgcx.fewer_names);
|
2019-08-27 12:25:35 -07:00
|
|
|
let llmod_raw = back::lto::parse_module(llcx, name, buffer, handler)?;
|
2020-11-30 08:39:08 -08:00
|
|
|
let tm_factory_config = TargetMachineFactoryConfig::new(&cgcx, name.to_str().unwrap());
|
2020-09-23 17:33:54 +01:00
|
|
|
let tm = match (cgcx.tm_factory)(tm_factory_config) {
|
2019-02-11 07:46:04 -08:00
|
|
|
Ok(m) => m,
|
|
|
|
Err(e) => {
|
|
|
|
handler.struct_err(&e).emit();
|
2019-12-22 17:42:04 -05:00
|
|
|
return Err(FatalError);
|
2019-02-11 07:46:04 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
Ok(ModuleLlvm { llmod_raw, llcx, tm })
|
2019-02-11 07:46:04 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-27 17:57:25 +03:00
|
|
|
fn llmod(&self) -> &llvm::Module {
|
2019-12-22 17:42:04 -05:00
|
|
|
unsafe { &*self.llmod_raw }
|
2018-06-27 17:57:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-06 16:04:20 -07:00
|
|
|
impl Drop for ModuleLlvm {
|
2017-07-23 08:14:38 -07:00
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
2018-06-27 17:57:25 +03:00
|
|
|
llvm::LLVMContextDispose(&mut *(self.llcx as *mut _));
|
|
|
|
llvm::LLVMRustDisposeTargetMachine(&mut *(self.tm as *mut _));
|
2017-07-23 08:14:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|