2014-05-26 00:27:36 +01:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
2013-03-13 22:25:28 -04:00
|
|
|
// 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.
|
|
|
|
|
2014-02-19 18:56:33 -08:00
|
|
|
use std::fmt;
|
2013-03-13 22:25:28 -04:00
|
|
|
|
2014-05-29 17:45:07 -07:00
|
|
|
#[deriving(PartialEq)]
|
2014-05-05 10:07:49 +03:00
|
|
|
pub enum Os { OsWin32, OsMacos, OsLinux, OsAndroid, OsFreebsd, OsiOS, }
|
2013-11-08 11:06:57 -08:00
|
|
|
|
2014-05-31 10:43:52 -07:00
|
|
|
#[deriving(PartialEq, Eq, Hash, Encodable, Decodable, Clone)]
|
2013-03-13 22:25:28 -04:00
|
|
|
pub enum Abi {
|
|
|
|
// NB: This ordering MUST match the AbiDatas array below.
|
|
|
|
// (This is ensured by the test indices_are_correct().)
|
|
|
|
|
|
|
|
// Single platform ABIs come first (`for_arch()` relies on this)
|
|
|
|
Cdecl,
|
|
|
|
Stdcall,
|
|
|
|
Fastcall,
|
|
|
|
Aapcs,
|
2013-11-16 18:25:56 -05:00
|
|
|
Win64,
|
2013-03-13 22:25:28 -04:00
|
|
|
|
|
|
|
// Multiplatform ABIs second
|
|
|
|
Rust,
|
|
|
|
C,
|
2013-11-08 11:06:57 -08:00
|
|
|
System,
|
2013-03-13 22:25:28 -04:00
|
|
|
RustIntrinsic,
|
|
|
|
}
|
|
|
|
|
2014-01-09 15:05:33 +02:00
|
|
|
#[allow(non_camel_case_types)]
|
2014-05-29 17:45:07 -07:00
|
|
|
#[deriving(PartialEq)]
|
2013-03-13 22:25:28 -04:00
|
|
|
pub enum Architecture {
|
|
|
|
// NB. You cannot change the ordering of these
|
|
|
|
// constants without adjusting IntelBits below.
|
|
|
|
// (This is ensured by the test indices_are_correct().)
|
|
|
|
X86,
|
|
|
|
X86_64,
|
|
|
|
Arm,
|
2014-06-17 09:16:03 +02:00
|
|
|
Mips,
|
|
|
|
Mipsel
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
2013-03-30 19:28:50 -07:00
|
|
|
static IntelBits: u32 = (1 << (X86 as uint)) | (1 << (X86_64 as uint));
|
|
|
|
static ArmBits: u32 = (1 << (Arm as uint));
|
2013-03-13 22:25:28 -04:00
|
|
|
|
2014-02-27 18:48:21 +11:00
|
|
|
pub struct AbiData {
|
2013-03-13 22:25:28 -04:00
|
|
|
abi: Abi,
|
|
|
|
|
|
|
|
// Name of this ABI as we like it called.
|
|
|
|
name: &'static str,
|
|
|
|
|
|
|
|
// Is it specific to a platform? If so, which one? Also, what is
|
|
|
|
// the name that LLVM gives it (in case we disagree)
|
|
|
|
abi_arch: AbiArchitecture
|
|
|
|
}
|
|
|
|
|
2014-02-27 18:48:21 +11:00
|
|
|
pub enum AbiArchitecture {
|
2013-03-13 22:25:28 -04:00
|
|
|
RustArch, // Not a real ABI (e.g., intrinsic)
|
|
|
|
AllArch, // An ABI that specifies cross-platform defaults (e.g., "C")
|
|
|
|
Archs(u32) // Multiple architectures (bitset)
|
|
|
|
}
|
|
|
|
|
|
|
|
static AbiDatas: &'static [AbiData] = &[
|
|
|
|
// Platform-specific ABIs
|
|
|
|
AbiData {abi: Cdecl, name: "cdecl", abi_arch: Archs(IntelBits)},
|
|
|
|
AbiData {abi: Stdcall, name: "stdcall", abi_arch: Archs(IntelBits)},
|
|
|
|
AbiData {abi: Fastcall, name:"fastcall", abi_arch: Archs(IntelBits)},
|
|
|
|
AbiData {abi: Aapcs, name: "aapcs", abi_arch: Archs(ArmBits)},
|
2013-11-16 18:25:56 -05:00
|
|
|
AbiData {abi: Win64, name: "win64",
|
|
|
|
abi_arch: Archs(1 << (X86_64 as uint))},
|
2013-03-13 22:25:28 -04:00
|
|
|
|
|
|
|
// Cross-platform ABIs
|
|
|
|
//
|
|
|
|
// NB: Do not adjust this ordering without
|
|
|
|
// adjusting the indices below.
|
|
|
|
AbiData {abi: Rust, name: "Rust", abi_arch: RustArch},
|
|
|
|
AbiData {abi: C, name: "C", abi_arch: AllArch},
|
2013-11-08 11:06:57 -08:00
|
|
|
AbiData {abi: System, name: "system", abi_arch: AllArch},
|
2013-03-13 22:25:28 -04:00
|
|
|
AbiData {abi: RustIntrinsic, name: "rust-intrinsic", abi_arch: RustArch},
|
|
|
|
];
|
|
|
|
|
2013-11-19 12:21:21 -08:00
|
|
|
fn each_abi(op: |abi: Abi| -> bool) -> bool {
|
2013-05-02 18:33:33 -04:00
|
|
|
/*!
|
|
|
|
*
|
|
|
|
* Iterates through each of the defined ABIs.
|
|
|
|
*/
|
|
|
|
|
2013-06-21 08:29:53 -04:00
|
|
|
AbiDatas.iter().advance(|abi_data| op(abi_data.abi))
|
2013-05-02 18:33:33 -04:00
|
|
|
}
|
2013-03-13 22:25:28 -04:00
|
|
|
|
|
|
|
pub fn lookup(name: &str) -> Option<Abi> {
|
|
|
|
/*!
|
|
|
|
*
|
|
|
|
* Returns the ABI with the given name (if any).
|
|
|
|
*/
|
|
|
|
|
2013-08-02 02:17:20 -04:00
|
|
|
let mut res = None;
|
|
|
|
|
2013-11-20 16:23:04 -08:00
|
|
|
each_abi(|abi| {
|
2013-03-13 22:25:28 -04:00
|
|
|
if name == abi.data().name {
|
2013-08-02 02:17:20 -04:00
|
|
|
res = Some(abi);
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
true
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
2013-11-20 16:23:04 -08:00
|
|
|
});
|
2013-08-02 02:17:20 -04:00
|
|
|
res
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
2014-02-28 13:09:09 -08:00
|
|
|
pub fn all_names() -> Vec<&'static str> {
|
2014-02-28 12:54:01 -08:00
|
|
|
AbiDatas.iter().map(|d| d.name).collect()
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
impl Abi {
|
2013-03-13 22:25:28 -04:00
|
|
|
#[inline]
|
2013-05-31 15:17:22 -07:00
|
|
|
pub fn index(&self) -> uint {
|
2013-03-13 22:25:28 -04:00
|
|
|
*self as uint
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2013-05-31 15:17:22 -07:00
|
|
|
pub fn data(&self) -> &'static AbiData {
|
2013-03-13 22:25:28 -04:00
|
|
|
&AbiDatas[self.index()]
|
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
pub fn name(&self) -> &'static str {
|
2013-03-13 22:25:28 -04:00
|
|
|
self.data().name
|
|
|
|
}
|
2013-11-08 11:06:57 -08:00
|
|
|
|
2014-04-02 01:19:41 -07:00
|
|
|
pub fn for_target(&self, os: Os, arch: Architecture) -> Option<Abi> {
|
|
|
|
// If this ABI isn't actually for the specified architecture, then we
|
|
|
|
// short circuit early
|
|
|
|
match self.data().abi_arch {
|
|
|
|
Archs(a) if a & arch.bit() == 0 => return None,
|
|
|
|
Archs(_) | RustArch | AllArch => {}
|
|
|
|
}
|
|
|
|
// Transform this ABI as appropriate for the requested os/arch
|
|
|
|
// combination.
|
|
|
|
Some(match (*self, os, arch) {
|
2013-11-08 11:06:57 -08:00
|
|
|
(System, OsWin32, X86) => Stdcall,
|
|
|
|
(System, _, _) => C,
|
|
|
|
(me, _, _) => me,
|
2014-04-02 01:19:41 -07:00
|
|
|
})
|
2013-11-08 11:06:57 -08:00
|
|
|
}
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Architecture {
|
|
|
|
fn bit(&self) -> u32 {
|
2014-04-21 17:58:52 -04:00
|
|
|
1 << (*self as uint)
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-19 18:56:33 -08:00
|
|
|
impl fmt::Show for Abi {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2014-05-10 14:05:06 -07:00
|
|
|
write!(f, "\"{}\"", self.name())
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-11 10:48:17 +03:00
|
|
|
impl fmt::Show for Os {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match *self {
|
|
|
|
OsLinux => "linux".fmt(f),
|
|
|
|
OsWin32 => "win32".fmt(f),
|
|
|
|
OsMacos => "macos".fmt(f),
|
|
|
|
OsiOS => "ios".fmt(f),
|
|
|
|
OsAndroid => "android".fmt(f),
|
|
|
|
OsFreebsd => "freebsd".fmt(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-26 00:27:36 +01:00
|
|
|
#[allow(non_snake_case_functions)]
|
2013-03-13 22:25:28 -04:00
|
|
|
#[test]
|
|
|
|
fn lookup_Rust() {
|
|
|
|
let abi = lookup("Rust");
|
2013-08-04 01:59:24 +02:00
|
|
|
assert!(abi.is_some() && abi.unwrap().data().name == "Rust");
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lookup_cdecl() {
|
|
|
|
let abi = lookup("cdecl");
|
2013-08-04 01:59:24 +02:00
|
|
|
assert!(abi.is_some() && abi.unwrap().data().name == "cdecl");
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lookup_baz() {
|
|
|
|
let abi = lookup("baz");
|
|
|
|
assert!(abi.is_none());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn indices_are_correct() {
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, abi_data) in AbiDatas.iter().enumerate() {
|
2013-11-08 11:06:57 -08:00
|
|
|
assert_eq!(i, abi_data.abi.index());
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
let bits = 1 << (X86 as uint);
|
|
|
|
let bits = bits | 1 << (X86_64 as uint);
|
2013-11-08 11:06:57 -08:00
|
|
|
assert_eq!(IntelBits, bits);
|
2013-03-13 22:25:28 -04:00
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
let bits = 1 << (Arm as uint);
|
2013-11-08 11:06:57 -08:00
|
|
|
assert_eq!(ArmBits, bits);
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pick_uniplatform() {
|
2014-04-03 14:09:16 -07:00
|
|
|
assert_eq!(Stdcall.for_target(OsLinux, X86), Some(Stdcall));
|
|
|
|
assert_eq!(Stdcall.for_target(OsLinux, Arm), None);
|
|
|
|
assert_eq!(System.for_target(OsLinux, X86), Some(C));
|
|
|
|
assert_eq!(System.for_target(OsWin32, X86), Some(Stdcall));
|
|
|
|
assert_eq!(System.for_target(OsWin32, X86_64), Some(C));
|
|
|
|
assert_eq!(System.for_target(OsWin32, Arm), Some(C));
|
|
|
|
assert_eq!(Stdcall.for_target(OsWin32, X86), Some(Stdcall));
|
|
|
|
assert_eq!(Stdcall.for_target(OsWin32, X86_64), Some(Stdcall));
|
2013-03-13 22:25:28 -04:00
|
|
|
}
|