use edition::Edition;
use crate::SyntaxKind;
#[allow(non_camel_case_types)]
type bits = u64;
pub struct Input {
kind: Vec<SyntaxKind>,
joint: Vec<bits>,
contextual_kind: Vec<SyntaxKind>,
edition: Vec<Edition>,
}
impl Input {
#[inline]
pub fn with_capacity(capacity: usize) -> Self {
Self {
kind: Vec::with_capacity(capacity),
joint: Vec::with_capacity(capacity / size_of::<bits>()),
contextual_kind: Vec::with_capacity(capacity),
edition: Vec::with_capacity(capacity),
}
}
#[inline]
pub fn push(&mut self, kind: SyntaxKind, edition: Edition) {
self.push_impl(kind, SyntaxKind::EOF, edition)
}
#[inline]
pub fn push_ident(&mut self, contextual_kind: SyntaxKind, edition: Edition) {
self.push_impl(SyntaxKind::IDENT, contextual_kind, edition)
}
#[inline]
pub fn was_joint(&mut self) {
let n = self.len() - 1;
let (idx, b_idx) = self.bit_index(n);
self.joint[idx] |= 1 << b_idx;
}
#[inline]
fn push_impl(&mut self, kind: SyntaxKind, contextual_kind: SyntaxKind, edition: Edition) {
let idx = self.len();
if idx.is_multiple_of(bits::BITS as usize) {
self.joint.push(0);
}
self.kind.push(kind);
self.contextual_kind.push(contextual_kind);
self.edition.push(edition);
}
}
impl Input {
pub(crate) fn kind(&self, idx: usize) -> SyntaxKind {
self.kind.get(idx).copied().unwrap_or(SyntaxKind::EOF)
}
pub(crate) fn contextual_kind(&self, idx: usize) -> SyntaxKind {
self.contextual_kind.get(idx).copied().unwrap_or(SyntaxKind::EOF)
}
pub(crate) fn edition(&self, idx: usize) -> Edition {
self.edition[idx]
}
pub(crate) fn is_joint(&self, n: usize) -> bool {
let (idx, b_idx) = self.bit_index(n);
self.joint[idx] & (1 << b_idx) != 0
}
}
impl Input {
fn bit_index(&self, n: usize) -> (usize, usize) {
let idx = n / (bits::BITS as usize);
let b_idx = n % (bits::BITS as usize);
(idx, b_idx)
}
pub fn len(&self) -> usize {
self.kind.len()
}
}