// SPDX-License-Identifier: LGPL-3.0-or-later // See Notices.txt for copyright information use crate::{ intern::{Intern, Interned, Memoize}, source_location::SourceLocation, ty::CanonicalType, util::get_many_mut, }; use hashbrown::HashMap; use num_bigint::BigInt; use num_traits::{One, Signed, ToPrimitive, Zero}; use std::{ borrow::BorrowMut, convert::Infallible, fmt, hash::{Hash, Hasher}, marker::PhantomData, ops::{Deref, DerefMut, Index, IndexMut}, }; use vec_map::VecMap; pub(crate) type SmallUInt = u64; pub(crate) type SmallSInt = i64; pub(crate) const MIN_BITS_FOR_NEEDING_BIG: usize = SmallUInt::BITS as usize + 1; #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] pub(crate) enum InsnFieldKind { Input, Output, Immediate, BranchTarget, } pub(crate) trait InsnFieldTypeTransform: Eq + Hash + fmt::Debug + Send + Sync { type Type: Eq + Hash + fmt::Debug + Send + Sync; fn empty_type() -> Self::Type<[(); 0]>; } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, Default)] pub(crate) struct InsnFieldTypeTransformUnit; impl InsnFieldTypeTransform for InsnFieldTypeTransformUnit { type Type = (); fn empty_type() -> Self::Type<[(); 0]> { () } } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] pub(crate) struct InsnFieldTypeTransformRef<'a>(PhantomData<&'a ()>); impl<'a> InsnFieldTypeTransform for InsnFieldTypeTransformRef<'a> { type Type = &'a FieldType; fn empty_type() -> Self::Type<[(); 0]> { &[] } } #[derive(PartialEq, Eq, Hash, Debug)] pub(crate) struct InsnFieldTypeTransformRefMut<'a>(PhantomData<&'a mut ()>); impl<'a> InsnFieldTypeTransform for InsnFieldTypeTransformRefMut<'a> { type Type = &'a mut FieldType; fn empty_type() -> Self::Type<[(); 0]> { &mut [] } } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] pub(crate) struct InsnFieldTypeTransformValue; impl InsnFieldTypeTransform for InsnFieldTypeTransformValue { type Type = FieldType; fn empty_type() -> Self::Type<[(); 0]> { [] } } pub trait InsnFieldTrait: Send + Sync + 'static + Copy + Eq + Hash + fmt::Debug { const UNIT: InsnFieldType; fn variant( v: Transform::Type, ) -> InsnFieldType; } macro_rules! insn_field_enum { ( $enum_vis:vis enum $InsnFieldType:ident<$Transform:ident: $InsnFieldTypeTransform:ident> { $($Variant:ident($Transform2:ident::$Type:ident<$variant_ty:ty>),)* } ) => { #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] $enum_vis enum $InsnFieldType<$Transform: $InsnFieldTypeTransform> { $($Variant($Transform2::$Type<$variant_ty>),)* } $(impl InsnFieldTrait for $variant_ty { const UNIT: $InsnFieldType = $InsnFieldType::$Variant(()); fn variant<$Transform2: $InsnFieldTypeTransform>( v: $Transform2::$Type, ) -> $InsnFieldType<$Transform2> { $InsnFieldType::$Variant(v) } })* }; } insn_field_enum! { pub(crate) enum InsnFieldType { SmallSlot(Transform::Type>), BigSlot(Transform::Type>), SmallSlotArrayIndexed(Transform::Type>), BigSlotArrayIndexed(Transform::Type>), SmallUInt(Transform::Type), SmallSInt(Transform::Type), InternedBigInt(Transform::Type>), U8(Transform::Type), USize(Transform::Type), Empty(Transform::Type<[(); 0]>), } } impl InsnFieldType { pub(crate) fn empty() -> Self { Self::Empty(Transform::empty_type()) } } #[derive(PartialEq, Eq, Hash, Debug)] pub(crate) struct InsnField { pub(crate) ty: InsnFieldType, pub(crate) kind: InsnFieldKind, } impl Clone for InsnField where InsnFieldType: Clone, { fn clone(&self) -> Self { Self { ty: self.ty.clone(), kind: self.kind.clone(), } } } impl Copy for InsnField where InsnFieldType: Copy { } fn make_array_into_iter( input: [T; I], mut default: impl FnMut() -> T, ) -> std::array::IntoIter { const { assert!(I <= N); }; let mut input = input.into_iter(); let array = std::array::from_fn(|_| input.next().unwrap_or_else(&mut default)); let mut retval = array.into_iter(); // remove unneeded trailing elements if I < N { retval.nth_back(N - I - 1); } retval } impl fmt::Debug for Insn { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { self.debug_fmt::(f, None, None) } } impl Insn { fn debug_fmt( &self, f: &mut fmt::Formatter<'_>, labels: Option<&Labels>, state_layout: Option<&StateLayout>, ) -> fmt::Result { let (insn_name, fields) = self.fields_with_names(); write!(f, "{insn_name}")?; if fields.len() == 0 { return Ok(()); } writeln!(f, " {{")?; for (field_name, field) in fields { write!(f, " {field_name}: ")?; match field.kind { InsnFieldKind::BranchTarget => match field.ty { InsnFieldType::USize(&label_index) => { if let Some(labels) = labels { write!(f, "L{label_index}")?; if let Some(label) = labels.labels.get(label_index) { if let Some(address) = label.address { write!(f, " (at {address})")?; } else { write!(f, " (not yet defined)")?; } } else { write!(f, " (invalid)")?; } writeln!(f, ",")?; continue; } } InsnFieldType::SmallSlot(_) | InsnFieldType::BigSlot(_) | InsnFieldType::SmallSlotArrayIndexed(_) | InsnFieldType::BigSlotArrayIndexed(_) | InsnFieldType::SmallUInt(_) | InsnFieldType::SmallSInt(_) | InsnFieldType::InternedBigInt(_) | InsnFieldType::U8(_) | InsnFieldType::Empty(_) => {} }, InsnFieldKind::Input | InsnFieldKind::Output | InsnFieldKind::Immediate => {} } match field.ty { InsnFieldType::SmallSlot(v) => { v.debug_fmt(f, ",", " // ", "", state_layout)?; } InsnFieldType::BigSlot(v) => { v.debug_fmt(f, ",", " // ", "", state_layout)?; } InsnFieldType::SmallSlotArrayIndexed(v) => { v.debug_fmt(f, ",", " // ", "", state_layout)?; } InsnFieldType::BigSlotArrayIndexed(v) => { v.debug_fmt(f, ",", " // ", "", state_layout)?; } InsnFieldType::SmallUInt(v) => fmt::Debug::fmt(v, f)?, InsnFieldType::SmallSInt(v) => fmt::Debug::fmt(v, f)?, InsnFieldType::InternedBigInt(v) => fmt::Debug::fmt(v, f)?, InsnFieldType::U8(v) => fmt::Debug::fmt(v, f)?, InsnFieldType::USize(v) => fmt::Debug::fmt(v, f)?, InsnFieldType::Empty(v) => fmt::Debug::fmt(v, f)?, } writeln!(f, ",")?; } write!(f, "}}") } } macro_rules! impl_insns { ( #[insn = $Insn:ident, next_macro = $next_macro:ident, branch_macro = $branch_macro:ident] $vis:vis fn $State:ident::$run:ident(&mut $self:ident) -> $run_ret_ty:ty { #[state] let mut $state:ident = $state_init:expr; setup! { $($setup:tt)* } main_loop!(); cleanup! { $($cleanup:tt)* } } $( $(#[$insn_meta:meta])* $insn_name:ident $({ $( #[kind = $field_kind:ident] $(#[$field_meta:meta])* $field_name:ident: $field_ty:ty, )* })? => $block:block )* ) => { #[derive(Copy, Clone, Eq, PartialEq, Hash)] $vis enum $Insn { $( $(#[$insn_meta])* $insn_name $({ $( $(#[$field_meta])* $field_name: $field_ty, )* })?, )* } impl $Insn { $vis const MAX_FIELDS: usize = { let mut retval = 0; $($( let fields = [$(stringify!($field_name),)*].len(); if retval < fields { retval = fields; } )?)* retval }; } impl $Insn { $vis const fn fields_unit(&self) -> &'static [InsnField] { match self { $( $Insn::$insn_name {..} => &[ $($(InsnField { ty: <$field_ty as InsnFieldTrait>::UNIT, kind: InsnFieldKind::$field_kind, },)*)? ], )* } } $vis fn fields<'a>(&'a self) -> std::array::IntoIter>, { $Insn::MAX_FIELDS }> { match self { $( $Insn::$insn_name $({ $($field_name,)* })? => make_array_into_iter([ $($(InsnField { ty: <$field_ty as InsnFieldTrait>::variant($field_name), kind: InsnFieldKind::$field_kind, },)*)? ], || InsnField { ty: InsnFieldType::empty(), kind: InsnFieldKind::Immediate, }, ), )* } } $vis fn fields_with_names<'a>(&'a self) -> (&'static str, std::array::IntoIter<(&'static str, InsnField>), { $Insn::MAX_FIELDS }>) { match self { $( $Insn::$insn_name $({ $($field_name,)* })? => ( stringify!($insn_name), make_array_into_iter([ $($((stringify!($field_name), InsnField { ty: <$field_ty as InsnFieldTrait>::variant($field_name), kind: InsnFieldKind::$field_kind, }),)*)? ], || ("", InsnField { ty: InsnFieldType::empty(), kind: InsnFieldKind::Immediate, }), ), ), )* } } $vis fn fields_mut<'a>(&'a mut self) -> std::array::IntoIter>, { $Insn::MAX_FIELDS }> { match self { $( $Insn::$insn_name $({ $($field_name,)* })? => make_array_into_iter([ $($(InsnField { ty: <$field_ty as InsnFieldTrait>::variant($field_name), kind: InsnFieldKind::$field_kind, },)*)? ], || InsnField { ty: InsnFieldType::empty(), kind: InsnFieldKind::Immediate, }, ), )* } } $vis fn into_fields(self) -> std::array::IntoIter, { $Insn::MAX_FIELDS }> { match self { $( $Insn::$insn_name $({ $($field_name,)* })? => make_array_into_iter([ $($(InsnField { ty: <$field_ty as InsnFieldTrait>::variant($field_name), kind: InsnFieldKind::$field_kind, },)*)? ], || InsnField { ty: InsnFieldType::empty(), kind: InsnFieldKind::Immediate, }, ), )* } } } impl $State { $vis fn $run(&mut $self) -> $run_ret_ty { let mut $state = $state_init; $($setup)* let mut insn = $state.insns[$state.pc]; let retval = 'main_loop: loop { macro_rules! $next_macro { () => { $state.pc += 1; insn = $state.insns[$state.pc]; continue 'main_loop; }; } macro_rules! $branch_macro { ($next_pc:expr) => { $state.pc = $next_pc; insn = $state.insns[$state.pc]; continue 'main_loop; }; } let _: Infallible = match insn { $( $Insn::$insn_name $({ $( $field_name, )* })? => { $block } )* }; }; $($cleanup)* retval } } }; } pub(crate) trait InsnsBuildingKind: Copy + Eq + fmt::Debug + Hash + Default { type Vec: fmt::Debug + Clone + Eq + Hash + Send + Sync + 'static + Default + Deref + FromIterator; type Labels: Default; fn labels(labels: &Self::Labels) -> Option<&Labels>; } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, Default)] pub(crate) struct InsnsBuildingDone; impl InsnsBuildingKind for InsnsBuildingDone { type Vec = Interned<[T]>; type Labels = (); fn labels(_labels: &Self::Labels) -> Option<&Labels> { None } } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, Default)] pub(crate) struct InsnsBuilding; impl InsnsBuildingKind for InsnsBuilding { type Vec = Vec; type Labels = Labels; fn labels(labels: &Self::Labels) -> Option<&Labels> { Some(labels) } } struct Label { address: Option, } #[derive(Default)] pub(crate) struct Labels { labels: Vec