use std::{
hash::{Hash, Hasher},
mem,
};
use is_macro::Is;
#[cfg(feature = "serde-impl")]
use serde::{Deserialize, Serialize};
use swc_atoms::Atom;
use swc_common::{ast_node, util::take::Take, EqIgnoreSpan, Span};
#[ast_node("PreservedToken")]
#[derive(Eq, Hash, EqIgnoreSpan)]
pub struct TokenAndSpan {
pub span: Span,
pub token: Token,
}
impl Take for TokenAndSpan {
fn dummy() -> Self {
Self {
span: Take::dummy(),
token: Take::dummy(),
}
}
}
#[derive(Debug, Clone, PartialEq, EqIgnoreSpan, Hash)]
#[cfg_attr(feature = "serde-impl", derive(serde::Serialize, serde::Deserialize))]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct UrlKeyValue(pub Atom, pub Atom);
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, Is, EqIgnoreSpan)]
#[cfg_attr(
feature = "rkyv",
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv", repr(u32))]
#[cfg_attr(
feature = "rkyv",
rkyv(serialize_bounds(__S: rkyv::ser::Writer + rkyv::ser::Allocator,
__S::Error: rkyv::rancor::Source))
)]
#[cfg_attr(feature = "serde-impl", derive(Serialize, Deserialize))]
pub enum NumberType {
#[cfg_attr(feature = "serde-impl", serde(rename = "integer"))]
Integer,
#[cfg_attr(feature = "serde-impl", serde(rename = "number"))]
Number,
}
#[derive(Debug, Clone, PartialEq, EqIgnoreSpan)]
#[cfg_attr(
feature = "rkyv",
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv", repr(C))]
#[cfg_attr(feature = "serde-impl", derive(Serialize, Deserialize))]
pub struct DimensionToken {
pub value: f64,
pub raw_value: Atom,
pub unit: Atom,
#[cfg_attr(feature = "serde-impl", serde(rename = "type"))]
pub type_flag: NumberType,
pub raw_unit: Atom,
}
#[derive(Debug, Clone, PartialEq, EqIgnoreSpan)]
#[cfg_attr(
feature = "rkyv",
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv", repr(u32))]
#[cfg_attr(
feature = "rkyv",
rkyv(serialize_bounds(__S: rkyv::ser::Writer + rkyv::ser::Allocator,
__S::Error: rkyv::rancor::Source))
)]
#[cfg_attr(feature = "serde-impl", derive(serde::Serialize, serde::Deserialize))]
pub enum Token {
Ident {
value: Atom,
raw: Atom,
},
Function {
value: Atom,
raw: Atom,
},
AtKeyword {
value: Atom,
raw: Atom,
},
Hash {
is_id: bool,
value: Atom,
raw: Atom,
},
String {
value: Atom,
raw: Atom,
},
BadString {
raw: Atom,
},
Url {
value: Atom,
raw: Box<UrlKeyValue>,
},
BadUrl {
raw: Atom,
},
Delim {
value: char,
},
Number {
value: f64,
raw: Atom,
#[cfg_attr(feature = "serde-impl", serde(rename = "type"))]
type_flag: NumberType,
},
Percentage {
value: f64,
raw: Atom,
},
Dimension(Box<DimensionToken>),
WhiteSpace {
value: Atom,
},
CDO,
CDC,
Colon,
Semi,
Comma,
LBracket,
RBracket,
LParen,
RParen,
LBrace,
RBrace,
}
impl Take for Token {
fn dummy() -> Self {
Self::Semi
}
}
#[allow(clippy::derived_hash_with_manual_eq)]
#[allow(clippy::transmute_float_to_int)]
impl Hash for Token {
fn hash<H: Hasher>(&self, state: &mut H) {
fn integer_decode(val: f64) -> (u64, i16, i8) {
let bits: u64 = unsafe { mem::transmute(val) };
let sign: i8 = if bits >> 63 == 0 { 1 } else { -1 };
let mut exponent: i16 = ((bits >> 52) & 0x7ff) as i16;
let mantissa = if exponent == 0 {
(bits & 0xfffffffffffff) << 1
} else {
(bits & 0xfffffffffffff) | 0x10000000000000
};
exponent -= 1023 + 52;
(mantissa, exponent, sign)
}
match self {
Token::Ident { value, raw } => {
value.hash(state);
raw.hash(state);
}
Token::Function { value, raw } => {
value.hash(state);
raw.hash(state);
}
Token::AtKeyword { value, raw } => {
value.hash(state);
raw.hash(state);
}
Token::String { value, raw } => {
value.hash(state);
raw.hash(state);
}
Token::BadString { raw } => {
raw.hash(state);
}
Token::Hash { value, raw, is_id } => {
value.hash(state);
raw.hash(state);
is_id.hash(state);
}
Token::Url { value, raw } => {
value.hash(state);
raw.hash(state);
}
Token::BadUrl { raw, .. } => {
raw.hash(state);
}
Token::Delim { value } => {
value.hash(state);
}
Token::Number {
value,
raw,
type_flag,
} => {
integer_decode(*value).hash(state);
raw.hash(state);
type_flag.hash(state);
}
Token::Percentage { value, raw } => {
integer_decode(*value).hash(state);
raw.hash(state);
}
Token::Dimension(dimension) => {
integer_decode(dimension.value).hash(state);
dimension.unit.hash(state);
dimension.type_flag.hash(state);
dimension.raw_value.hash(state);
dimension.raw_unit.hash(state);
}
Token::WhiteSpace { value, .. } => {
value.hash(state);
}
Token::CDO
| Token::CDC
| Token::Colon
| Token::Semi
| Token::Comma
| Token::LBracket
| Token::RBracket
| Token::LParen
| Token::RParen
| Token::LBrace
| Token::RBrace => {}
}
}
}
impl Eq for Token {}