use crate::ast::*;
use crate::formatter::{FmtCtx, Formatter, IntoFormatter};
use crate::ids::Vector;
use crate::reorder_decls::DeclarationsGroups;
use derive_generic_visitor::{ControlFlow, Drive, DriveMut};
use hashlink::LinkedHashSet;
use macros::{EnumAsGetters, EnumIsA, VariantIndexArity, VariantName};
use serde::{Deserialize, Serialize};
use serde_map_to_array::HashMapToArray;
use std::cmp::{Ord, PartialOrd};
use std::collections::HashMap;
use std::fmt;
generate_index_type!(FunDeclId, "Fun");
generate_index_type!(TypeDeclId, "Adt");
generate_index_type!(GlobalDeclId, "Global");
generate_index_type!(TraitDeclId, "TraitDecl");
generate_index_type!(TraitImplId, "TraitImpl");
#[derive(
Copy,
Clone,
Debug,
PartialOrd,
Ord,
PartialEq,
Eq,
Hash,
EnumIsA,
EnumAsGetters,
VariantName,
VariantIndexArity,
Serialize,
Deserialize,
Drive,
DriveMut,
)]
#[charon::rename("AnyDeclId")]
#[charon::variants_prefix("Id")]
pub enum AnyTransId {
Type(TypeDeclId),
Fun(FunDeclId),
Global(GlobalDeclId),
TraitDecl(TraitDeclId),
TraitImpl(TraitImplId),
}
macro_rules! wrap_unwrap_enum {
($enum:ident::$variant:ident($variant_ty:ident)) => {
impl TryFrom<$enum> for $variant_ty {
type Error = ();
fn try_from(x: $enum) -> Result<Self, Self::Error> {
match x {
$enum::$variant(x) => Ok(x),
_ => Err(()),
}
}
}
impl From<$variant_ty> for $enum {
fn from(x: $variant_ty) -> Self {
$enum::$variant(x)
}
}
};
}
wrap_unwrap_enum!(AnyTransId::Fun(FunDeclId));
wrap_unwrap_enum!(AnyTransId::Global(GlobalDeclId));
wrap_unwrap_enum!(AnyTransId::Type(TypeDeclId));
wrap_unwrap_enum!(AnyTransId::TraitDecl(TraitDeclId));
wrap_unwrap_enum!(AnyTransId::TraitImpl(TraitImplId));
#[derive(
Debug, Clone, Copy, EnumIsA, EnumAsGetters, VariantName, VariantIndexArity, Drive, DriveMut,
)]
pub enum AnyTransItem<'ctx> {
Type(&'ctx TypeDecl),
Fun(&'ctx FunDecl),
Global(&'ctx GlobalDecl),
TraitDecl(&'ctx TraitDecl),
TraitImpl(&'ctx TraitImpl),
}
#[derive(Debug, EnumIsA, EnumAsGetters, VariantName, VariantIndexArity, Drive, DriveMut)]
pub enum AnyTransItemMut<'ctx> {
Type(&'ctx mut TypeDecl),
Fun(&'ctx mut FunDecl),
Global(&'ctx mut GlobalDecl),
TraitDecl(&'ctx mut TraitDecl),
TraitImpl(&'ctx mut TraitImpl),
}
#[derive(Default, Clone, Drive, DriveMut, Serialize, Deserialize)]
pub struct TranslatedCrate {
#[drive(skip)]
pub crate_name: String,
#[drive(skip)]
pub real_crate_name: String,
#[drive(skip)]
pub options: crate::options::CliOpts,
#[drive(skip)]
pub all_ids: LinkedHashSet<AnyTransId>,
#[serde(with = "HashMapToArray::<AnyTransId, Name>")]
pub item_names: HashMap<AnyTransId, Name>,
#[drive(skip)]
pub files: Vector<FileId, File>,
pub type_decls: Vector<TypeDeclId, TypeDecl>,
pub fun_decls: Vector<FunDeclId, FunDecl>,
pub global_decls: Vector<GlobalDeclId, GlobalDecl>,
pub trait_decls: Vector<TraitDeclId, TraitDecl>,
pub trait_impls: Vector<TraitImplId, TraitImpl>,
#[drive(skip)]
pub ordered_decls: Option<DeclarationsGroups>,
}
impl TranslatedCrate {
pub fn item_name(&self, trans_id: impl Into<AnyTransId>) -> Option<&Name> {
self.item_names.get(&trans_id.into())
}
pub fn get_item(&self, trans_id: impl Into<AnyTransId>) -> Option<AnyTransItem<'_>> {
match trans_id.into() {
AnyTransId::Type(id) => self.type_decls.get(id).map(AnyTransItem::Type),
AnyTransId::Fun(id) => self.fun_decls.get(id).map(AnyTransItem::Fun),
AnyTransId::Global(id) => self.global_decls.get(id).map(AnyTransItem::Global),
AnyTransId::TraitDecl(id) => self.trait_decls.get(id).map(AnyTransItem::TraitDecl),
AnyTransId::TraitImpl(id) => self.trait_impls.get(id).map(AnyTransItem::TraitImpl),
}
}
pub fn get_item_mut(&mut self, trans_id: AnyTransId) -> Option<AnyTransItemMut<'_>> {
match trans_id {
AnyTransId::Type(id) => self.type_decls.get_mut(id).map(AnyTransItemMut::Type),
AnyTransId::Fun(id) => self.fun_decls.get_mut(id).map(AnyTransItemMut::Fun),
AnyTransId::Global(id) => self.global_decls.get_mut(id).map(AnyTransItemMut::Global),
AnyTransId::TraitDecl(id) => {
self.trait_decls.get_mut(id).map(AnyTransItemMut::TraitDecl)
}
AnyTransId::TraitImpl(id) => {
self.trait_impls.get_mut(id).map(AnyTransItemMut::TraitImpl)
}
}
}
pub fn all_items(&self) -> impl Iterator<Item = AnyTransItem<'_>> {
self.all_items_with_ids().map(|(_, item)| item)
}
pub fn all_items_with_ids(&self) -> impl Iterator<Item = (AnyTransId, AnyTransItem<'_>)> {
self.all_ids
.iter()
.flat_map(|id| Some((*id, self.get_item(*id)?)))
}
pub fn all_items_mut(&mut self) -> impl Iterator<Item = AnyTransItemMut<'_>> {
self.type_decls
.iter_mut()
.map(AnyTransItemMut::Type)
.chain(self.fun_decls.iter_mut().map(AnyTransItemMut::Fun))
.chain(self.global_decls.iter_mut().map(AnyTransItemMut::Global))
.chain(self.trait_decls.iter_mut().map(AnyTransItemMut::TraitDecl))
.chain(self.trait_impls.iter_mut().map(AnyTransItemMut::TraitImpl))
}
}
impl<'ctx> AnyTransItem<'ctx> {
pub fn id(&self) -> AnyTransId {
match self {
AnyTransItem::Type(d) => d.def_id.into(),
AnyTransItem::Fun(d) => d.def_id.into(),
AnyTransItem::Global(d) => d.def_id.into(),
AnyTransItem::TraitDecl(d) => d.def_id.into(),
AnyTransItem::TraitImpl(d) => d.def_id.into(),
}
}
pub fn item_meta(&self) -> &'ctx ItemMeta {
match self {
AnyTransItem::Type(d) => &d.item_meta,
AnyTransItem::Fun(d) => &d.item_meta,
AnyTransItem::Global(d) => &d.item_meta,
AnyTransItem::TraitDecl(d) => &d.item_meta,
AnyTransItem::TraitImpl(d) => &d.item_meta,
}
}
pub fn generic_params(&self) -> &'ctx GenericParams {
match self {
AnyTransItem::Type(d) => &d.generics,
AnyTransItem::Fun(d) => &d.signature.generics,
AnyTransItem::Global(d) => &d.generics,
AnyTransItem::TraitDecl(d) => &d.generics,
AnyTransItem::TraitImpl(d) => &d.generics,
}
}
pub fn identity_args(&self) -> GenericArgs {
self.generic_params()
.identity_args(GenericsSource::Item(self.id()))
}
pub fn drive<V: VisitAst>(&self, visitor: &mut V) -> ControlFlow<V::Break> {
match *self {
AnyTransItem::Type(d) => visitor.visit(d),
AnyTransItem::Fun(d) => visitor.visit(d),
AnyTransItem::Global(d) => visitor.visit(d),
AnyTransItem::TraitDecl(d) => visitor.visit(d),
AnyTransItem::TraitImpl(d) => visitor.visit(d),
}
}
}
impl<'ctx> AnyTransItemMut<'ctx> {
pub fn as_ref(&self) -> AnyTransItem<'_> {
match self {
AnyTransItemMut::Type(d) => AnyTransItem::Type(d),
AnyTransItemMut::Fun(d) => AnyTransItem::Fun(d),
AnyTransItemMut::Global(d) => AnyTransItem::Global(d),
AnyTransItemMut::TraitDecl(d) => AnyTransItem::TraitDecl(d),
AnyTransItemMut::TraitImpl(d) => AnyTransItem::TraitImpl(d),
}
}
pub fn generic_params(&mut self) -> &mut GenericParams {
match self {
AnyTransItemMut::Type(d) => &mut d.generics,
AnyTransItemMut::Fun(d) => &mut d.signature.generics,
AnyTransItemMut::Global(d) => &mut d.generics,
AnyTransItemMut::TraitDecl(d) => &mut d.generics,
AnyTransItemMut::TraitImpl(d) => &mut d.generics,
}
}
pub fn drive_mut<V: VisitAstMut>(&mut self, visitor: &mut V) -> ControlFlow<V::Break> {
match self {
AnyTransItemMut::Type(d) => visitor.visit(*d),
AnyTransItemMut::Fun(d) => visitor.visit(*d),
AnyTransItemMut::Global(d) => visitor.visit(*d),
AnyTransItemMut::TraitDecl(d) => visitor.visit(*d),
AnyTransItemMut::TraitImpl(d) => visitor.visit(*d),
}
}
}
impl fmt::Display for TranslatedCrate {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let fmt: FmtCtx = self.into_fmt();
match &self.ordered_decls {
None => {
for d in &self.type_decls {
writeln!(f, "{}\n", fmt.format_object(d))?
}
for d in &self.global_decls {
writeln!(f, "{}\n", fmt.format_object(d))?
}
for d in &self.trait_decls {
writeln!(f, "{}\n", fmt.format_object(d))?
}
for d in &self.trait_impls {
writeln!(f, "{}\n", fmt.format_object(d))?
}
for d in &self.fun_decls {
writeln!(f, "{}\n", fmt.format_object(d))?
}
}
Some(ordered_decls) => {
for gr in ordered_decls {
for id in gr.get_ids() {
writeln!(f, "{}\n", fmt.format_decl_id(id))?
}
}
}
}
fmt::Result::Ok(())
}
}
impl<'tcx, 'ctx, 'a> IntoFormatter for &'a TranslatedCrate {
type C = FmtCtx<'a>;
fn into_fmt(self) -> Self::C {
FmtCtx {
translated: Some(self),
..Default::default()
}
}
}
macro_rules! mk_index_impls {
($ty:ident.$field:ident[$idx:ty]: $output:ty) => {
impl std::ops::Index<$idx> for $ty {
type Output = $output;
fn index(&self, index: $idx) -> &Self::Output {
&self.$field[index]
}
}
impl std::ops::IndexMut<$idx> for $ty {
fn index_mut(&mut self, index: $idx) -> &mut Self::Output {
&mut self.$field[index]
}
}
};
}
pub(crate) use mk_index_impls;
mk_index_impls!(TranslatedCrate.type_decls[TypeDeclId]: TypeDecl);
mk_index_impls!(TranslatedCrate.fun_decls[FunDeclId]: FunDecl);
mk_index_impls!(TranslatedCrate.global_decls[GlobalDeclId]: GlobalDecl);
mk_index_impls!(TranslatedCrate.trait_decls[TraitDeclId]: TraitDecl);
mk_index_impls!(TranslatedCrate.trait_impls[TraitImplId]: TraitImpl);