use std::rc::Rc;
use debris_common::{CodeId, Ident, InputFiles, Span, SpecialIdent};
use super::{IdentifierPath, SpannedIdentifier};
#[derive(Debug, Eq, PartialEq)]
pub enum HirConstValue {
Integer {
span: Span,
value: i32,
},
Bool {
span: Span,
value: bool,
},
Fixed {
span: Span,
value: i32,
},
String {
span: Span,
value: Rc<str>,
},
FormatString {
span: Span,
value: Vec<HirFormatStringMember>,
},
}
#[derive(Debug, PartialEq, Eq)]
pub enum HirFormatStringMember {
String(Rc<str>),
Variable(Box<HirExpression>),
}
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
pub enum HirComparisonOperator {
Eq,
Ne,
Gt,
Ge,
Lt,
Le,
}
#[derive(Debug, Eq, PartialEq, Clone)]
pub enum HirInfixOperator {
Comparison(HirComparisonOperator),
And,
Or,
Plus,
Minus,
Times,
Divide,
Modulo,
}
#[derive(Debug, Eq, PartialEq, Clone)]
pub struct HirInfix {
pub span: Span,
pub operator: HirInfixOperator,
}
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
pub enum HirPrefixOperator {
Minus,
Not,
}
#[derive(Debug, Eq, PartialEq, Clone)]
pub struct HirPrefix {
pub span: Span,
pub operator: HirPrefixOperator,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirImport {
pub span: Span,
pub ident: SpannedIdentifier,
pub id: usize,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum HirControlKind {
Return,
Break,
Continue,
}
impl HirControlKind {
pub fn returns(self) -> bool {
match self {
HirControlKind::Return | HirControlKind::Break => true,
HirControlKind::Continue => false,
}
}
pub fn takes_value(self) -> bool {
match self {
HirControlKind::Return | HirControlKind::Break => true,
HirControlKind::Continue => false,
}
}
}
impl std::fmt::Display for HirControlKind {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
HirControlKind::Break => write!(f, "break"),
HirControlKind::Continue => write!(f, "continue"),
HirControlKind::Return => write!(f, "return"),
}
}
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirControlFlow {
pub span: Span,
pub kind: HirControlKind,
pub expression: Option<Box<HirExpression>>,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirInfiniteLoop {
pub span: Span,
pub block: Box<HirBlock>,
}
#[derive(Debug, PartialEq, Eq, Clone)]
pub struct HirParameterDeclaration {
pub span: Span,
pub ident: SpannedIdentifier,
pub typ: HirTypePattern,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum HirDeclarationMode {
Let,
Comptime,
}
#[derive(Debug, PartialEq, Eq)]
pub enum HirVariablePattern {
Path(IdentifierPath),
Tuple(Vec<HirVariablePattern>),
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirVariableInitialization {
pub span: Span,
pub pattern: HirVariablePattern,
pub value: Box<HirExpression>,
pub mode: HirDeclarationMode,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirVariableUpdate {
pub span: Span,
pub pattern: HirVariablePattern,
pub value: Box<HirExpression>,
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirPropertyDeclaration {
pub span: Span,
pub ident: SpannedIdentifier,
pub datatype: HirTypePattern,
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirFunctionCall {
pub span: Span,
pub value: Box<HirExpression>,
pub parameters_span: Span,
pub parameters: Vec<HirExpression>,
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirConditionalBranch {
pub span: Span,
pub is_comptime: bool,
pub condition: Box<HirExpression>,
pub block_positive: Box<HirBlock>,
pub block_negative: Option<Box<HirBlock>>,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirStructInitialization {
pub span: Span,
pub base: Box<HirExpression>,
pub values: Vec<(SpannedIdentifier, HirExpression)>,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirTupleInitialization {
pub span: Span,
pub values: Vec<HirExpression>,
}
#[derive(Debug, Eq, PartialEq)]
pub enum HirExpression {
Variable(SpannedIdentifier),
Path(IdentifierPath),
PropertyAccess {
lhs: Box<HirExpression>,
rhs: SpannedIdentifier,
},
Value(HirConstValue),
UnaryOperation {
operation: HirPrefix,
value: Box<HirExpression>,
},
BinaryOperation {
operation: HirInfix,
lhs: Box<HirExpression>,
rhs: Box<HirExpression>,
},
Block(HirBlock),
Function(HirFunction),
FunctionCall(HirFunctionCall),
ConditionalBranch(HirConditionalBranch),
StructInitialization(HirStructInitialization),
TupleInitialization(HirTupleInitialization),
InfiniteLoop(HirInfiniteLoop),
ControlFlow(HirControlFlow),
}
#[derive(Debug, Eq, PartialEq)]
pub enum HirStatement {
VariableDecl(HirVariableInitialization),
VariableUpdate(HirVariableUpdate),
ControlFlow(HirControlFlow),
Block(HirBlock),
ConditionalBranch(HirConditionalBranch),
InfiniteLoop(HirInfiniteLoop),
Expression(HirExpression),
}
#[derive(Debug, PartialEq, Eq, Clone)]
pub enum HirTypePattern {
Path(IdentifierPath),
Function {
span: Span,
parameters: Vec<HirTypePattern>,
return_type: Option<Box<HirTypePattern>>,
},
Tuple {
span: Span,
values: Vec<HirTypePattern>,
},
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirBlock {
pub span: Span,
pub statements: Vec<HirStatement>,
pub return_value: Option<Box<HirExpression>>,
pub objects: Vec<HirObject>,
}
#[derive(Debug, PartialEq, Eq)]
pub struct Attribute {
pub expression: HirExpression,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum HirFunctionName {
Ident(SpannedIdentifier),
Unnamed { file: CodeId, pos: Span },
}
impl HirFunctionName {
pub fn span(&self) -> Span {
match self {
HirFunctionName::Ident(ident) => ident.span,
HirFunctionName::Unnamed { pos, .. } => *pos,
}
}
pub fn spanned_ident(&self) -> Option<SpannedIdentifier> {
match self {
HirFunctionName::Ident(spanned_ident) => Some(*spanned_ident),
HirFunctionName::Unnamed { .. } => None,
}
}
pub fn is_unknown(&self) -> bool {
matches!(self, HirFunctionName::Unnamed { .. })
}
pub fn into_ident(&self, input_files: &InputFiles) -> Ident {
match self {
HirFunctionName::Ident(span) => input_files.get_span_str(span.span).into(),
HirFunctionName::Unnamed { file, pos } => {
let code_ref = input_files.get_code_ref(*file);
let file_name = code_ref
.get_code()
.path
.as_ref()
.map_or_else(|| file.to_string(), ToString::to_string);
let (line, col) = input_files.line_col(*pos);
let line = line + 1;
let col = col + 1;
Ident::Value(format!("<'{file_name}:{line}:{col}'>").into())
}
}
}
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirFunction {
pub span: Span,
pub signature_span: Span,
pub parameter_span: Span,
pub is_comptime: bool,
pub attributes: Vec<Attribute>,
pub ident: HirFunctionName,
pub block: HirBlock,
pub parameters: Vec<HirParameterDeclaration>,
pub return_type: Option<HirTypePattern>,
}
#[derive(Debug, Eq, PartialEq)]
pub struct HirStruct {
pub span: Span,
pub ident: SpannedIdentifier,
pub attributes: Vec<Attribute>,
pub properties: Vec<HirPropertyDeclaration>,
pub objects: Vec<HirObject>,
}
#[derive(Debug, PartialEq, Eq)]
pub struct HirModule {
pub span: Span,
pub attributes: Vec<Attribute>,
pub ident: SpannedIdentifier,
pub block: HirBlock,
}
#[derive(Debug, PartialEq, Eq)]
pub enum HirObject {
Function(HirFunction),
Struct(HirStruct),
Module(HirModule),
Import(HirImport),
}
impl HirObject {
pub fn spanned_ident(&self) -> Option<SpannedIdentifier> {
Some(match self {
HirObject::Function(func) => func.ident.spanned_ident()?,
HirObject::Struct(strukt) => strukt.ident,
HirObject::Module(module) => module.ident,
HirObject::Import(import) => import.ident,
})
}
}
#[derive(Debug, Eq, PartialEq)]
pub enum HirItem {
Object(HirObject),
Statement(HirStatement),
}
impl HirConstValue {
pub fn span(&self) -> Span {
match self {
HirConstValue::Fixed { span, .. }
| HirConstValue::Integer { span, .. }
| HirConstValue::Bool { span, .. }
| HirConstValue::String { span, .. }
| HirConstValue::FormatString { span, .. } => *span,
}
}
}
impl HirComparisonOperator {
pub fn get_raw_special_ident(&self) -> SpecialIdent {
use HirComparisonOperator::*;
match self {
Eq => SpecialIdent::CmpEq,
Ne => SpecialIdent::CmpNe,
Gt => SpecialIdent::CmpGt,
Ge => SpecialIdent::CmpGe,
Lt => SpecialIdent::CmpLt,
Le => SpecialIdent::CmpLe,
}
}
}
impl HirInfixOperator {
pub fn get_special_ident(&self) -> SpecialIdent {
use HirInfixOperator::*;
match self {
And => SpecialIdent::And,
Or => SpecialIdent::Or,
Plus => SpecialIdent::Add,
Minus => SpecialIdent::Sub,
Times => SpecialIdent::Mul,
Divide => SpecialIdent::Div,
Modulo => SpecialIdent::Mod,
Comparison(value) => value.get_raw_special_ident(),
}
}
}
impl HirPrefixOperator {
pub fn get_ident(&self) -> Ident {
Ident::Special(match self {
HirPrefixOperator::Minus => SpecialIdent::UnaryMinus,
HirPrefixOperator::Not => SpecialIdent::Not,
})
}
}
impl HirBlock {
pub fn last_item_span(&self) -> Span {
if let Some(value) = &self.return_value {
value.span()
} else if let [.., last] = self.statements.as_slice() {
last.span()
} else {
self.span
}
}
}
impl HirVariablePattern {
pub fn span(&self) -> Span {
match self {
HirVariablePattern::Path(path) => path.span(),
HirVariablePattern::Tuple(tuple) => match tuple.as_slice() {
[] => Span::EMPTY,
[single] => single.span(),
[first, .., last] => first.span().until(last.span()),
},
}
}
}
impl HirExpression {
pub fn span(&self) -> Span {
match self {
HirExpression::Value(number) => number.span(),
HirExpression::Variable(var) => var.span,
HirExpression::Path(path) => path.span(),
HirExpression::PropertyAccess { lhs, rhs } => lhs.span().until(rhs.span),
HirExpression::BinaryOperation {
lhs,
operation: _,
rhs,
} => lhs.span().until(rhs.span()),
HirExpression::UnaryOperation { operation, value } => {
operation.span.until(value.span())
}
HirExpression::Block(block) => block.span,
HirExpression::Function(function) => function.span,
HirExpression::FunctionCall(call) => call.span,
HirExpression::ConditionalBranch(branch) => branch.span,
HirExpression::StructInitialization(struct_instantiation) => struct_instantiation.span,
HirExpression::TupleInitialization(tuple_initialization) => tuple_initialization.span,
HirExpression::InfiniteLoop(inf_loop) => inf_loop.span,
HirExpression::ControlFlow(control_flow) => control_flow.span,
}
}
}
impl HirStatement {
pub fn span(&self) -> Span {
match self {
HirStatement::VariableDecl(var_decl) => var_decl.span,
HirStatement::VariableUpdate(var_update) => var_update.span,
HirStatement::ControlFlow(control_flow) => control_flow.span,
HirStatement::Block(block) => block.span,
HirStatement::ConditionalBranch(branch) => branch.span,
HirStatement::InfiniteLoop(inf_loop) => inf_loop.span,
HirStatement::Expression(expr) => expr.span(),
}
}
}
impl HirTypePattern {
pub fn span(&self) -> Span {
match self {
HirTypePattern::Function { span, .. } | HirTypePattern::Tuple { span, .. } => *span,
HirTypePattern::Path(path) => path.span(),
}
}
}
impl Attribute {
pub fn span(&self) -> Span {
self.expression.span()
}
}
impl HirFunction {
pub fn return_type_span(&self) -> Span {
match &self.return_type {
Some(path) => path.span(),
None => Span::new(self.parameter_span.end(), 1),
}
}
}