Skip to content

Commit

Permalink
Driver provided "intrinsics".
Browse files Browse the repository at this point in the history
Allows drivers to define functions which are only expanded by the plugin
after all type checking, inference, etc etc, ie only once during mono
item collection.

The form of the interface with the plugins is slightly different than
what was proposed in rust-lang#51623.
Additionally, signature checking is added.
  • Loading branch information
DiamondLovesYou committed Oct 3, 2020
1 parent b53cd5c commit 2621a36
Show file tree
Hide file tree
Showing 23 changed files with 473 additions and 19 deletions.
10 changes: 8 additions & 2 deletions compiler/rustc_codegen_ssa/src/mir/block.rs
Original file line number Diff line number Diff line change
Expand Up @@ -624,9 +624,14 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
let arg_count = fn_abi.args.len() + fn_abi.ret.is_indirect() as usize;
let mut llargs = Vec::with_capacity(arg_count);

// Custom intrinsics are treated as-if they were normal functions here.
let is_custom_intrinsic = intrinsic.and_then(|_| instance )
.map(|instance| bx.tcx().custom_intrinsic_mir(instance).is_some() )
.unwrap_or_default();

// Prepare the return value destination
let ret_dest = if let Some((dest, _)) = *destination {
let is_intrinsic = intrinsic.is_some();
let is_intrinsic = intrinsic.is_some() && !is_custom_intrinsic;
self.make_return_dest(&mut bx, dest, &fn_abi.ret, &mut llargs, is_intrinsic)
} else {
ReturnDest::Nothing
Expand All @@ -647,7 +652,8 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
return;
}

if intrinsic.is_some() && intrinsic != Some(sym::drop_in_place) {
if intrinsic.is_some() && intrinsic != Some(sym::drop_in_place) &&
!is_custom_intrinsic {
let intrinsic = intrinsic.unwrap();
let dest = match ret_dest {
_ if fn_abi.ret.is_indirect() => llargs[0],
Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_codegen_ssa/src/mir/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ pub fn codegen_mir<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(

let llfn = cx.get_fn(instance);

let mir = cx.tcx().instance_mir(instance.def);
let mir = cx.tcx().instance_mir(instance);

let fn_abi = FnAbi::of_instance(cx, instance, &[]);
debug!("fn_abi: {:?}", fn_abi);
Expand Down
53 changes: 53 additions & 0 deletions compiler/rustc_data_structures/src/sync.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ use crate::owning_ref::{Erased, OwningRef};
use std::collections::HashMap;
use std::hash::{BuildHasher, Hash};
use std::ops::{Deref, DerefMut};
use std::sync::{Arc, atomic::AtomicPtr, atomic, };

pub use std::sync::atomic::Ordering;
pub use std::sync::atomic::Ordering::SeqCst;
Expand Down Expand Up @@ -656,3 +657,55 @@ impl<T> DerefMut for OneThread<T> {
&mut self.inner
}
}

/// Provides atomic mutability by replacing the value inside this `ArcCell`.
/// Similar to the `crossbeam` structure of the same name.
#[derive(Debug)]
pub struct ArcCell<T>(AtomicPtr<T>);
impl<T> ArcCell<T> {
pub fn new(v: Arc<T>) -> Self {
ArcCell(AtomicPtr::new(Arc::into_raw(v) as *mut _))
}
pub fn get(&self) -> Arc<T> {
let ptr = self.0.load(atomic::Ordering::Acquire);
let arc = unsafe { Arc::from_raw(ptr as *const T) };
let ret = arc.clone();
// don't drop our copy:
::std::mem::forget(arc);
ret
}
/// Update the value, returning the previous value.
pub fn set(&self, v: Arc<T>) -> Arc<T> {
let new = Arc::into_raw(v) as *mut _;
let mut expected = self.0.load(atomic::Ordering::Acquire);
loop {
match self.0.compare_exchange_weak(expected, new,
atomic::Ordering::SeqCst,
atomic::Ordering::Acquire) {
Ok(old) => {
return unsafe { Arc::from_raw(old as *const T) };
},
Err(v) => {
expected = v;
},
}
}
}
}
impl<T> Drop for ArcCell<T> {
fn drop(&mut self) {
let ptr = self.0.load(atomic::Ordering::Acquire);
// drop our copy of the arc:
unsafe { Arc::from_raw(ptr as *const _) };
}
}
impl<T> Clone for ArcCell<T> {
fn clone(&self) -> Self {
ArcCell::new(self.get())
}
}
impl<T> From<Arc<T>> for ArcCell<T> {
fn from(v: Arc<T>) -> Self {
Self::new(v)
}
}
32 changes: 31 additions & 1 deletion compiler/rustc_middle/src/mir/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ mod type_foldable;
pub mod visit;

/// Types for locals
type LocalDecls<'tcx> = IndexVec<Local, LocalDecl<'tcx>>;
pub type LocalDecls<'tcx> = IndexVec<Local, LocalDecl<'tcx>>;

pub trait HasLocalDecls<'tcx> {
fn local_decls(&self) -> &LocalDecls<'tcx>;
Expand Down Expand Up @@ -2674,3 +2674,33 @@ impl Location {
}
}
}

pub trait CustomIntrinsicMirGen: Sync + Send {
/// Codegen a plugin-defined intrinsic. This is intended to be used to
/// "return" values based on the monomorphized and erased types of the
/// function call. Codegen will codegen the `extra_stmts` and then insert
/// an unconditional branch to the exit block.
///
/// Consider this to be highly unstable; it will likely change without
/// warning. There is also no spec for this, it is 100% implementation
/// defined, and may not be implemented at all for some codegen backends.
///
/// If the codegen backend is multithreaded, this will be called from
/// any number of threads, hence `Sync + Send`.
///
/// YOU ARE RESPONSIBLE FOR THE SAFETY OF THE EXTRA STATEMENTS.
/// You have been warned. Good luck, have fun.
fn mirgen_simple_intrinsic<'tcx>(&self,
tcx: TyCtxt<'tcx>,
instance: ty::Instance<'tcx>,
mir: &mut Body<'tcx>);

/// The following are used for typeck-ing:
/// The number of generic parameters expected.
fn generic_parameter_count<'tcx>(&self, tcx: TyCtxt<'tcx>) -> usize;
/// The types of the input args.
fn inputs<'tcx>(&self, tcx: TyCtxt<'tcx>) -> &'tcx List<Ty<'tcx>>;
/// The return type.
fn output<'tcx>(&self, tcx: TyCtxt<'tcx>) -> Ty<'tcx>;
}
15 changes: 15 additions & 0 deletions compiler/rustc_middle/src/query/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -306,6 +306,21 @@ rustc_queries! {
}
}

