compiler-builtins-zynq/src/lib.rs

121 lines
2.4 KiB
Rust
Raw Normal View History

2016-08-08 04:58:05 +08:00
#![allow(unused_features)]
#![cfg_attr(not(test), no_std)]
#![feature(asm)]
#![feature(core_intrinsics)]
#![feature(naked_functions)]
// TODO(rust-lang/rust#35021) uncomment when that PR lands
// #![feature(rustc_builtins)]
#[cfg(test)]
extern crate core;
2016-08-11 08:12:37 +08:00
#[cfg(test)]
#[macro_use]
extern crate quickcheck;
2016-08-08 04:58:05 +08:00
#[cfg(target_arch = "arm")]
pub mod arm;
pub mod div;
2016-08-08 04:58:05 +08:00
#[cfg(test)]
mod test;
use core::ops::{Index, IndexMut, RangeFull};
2016-08-11 12:28:15 +08:00
/// Trait for some basic operations on integers
trait Int {
fn bits() -> usize;
}
// TODO: Once i128/u128 support lands, we'll want to add impls for those as well
impl Int for u32 {
fn bits() -> usize {
32
}
2016-08-11 12:28:15 +08:00
}
impl Int for i32 {
fn bits() -> usize {
32
}
2016-08-11 12:28:15 +08:00
}
impl Int for u64 {
fn bits() -> usize {
64
}
2016-08-11 12:28:15 +08:00
}
impl Int for i64 {
fn bits() -> usize {
64
}
2016-08-11 12:28:15 +08:00
}
/// Trait to convert an integer to/from smaller parts
trait LargeInt {
type LowHalf;
type HighHalf;
fn low(self) -> Self::LowHalf;
fn high(self) -> Self::HighHalf;
fn from_parts(low: Self::LowHalf, high: Self::HighHalf) -> Self;
}
// TODO: Once i128/u128 support lands, we'll want to add impls for those as well
impl LargeInt for u64 {
type LowHalf = u32;
type HighHalf = u32;
fn low(self) -> u32 {
self as u32
}
fn high(self) -> u32 {
(self >> 32) as u32
}
fn from_parts(low: u32, high: u32) -> u64 {
low as u64 | ((high as u64) << 32)
}
}
impl LargeInt for i64 {
type LowHalf = u32;
type HighHalf = i32;
fn low(self) -> u32 {
self as u32
}
fn high(self) -> i32 {
(self >> 32) as i32
}
fn from_parts(low: u32, high: i32) -> i64 {
low as i64 | ((high as i64) << 32)
}
}
2016-08-08 04:58:05 +08:00
/// Union-like access to the 32-bit words that make an `u64`: `x.low` and `x.high`. The whole `u64`
/// can be accessed via the expression `x[..]`, which can be used in lvalue or rvalue position.
#[cfg(target_endian = "little")]
#[repr(C)]
struct U64 {
low: u32,
high: u32,
}
#[cfg(target_endian = "big")]
#[repr(C)]
struct U64 {
high: u32,
low: u32,
}
impl Index<RangeFull> for U64 {
type Output = u64;
fn index(&self, _: RangeFull) -> &u64 {
unsafe { &*(self as *const _ as *const u64) }
}
}
impl IndexMut<RangeFull> for U64 {
fn index_mut(&mut self, _: RangeFull) -> &mut u64 {
unsafe { &mut *(self as *const _ as *mut u64) }
}
}