2019-11-26 06:40:55 -06:00
|
|
|
//! Defines primitive types, which have a couple of peculiarities:
|
|
|
|
//!
|
|
|
|
//! * during type inference, they can be uncertain (ie, `let x = 92;`)
|
|
|
|
//! * they don't belong to any particular crate.
|
2019-11-26 05:35:23 -06:00
|
|
|
|
|
|
|
use std::fmt;
|
|
|
|
|
|
|
|
pub use hir_def::builtin_type::{BuiltinFloat, BuiltinInt, FloatBitness, IntBitness, Signedness};
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Eq, PartialEq, Hash, Debug)]
|
|
|
|
pub enum Uncertain<T> {
|
|
|
|
Unknown,
|
|
|
|
Known(T),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<IntTy> for Uncertain<IntTy> {
|
|
|
|
fn from(ty: IntTy) -> Self {
|
|
|
|
Uncertain::Known(ty)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Uncertain<IntTy> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match *self {
|
|
|
|
Uncertain::Unknown => write!(f, "{{integer}}"),
|
|
|
|
Uncertain::Known(ty) => write!(f, "{}", ty),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<FloatTy> for Uncertain<FloatTy> {
|
|
|
|
fn from(ty: FloatTy) -> Self {
|
|
|
|
Uncertain::Known(ty)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Uncertain<FloatTy> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match *self {
|
|
|
|
Uncertain::Unknown => write!(f, "{{float}}"),
|
|
|
|
Uncertain::Known(ty) => write!(f, "{}", ty),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, Eq, PartialEq, Hash)]
|
|
|
|
pub struct IntTy {
|
|
|
|
pub signedness: Signedness,
|
|
|
|
pub bitness: IntBitness,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for IntTy {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(self, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for IntTy {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "{}", self.ty_to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntTy {
|
|
|
|
pub fn isize() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::Xsize }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn i8() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::X8 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn i16() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::X16 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn i32() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::X32 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn i64() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::X64 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn i128() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Signed, bitness: IntBitness::X128 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn usize() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::Xsize }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn u8() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::X8 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn u16() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::X16 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn u32() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::X32 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn u64() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::X64 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn u128() -> IntTy {
|
|
|
|
IntTy { signedness: Signedness::Unsigned, bitness: IntBitness::X128 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty_to_string(self) -> &'static str {
|
|
|
|
match (self.signedness, self.bitness) {
|
|
|
|
(Signedness::Signed, IntBitness::Xsize) => "isize",
|
|
|
|
(Signedness::Signed, IntBitness::X8) => "i8",
|
|
|
|
(Signedness::Signed, IntBitness::X16) => "i16",
|
|
|
|
(Signedness::Signed, IntBitness::X32) => "i32",
|
|
|
|
(Signedness::Signed, IntBitness::X64) => "i64",
|
|
|
|
(Signedness::Signed, IntBitness::X128) => "i128",
|
|
|
|
(Signedness::Unsigned, IntBitness::Xsize) => "usize",
|
|
|
|
(Signedness::Unsigned, IntBitness::X8) => "u8",
|
|
|
|
(Signedness::Unsigned, IntBitness::X16) => "u16",
|
|
|
|
(Signedness::Unsigned, IntBitness::X32) => "u32",
|
|
|
|
(Signedness::Unsigned, IntBitness::X64) => "u64",
|
|
|
|
(Signedness::Unsigned, IntBitness::X128) => "u128",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub struct FloatTy {
|
|
|
|
pub bitness: FloatBitness,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for FloatTy {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(self, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for FloatTy {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "{}", self.ty_to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FloatTy {
|
|
|
|
pub fn f32() -> FloatTy {
|
|
|
|
FloatTy { bitness: FloatBitness::X32 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn f64() -> FloatTy {
|
|
|
|
FloatTy { bitness: FloatBitness::X64 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty_to_string(self) -> &'static str {
|
|
|
|
match self.bitness {
|
|
|
|
FloatBitness::X32 => "f32",
|
|
|
|
FloatBitness::X64 => "f64",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<BuiltinInt> for IntTy {
|
|
|
|
fn from(t: BuiltinInt) -> Self {
|
|
|
|
IntTy { signedness: t.signedness, bitness: t.bitness }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<BuiltinFloat> for FloatTy {
|
|
|
|
fn from(t: BuiltinFloat) -> Self {
|
|
|
|
FloatTy { bitness: t.bitness }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Option<BuiltinInt>> for Uncertain<IntTy> {
|
|
|
|
fn from(t: Option<BuiltinInt>) -> Self {
|
|
|
|
match t {
|
|
|
|
None => Uncertain::Unknown,
|
|
|
|
Some(t) => Uncertain::Known(t.into()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Option<BuiltinFloat>> for Uncertain<FloatTy> {
|
|
|
|
fn from(t: Option<BuiltinFloat>) -> Self {
|
|
|
|
match t {
|
|
|
|
None => Uncertain::Unknown,
|
|
|
|
Some(t) => Uncertain::Known(t.into()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|