/// If defined by the driver, returns the extra mir statements to codegen,
/// else returns `None`.
query custom_intrinsic_mirgen(key: DefId) -> Option<Lrc<dyn mir::CustomIntrinsicMirGen>> {
anon
no_hash

desc { |tcx| "asking for the custom MIR generator of `{}`", tcx.def_path_str(key) }
}
/// The monomorphized MIR for a custom intrinsic instance.
query custom_intrinsic_mir(inst: ty::Instance<'tcx>) -> Option<&'tcx mir::Body<'tcx>> {
anon

desc { |tcx| "asking for the custom MIR of `{}`", tcx.def_path_str(inst.def_id()) }
}

/// Returns coverage summary info for a function, after executing the `InstrumentCoverage`
/// MIR pass (assuming the -Zinstrument-coverage option is enabled).
query coverageinfo(key: DefId) -> mir::CoverageInfo {
Expand Down
14 changes: 10 additions & 4 deletions compiler/rustc_middle/src/ty/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2951,23 +2951,29 @@ impl<'tcx> TyCtxt<'tcx> {
}

/// Returns the possibly-auto-generated MIR of a `(DefId, Subst)` pair.
pub fn instance_mir(self, instance: ty::InstanceDef<'tcx>) -> &'tcx Body<'tcx> {
match instance {
pub fn instance_mir(self, instance: ty::Instance<'tcx>) -> &'tcx Body<'tcx> {
match instance.def {
ty::InstanceDef::Item(def) => {
if let Some((did, param_did)) = def.as_const_arg() {
self.optimized_mir_of_const_arg((did, param_did))
} else {
self.optimized_mir(def.did)
}
}
ty::InstanceDef::Intrinsic(..) => {
if let Some(mir) = self.custom_intrinsic_mir(instance) {
mir
} else {
self.mir_shims(instance.def)
}
}
ty::InstanceDef::VtableShim(..)
| ty::InstanceDef::ReifyShim(..)
| ty::InstanceDef::Intrinsic(..)
| ty::InstanceDef::FnPtrShim(..)
| ty::InstanceDef::Virtual(..)
| ty::InstanceDef::ClosureOnceShim { .. }
| ty::InstanceDef::DropGlue(..)
| ty::InstanceDef::CloneShim(..) => self.mir_shims(instance),
| ty::InstanceDef::CloneShim(..) => self.mir_shims(instance.def),
}
}

Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_mir/src/const_eval/eval_queries.rs
Original file line number Diff line number Diff line change
Expand Up @@ -293,7 +293,7 @@ pub fn eval_to_allocation_raw_provider<'tcx>(
MemoryExtra { can_access_statics: is_static },
);

let res = ecx.load_mir(cid.instance.def, cid.promoted);
let res = ecx.load_mir(cid.instance, cid.promoted);
match res.and_then(|body| eval_body_using_ecx(&mut ecx, cid, &body)) {
Err(error) => {
let err = ConstEvalErr::new(&ecx, error, None);
Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_mir/src/const_eval/machine.rs
Original file line number Diff line number Diff line change
Expand Up @@ -243,7 +243,7 @@ impl<'mir, 'tcx> interpret::Machine<'mir, 'tcx> for CompileTimeInterpreter<'mir,
}
}
// This is a const fn. Call it.
Ok(Some(match ecx.load_mir(instance.def, None) {
Ok(Some(match ecx.load_mir(instance, None) {
Ok(body) => body,
Err(err) => {
if let err_unsup!(NoMirFor(did)) = err.kind {
Expand Down
6 changes: 3 additions & 3 deletions compiler/rustc_mir/src/interpret/eval_context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -424,11 +424,11 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {

pub fn load_mir(
&self,
instance: ty::InstanceDef<'tcx>,
instance: ty::Instance<'tcx>,
promoted: Option<mir::Promoted>,
) -> InterpResult<'tcx, &'tcx mir::Body<'tcx>> {
// do not continue if typeck errors occurred (can only occur in local crate)
let def = instance.with_opt_param();
let def = instance.def.with_opt_param();
if let Some(def) = def.as_local() {
if self.tcx.has_typeck_results(def.did) {
if let Some(error_reported) = self.tcx.typeck_opt_const_arg(def).tainted_by_errors {
Expand All @@ -440,7 +440,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
if let Some(promoted) = promoted {
return Ok(&self.tcx.promoted_mir_of_opt_const_arg(def)[promoted]);
}
match instance {
match instance.def {
ty::InstanceDef::Item(def) => {
if self.tcx.is_mir_available(def.did) {
if let Some((did, param_did)) = def.as_const_arg() {
Expand Down
3 changes: 3 additions & 0 deletions compiler/rustc_mir/src/interpret/terminator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,8 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
}
};

let custom = self.tcx.custom_intrinsic_mir(instance);

// ABI check
{
let callee_abi = {
Expand Down Expand Up @@ -301,6 +303,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
// The Rust ABI is special: ZST get skipped.
let rust_abi = match caller_abi {
Abi::Rust | Abi::RustCall => true,
Abi::RustIntrinsic if custom.is_some() => true,
_ => false,
};
// We have two iterators: Where the arguments come from,
Expand Down
1 change: 1 addition & 0 deletions compiler/rustc_mir/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@ pub fn provide(providers: &mut Providers) {
const_eval::provide(providers);
shim::provide(providers);
transform::provide(providers);
monomorphize::provide(providers);
monomorphize::partitioning::provide(providers);
monomorphize::polymorphize::provide(providers);
providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider;
Expand Down
6 changes: 5 additions & 1 deletion compiler/rustc_mir/src/monomorphize/collector.rs
Original file line number Diff line number Diff line change
Expand Up @@ -769,6 +769,10 @@ fn visit_instance_use<'tcx>(
if !is_direct_call {
bug!("{:?} being reified", instance);
}

if let Some(_mir) = tcx.custom_intrinsic_mir(instance) {
output.push(create_fn_mono_item(tcx, instance, source));
}
}
ty::InstanceDef::DropGlue(_, None) => {
// Don't need to emit noop drop glue if we are calling directly.
Expand Down Expand Up @@ -1237,7 +1241,7 @@ fn collect_neighbours<'tcx>(
output: &mut Vec<Spanned<MonoItem<'tcx>>>,
) {
debug!("collect_neighbours: {:?}", instance.def_id());
let body = tcx.instance_mir(instance.def);
let body = tcx.instance_mir(instance);

MirNeighborCollector { tcx, body: &body, output, instance }.visit_body(&body);
}
Expand Down
70 changes: 68 additions & 2 deletions compiler/rustc_mir/src/monomorphize/mod.rs
Original file line number Diff line number Diff line change
@@ -1,13 +1,79 @@
use rustc_middle::traits;
use rustc_index::vec::IndexVec;
use rustc_middle::{traits, mir};
use rustc_middle::ty::adjustment::CustomCoerceUnsized;
use rustc_middle::ty::{self, Ty, TyCtxt};
use rustc_middle::ty::{self, Ty, TyCtxt, Instance};
use rustc_middle::ty::query::Providers;
use rustc_span::DUMMY_SP;

use rustc_hir::lang_items::LangItem;

pub mod collector;
pub mod partitioning;
pub mod polymorphize;

pub fn provide(providers: &mut Providers) {
providers.custom_intrinsic_mirgen = |_, _| { None };
providers.custom_intrinsic_mir = custom_intrinsic_mir;
}

fn custom_intrinsic_mir<'tcx>(tcx: TyCtxt<'tcx>,
instance: Instance<'tcx>)
-> Option<&'tcx mir::Body<'tcx>>
{
let mirgen = tcx.custom_intrinsic_mirgen(instance.def_id())?;

let ty = instance.ty(tcx, ty::ParamEnv::reveal_all());
let sig = ty.fn_sig(tcx);
let sig = tcx.normalize_erasing_late_bound_regions(
ty::ParamEnv::reveal_all(),
&sig,
);

// no var arg calls, so we can skip monomorphizing extra arguments.
assert!(!sig.c_variadic);

let source_scope_local_data = mir::ClearCrossCrate::Clear;
let source_scope = mir::SourceScopeData {
span: DUMMY_SP,
parent_scope: None,
local_data: source_scope_local_data,
};
let source_info = mir::SourceInfo {
span: DUMMY_SP,
scope: mir::OUTERMOST_SOURCE_SCOPE,
};

let mut source_scopes = IndexVec::new();
source_scopes.push(source_scope.clone());

let ret_decl = mir::LocalDecl::new(sig.output(), DUMMY_SP);
let mut local_decls = IndexVec::from_elem_n(ret_decl, 1);
for &arg in sig.inputs().iter() {
local_decls.push(mir::LocalDecl {
mutability: mir::Mutability::Mut,
local_info: None,
ty: arg,
source_info,
internal: false,
user_ty: None,
is_block_tail: None,
});
}

let mut gen = mir::Body::new(IndexVec::new(),
source_scopes,
local_decls,
Default::default(),
sig.inputs().len(),
Vec::new(),
source_scope.span,
None);

mirgen.mirgen_simple_intrinsic(tcx, instance, &mut gen);

Some(tcx.arena.alloc(gen))
}

pub fn custom_coerce_unsize_info<'tcx>(
tcx: TyCtxt<'tcx>,
source_ty: Ty<'tcx>,
Expand Down
4 changes: 4 additions & 0 deletions compiler/rustc_symbol_mangling/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -178,6 +178,10 @@ fn compute_symbol_name(
Node::ForeignItem(_) => true,
_ => false,
}
} else if let ty::InstanceDef::Intrinsic(_) = instance.def {
// custom intrinsics should never be foreign, otherwise
// generic parameters will cause duplicate symbols names.
tcx.custom_intrinsic_mir(instance).is_none()
} else {
tcx.is_foreign_item(def_id)
};
Expand Down
5 changes: 4 additions & 1 deletion compiler/rustc_ty/src/ty.rs
Original file line number Diff line number Diff line change
Expand Up @@ -423,7 +423,10 @@ fn instance_def_size_estimate<'tcx>(

match instance_def {
InstanceDef::Item(..) | InstanceDef::DropGlue(..) => {
let mir = tcx.instance_mir(instance_def);
let mir = tcx.instance_mir(ty::Instance {
def: instance_def,
substs: tcx.intern_substs(&[]),
});
mir.basic_blocks().iter().map(|bb| bb.statements.len()).sum()
}
// Estimate the size of other compiler-generated shims to be 1.
Expand Down
Loading

0 comments on commit 2621a36

Please sign in to comment.