Organize compiler and clean up type checking
This commit is contained in:
parent
e2a462c3f6
commit
2df86c58bf
@ -21,11 +21,13 @@
|
|||||||
mod error;
|
mod error;
|
||||||
mod optimize;
|
mod optimize;
|
||||||
mod parse_rule;
|
mod parse_rule;
|
||||||
|
mod type_checks;
|
||||||
|
|
||||||
pub use error::CompileError;
|
pub use error::CompileError;
|
||||||
use parse_rule::{ParseRule, Precedence};
|
use parse_rule::{ParseRule, Precedence};
|
||||||
|
use type_checks::{check_math_type, check_math_types};
|
||||||
|
|
||||||
use std::mem::{replace, swap};
|
use std::mem::replace;
|
||||||
|
|
||||||
use colored::Colorize;
|
use colored::Colorize;
|
||||||
use optimize::control_flow_register_consolidation;
|
use optimize::control_flow_register_consolidation;
|
||||||
@ -754,6 +756,8 @@ impl<'src> Compiler<'src> {
|
|||||||
| Token::PercentEqual
|
| Token::PercentEqual
|
||||||
);
|
);
|
||||||
|
|
||||||
|
check_math_type(&left_type, operator, &left_position)?;
|
||||||
|
|
||||||
if is_assignment && !left_is_mutable_local {
|
if is_assignment && !left_is_mutable_local {
|
||||||
return Err(CompileError::ExpectedMutableVariable {
|
return Err(CompileError::ExpectedMutableVariable {
|
||||||
found: self.previous_token.to_owned(),
|
found: self.previous_token.to_owned(),
|
||||||
@ -767,59 +771,14 @@ impl<'src> Compiler<'src> {
|
|||||||
let (right_instruction, right_type, right_position) = self.instructions.pop().unwrap();
|
let (right_instruction, right_type, right_position) = self.instructions.pop().unwrap();
|
||||||
let (right, push_back_right) = self.handle_binary_argument(&right_instruction)?;
|
let (right, push_back_right) = self.handle_binary_argument(&right_instruction)?;
|
||||||
|
|
||||||
match operator {
|
check_math_type(&right_type, operator, &right_position)?;
|
||||||
Token::Plus | Token::PlusEqual => {
|
check_math_types(
|
||||||
Compiler::expect_addable_type(&left_type, &left_position)?;
|
&left_type,
|
||||||
Compiler::expect_addable_type(&right_type, &right_position)?;
|
&left_position,
|
||||||
Compiler::expect_addable_types(
|
operator,
|
||||||
&left_type,
|
&right_type,
|
||||||
&left_position,
|
&right_position,
|
||||||
&right_type,
|
)?;
|
||||||
&right_position,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
Token::Minus | Token::MinusEqual => {
|
|
||||||
Compiler::expect_subtractable_type(&left_type, &left_position)?;
|
|
||||||
Compiler::expect_subtractable_type(&right_type, &right_position)?;
|
|
||||||
Compiler::expect_subtractable_types(
|
|
||||||
&left_type,
|
|
||||||
&left_position,
|
|
||||||
&right_type,
|
|
||||||
&right_position,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
Token::Slash | Token::SlashEqual => {
|
|
||||||
Compiler::expect_dividable_type(&left_type, &left_position)?;
|
|
||||||
Compiler::expect_dividable_type(&right_type, &right_position)?;
|
|
||||||
Compiler::expect_dividable_types(
|
|
||||||
&left_type,
|
|
||||||
&left_position,
|
|
||||||
&right_type,
|
|
||||||
&right_position,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
Token::Star | Token::StarEqual => {
|
|
||||||
Compiler::expect_multipliable_type(&left_type, &left_position)?;
|
|
||||||
Compiler::expect_multipliable_type(&right_type, &right_position)?;
|
|
||||||
Compiler::expect_multipliable_types(
|
|
||||||
&left_type,
|
|
||||||
&left_position,
|
|
||||||
&right_type,
|
|
||||||
&right_position,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
Token::Percent | Token::PercentEqual => {
|
|
||||||
Compiler::expect_modulable_type(&left_type, &left_position)?;
|
|
||||||
Compiler::expect_modulable_type(&right_type, &right_position)?;
|
|
||||||
Compiler::expect_modulable_types(
|
|
||||||
&left_type,
|
|
||||||
&left_position,
|
|
||||||
&right_type,
|
|
||||||
&right_position,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
|
|
||||||
if push_back_right {
|
if push_back_right {
|
||||||
self.instructions
|
self.instructions
|
||||||
@ -1756,169 +1715,4 @@ impl<'src> Compiler<'src> {
|
|||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn expect_addable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
argument_type,
|
|
||||||
Type::Byte | Type::Character | Type::Float | Type::Integer | Type::String
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotAddType {
|
|
||||||
argument_type: argument_type.clone(),
|
|
||||||
position: *position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_addable_types(
|
|
||||||
left: &Type,
|
|
||||||
left_position: &Span,
|
|
||||||
right: &Type,
|
|
||||||
right_position: &Span,
|
|
||||||
) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
(left, right),
|
|
||||||
(Type::Byte, Type::Byte)
|
|
||||||
| (Type::Character, Type::String)
|
|
||||||
| (Type::Character, Type::Character)
|
|
||||||
| (Type::Float, Type::Float)
|
|
||||||
| (Type::Integer, Type::Integer)
|
|
||||||
| (Type::String, Type::Character)
|
|
||||||
| (Type::String, Type::String),
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotAddArguments {
|
|
||||||
left_type: left.clone(),
|
|
||||||
left_position: *left_position,
|
|
||||||
right_type: right.clone(),
|
|
||||||
right_position: *right_position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_dividable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
|
||||||
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotDivideType {
|
|
||||||
argument_type: argument_type.clone(),
|
|
||||||
position: *position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_dividable_types(
|
|
||||||
left: &Type,
|
|
||||||
left_position: &Span,
|
|
||||||
right: &Type,
|
|
||||||
right_position: &Span,
|
|
||||||
) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
(left, right),
|
|
||||||
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotDivideArguments {
|
|
||||||
left_type: left.clone(),
|
|
||||||
right_type: right.clone(),
|
|
||||||
position: Span(left_position.0, right_position.1),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_modulable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
|
||||||
if matches!(argument_type, Type::Byte | Type::Integer | Type::Float) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotModuloType {
|
|
||||||
argument_type: argument_type.clone(),
|
|
||||||
position: *position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_modulable_types(
|
|
||||||
left: &Type,
|
|
||||||
left_position: &Span,
|
|
||||||
right: &Type,
|
|
||||||
right_position: &Span,
|
|
||||||
) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
(left, right),
|
|
||||||
(Type::Byte, Type::Byte) | (Type::Integer, Type::Integer) | (Type::Float, Type::Float)
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotModuloArguments {
|
|
||||||
left_type: left.clone(),
|
|
||||||
right_type: right.clone(),
|
|
||||||
position: Span(left_position.0, right_position.1),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_multipliable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
|
||||||
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotMultiplyType {
|
|
||||||
argument_type: argument_type.clone(),
|
|
||||||
position: *position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_multipliable_types(
|
|
||||||
left: &Type,
|
|
||||||
left_position: &Span,
|
|
||||||
right: &Type,
|
|
||||||
right_position: &Span,
|
|
||||||
) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
(left, right),
|
|
||||||
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotMultiplyArguments {
|
|
||||||
left_type: left.clone(),
|
|
||||||
right_type: right.clone(),
|
|
||||||
position: Span(left_position.0, right_position.1),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_subtractable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
|
||||||
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotSubtractType {
|
|
||||||
argument_type: argument_type.clone(),
|
|
||||||
position: *position,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn expect_subtractable_types(
|
|
||||||
left: &Type,
|
|
||||||
left_position: &Span,
|
|
||||||
right: &Type,
|
|
||||||
right_position: &Span,
|
|
||||||
) -> Result<(), CompileError> {
|
|
||||||
if matches!(
|
|
||||||
(left, right),
|
|
||||||
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
|
||||||
) {
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
Err(CompileError::CannotSubtractArguments {
|
|
||||||
left_type: left.clone(),
|
|
||||||
right_type: right.clone(),
|
|
||||||
position: Span(left_position.0, right_position.1),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
200
dust-lang/src/compiler/type_checks.rs
Normal file
200
dust-lang/src/compiler/type_checks.rs
Normal file
@ -0,0 +1,200 @@
|
|||||||
|
use crate::{Span, Token, Type};
|
||||||
|
|
||||||
|
use super::CompileError;
|
||||||
|
|
||||||
|
pub fn check_math_type(
|
||||||
|
r#type: &Type,
|
||||||
|
operator: Token,
|
||||||
|
position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
match operator {
|
||||||
|
Token::Plus => expect_addable_type(r#type, position),
|
||||||
|
Token::Minus => expect_subtractable_type(r#type, position),
|
||||||
|
Token::Star => expect_multipliable_type(r#type, position),
|
||||||
|
Token::Slash => expect_dividable_type(r#type, position),
|
||||||
|
Token::Percent => expect_modulable_type(r#type, position),
|
||||||
|
_ => Ok(()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn check_math_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
operator: Token,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
match operator {
|
||||||
|
Token::Plus => expect_addable_types(left, left_position, right, right_position),
|
||||||
|
Token::Minus => expect_subtractable_types(left, left_position, right, right_position),
|
||||||
|
Token::Star => expect_multipliable_types(left, left_position, right, right_position),
|
||||||
|
Token::Slash => expect_dividable_types(left, left_position, right, right_position),
|
||||||
|
Token::Percent => expect_modulable_types(left, left_position, right, right_position),
|
||||||
|
_ => Ok(()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_addable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
argument_type,
|
||||||
|
Type::Byte | Type::Character | Type::Float | Type::Integer | Type::String
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotAddType {
|
||||||
|
argument_type: argument_type.clone(),
|
||||||
|
position: *position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_addable_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
(left, right),
|
||||||
|
(Type::Byte, Type::Byte)
|
||||||
|
| (Type::Character, Type::String)
|
||||||
|
| (Type::Character, Type::Character)
|
||||||
|
| (Type::Float, Type::Float)
|
||||||
|
| (Type::Integer, Type::Integer)
|
||||||
|
| (Type::String, Type::Character)
|
||||||
|
| (Type::String, Type::String),
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotAddArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
left_position: *left_position,
|
||||||
|
right_type: right.clone(),
|
||||||
|
right_position: *right_position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_dividable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
||||||
|
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotDivideType {
|
||||||
|
argument_type: argument_type.clone(),
|
||||||
|
position: *position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_dividable_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
(left, right),
|
||||||
|
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotDivideArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
right_type: right.clone(),
|
||||||
|
position: Span(left_position.0, right_position.1),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_modulable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
||||||
|
if matches!(argument_type, Type::Byte | Type::Integer | Type::Float) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotModuloType {
|
||||||
|
argument_type: argument_type.clone(),
|
||||||
|
position: *position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_modulable_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
(left, right),
|
||||||
|
(Type::Byte, Type::Byte) | (Type::Integer, Type::Integer) | (Type::Float, Type::Float)
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotModuloArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
right_type: right.clone(),
|
||||||
|
position: Span(left_position.0, right_position.1),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_multipliable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
||||||
|
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotMultiplyType {
|
||||||
|
argument_type: argument_type.clone(),
|
||||||
|
position: *position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_multipliable_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
(left, right),
|
||||||
|
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotMultiplyArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
right_type: right.clone(),
|
||||||
|
position: Span(left_position.0, right_position.1),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_subtractable_type(argument_type: &Type, position: &Span) -> Result<(), CompileError> {
|
||||||
|
if matches!(argument_type, Type::Byte | Type::Float | Type::Integer) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotSubtractType {
|
||||||
|
argument_type: argument_type.clone(),
|
||||||
|
position: *position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn expect_subtractable_types(
|
||||||
|
left: &Type,
|
||||||
|
left_position: &Span,
|
||||||
|
right: &Type,
|
||||||
|
right_position: &Span,
|
||||||
|
) -> Result<(), CompileError> {
|
||||||
|
if matches!(
|
||||||
|
(left, right),
|
||||||
|
(Type::Byte, Type::Byte) | (Type::Float, Type::Float) | (Type::Integer, Type::Integer)
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotSubtractArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
right_type: right.clone(),
|
||||||
|
position: Span(left_position.0, right_position.1),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user