2015-07-16 16:46:36 -07:00
|
|
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
#![cfg_attr(stage0, feature(custom_attribute))]
|
|
|
|
#![crate_name = "rustc_platform_intrinsics"]
|
|
|
|
#![unstable(feature = "rustc_private", issue = "27812")]
|
|
|
|
#![staged_api]
|
|
|
|
#![crate_type = "dylib"]
|
|
|
|
#![crate_type = "rlib"]
|
|
|
|
#![feature(staged_api, rustc_private)]
|
|
|
|
|
|
|
|
extern crate rustc_llvm as llvm;
|
|
|
|
extern crate rustc;
|
|
|
|
|
|
|
|
use rustc::middle::ty;
|
|
|
|
|
|
|
|
pub struct Intrinsic {
|
|
|
|
pub inputs: Vec<Type>,
|
|
|
|
pub output: Type,
|
|
|
|
|
|
|
|
pub definition: IntrinsicDef,
|
|
|
|
}
|
|
|
|
|
2015-08-05 09:08:03 -07:00
|
|
|
#[derive(Clone, Hash, Eq, PartialEq)]
|
2015-07-16 16:46:36 -07:00
|
|
|
pub enum Type {
|
2015-08-29 20:01:33 -07:00
|
|
|
Integer(/* signed */ bool, u8, /* llvm width */ u8),
|
2015-07-16 16:46:36 -07:00
|
|
|
Float(u8),
|
2015-09-02 13:59:35 -07:00
|
|
|
Pointer(Box<Type>, Option<Box<Type>>, /* const */ bool),
|
2015-07-16 16:46:36 -07:00
|
|
|
Vector(Box<Type>, u8),
|
2015-08-28 16:34:07 -07:00
|
|
|
Aggregate(bool, Vec<Type>),
|
2015-07-16 16:46:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub enum IntrinsicDef {
|
|
|
|
Named(&'static str),
|
|
|
|
}
|
|
|
|
|
2015-08-29 20:01:33 -07:00
|
|
|
fn i(width: u8) -> Type { Type::Integer(true, width, width) }
|
|
|
|
fn i_(width: u8, llvm_width: u8) -> Type { Type::Integer(true, width, llvm_width) }
|
|
|
|
fn u(width: u8) -> Type { Type::Integer(false, width, width) }
|
|
|
|
#[allow(dead_code)]
|
|
|
|
fn u_(width: u8, llvm_width: u8) -> Type { Type::Integer(false, width, llvm_width) }
|
2015-07-16 16:46:36 -07:00
|
|
|
fn f(width: u8) -> Type { Type::Float(width) }
|
|
|
|
fn v(x: Type, length: u8) -> Type { Type::Vector(Box::new(x), length) }
|
2015-08-28 16:34:07 -07:00
|
|
|
fn agg(flatten: bool, types: Vec<Type>) -> Type {
|
|
|
|
Type::Aggregate(flatten, types)
|
|
|
|
}
|
2015-09-02 13:59:35 -07:00
|
|
|
fn p(const_: bool, elem: Type, llvm_elem: Option<Type>) -> Type {
|
|
|
|
Type::Pointer(Box::new(elem), llvm_elem.map(Box::new), const_)
|
|
|
|
}
|
2015-07-16 16:46:36 -07:00
|
|
|
|
|
|
|
mod x86;
|
|
|
|
mod arm;
|
|
|
|
mod aarch64;
|
|
|
|
|
|
|
|
impl Intrinsic {
|
|
|
|
pub fn find<'tcx>(tcx: &ty::ctxt<'tcx>, name: &str) -> Option<Intrinsic> {
|
|
|
|
if name.starts_with("x86_") {
|
2015-08-28 16:38:25 -07:00
|
|
|
x86::find(tcx, name)
|
2015-07-16 16:46:36 -07:00
|
|
|
} else if name.starts_with("arm_") {
|
2015-08-28 16:38:25 -07:00
|
|
|
arm::find(tcx, name)
|
2015-07-16 16:46:36 -07:00
|
|
|
} else if name.starts_with("aarch64_") {
|
2015-08-28 16:38:25 -07:00
|
|
|
aarch64::find(tcx, name)
|
2015-07-16 16:46:36 -07:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|