Begin adding new tests
This commit is contained in:
parent
636bd7ce5b
commit
672f10e15a
@ -131,7 +131,7 @@ impl<'src> Compiler<'src> {
|
|||||||
loop {
|
loop {
|
||||||
self.parse(Precedence::None)?;
|
self.parse(Precedence::None)?;
|
||||||
|
|
||||||
if self.is_eof() || self.allow(Token::RightBrace)? {
|
if matches!(self.current_token, Token::Eof | Token::RightBrace) {
|
||||||
self.parse_implicit_return()?;
|
self.parse_implicit_return()?;
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -150,7 +150,9 @@ impl<'src> Compiler<'src> {
|
|||||||
.iter()
|
.iter()
|
||||||
.rev()
|
.rev()
|
||||||
.find_map(|(instruction, _, _)| {
|
.find_map(|(instruction, _, _)| {
|
||||||
if instruction.yields_value() {
|
let is_get_local = matches!(instruction.operation(), Operation::GetLocal);
|
||||||
|
|
||||||
|
if instruction.yields_value() && !is_get_local {
|
||||||
Some(instruction.a() + 1)
|
Some(instruction.a() + 1)
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
@ -627,11 +629,15 @@ impl<'src> Compiler<'src> {
|
|||||||
};
|
};
|
||||||
let operator = self.current_token;
|
let operator = self.current_token;
|
||||||
let operator_position = self.current_position;
|
let operator_position = self.current_position;
|
||||||
|
|
||||||
|
Compiler::expect_addable_type(&left_type, &left_position)?;
|
||||||
|
|
||||||
let rule = ParseRule::from(&operator);
|
let rule = ParseRule::from(&operator);
|
||||||
let is_assignment = matches!(
|
let is_assignment = matches!(
|
||||||
operator,
|
operator,
|
||||||
Token::PlusEqual | Token::MinusEqual | Token::StarEqual | Token::SlashEqual
|
Token::PlusEqual | Token::MinusEqual | Token::StarEqual | Token::SlashEqual
|
||||||
);
|
);
|
||||||
|
|
||||||
let r#type = if is_assignment {
|
let r#type = if is_assignment {
|
||||||
Type::None
|
Type::None
|
||||||
} else {
|
} else {
|
||||||
@ -647,7 +653,7 @@ impl<'src> Compiler<'src> {
|
|||||||
|
|
||||||
if push_back_left {
|
if push_back_left {
|
||||||
self.instructions
|
self.instructions
|
||||||
.push((left_instruction, left_type, left_position));
|
.push((left_instruction, left_type.clone(), left_position));
|
||||||
}
|
}
|
||||||
|
|
||||||
self.advance()?;
|
self.advance()?;
|
||||||
@ -656,6 +662,9 @@ impl<'src> Compiler<'src> {
|
|||||||
let (right_instruction, right_type, right_position) = self.pop_last_instruction()?;
|
let (right_instruction, right_type, right_position) = self.pop_last_instruction()?;
|
||||||
let (right, push_back_right) = self.handle_binary_argument(&right_instruction)?;
|
let (right, push_back_right) = self.handle_binary_argument(&right_instruction)?;
|
||||||
|
|
||||||
|
Compiler::expect_addable_type(&right_type, &right_position)?;
|
||||||
|
Compiler::expect_addable_types(&left_type, &left_position, &right_type, &right_position)?;
|
||||||
|
|
||||||
if push_back_right {
|
if push_back_right {
|
||||||
self.instructions
|
self.instructions
|
||||||
.push((right_instruction, right_type, right_position));
|
.push((right_instruction, right_type, right_position));
|
||||||
@ -986,7 +995,7 @@ impl<'src> Compiler<'src> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let destination = Destination::Register(self.next_register());
|
let destination = Destination::Register(self.next_register());
|
||||||
let end = self.current_position.1;
|
let end = self.previous_position.1;
|
||||||
let load_list = Instruction::from(LoadList {
|
let load_list = Instruction::from(LoadList {
|
||||||
destination,
|
destination,
|
||||||
start_register,
|
start_register,
|
||||||
@ -1357,7 +1366,7 @@ impl<'src> Compiler<'src> {
|
|||||||
|
|
||||||
let mut value_parameters: Option<Vec<(u16, Type)>> = None;
|
let mut value_parameters: Option<Vec<(u16, Type)>> = None;
|
||||||
|
|
||||||
while function_compiler.current_token != Token::RightParenthesis {
|
while !function_compiler.allow(Token::RightParenthesis)? {
|
||||||
let is_mutable = function_compiler.allow(Token::Mut)?;
|
let is_mutable = function_compiler.allow(Token::Mut)?;
|
||||||
let parameter = if let Token::Identifier(text) = function_compiler.current_token {
|
let parameter = if let Token::Identifier(text) = function_compiler.current_token {
|
||||||
function_compiler.advance()?;
|
function_compiler.advance()?;
|
||||||
@ -1398,8 +1407,6 @@ impl<'src> Compiler<'src> {
|
|||||||
function_compiler.allow(Token::Comma)?;
|
function_compiler.allow(Token::Comma)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
function_compiler.advance()?;
|
|
||||||
|
|
||||||
let return_type = if function_compiler.allow(Token::ArrowThin)? {
|
let return_type = if function_compiler.allow(Token::ArrowThin)? {
|
||||||
let r#type = function_compiler.parse_type_from(
|
let r#type = function_compiler.parse_type_from(
|
||||||
function_compiler.current_token,
|
function_compiler.current_token,
|
||||||
@ -1417,6 +1424,9 @@ impl<'src> Compiler<'src> {
|
|||||||
|
|
||||||
function_compiler.expect(Token::LeftBrace)?;
|
function_compiler.expect(Token::LeftBrace)?;
|
||||||
function_compiler.compile()?;
|
function_compiler.compile()?;
|
||||||
|
function_compiler.expect(Token::RightBrace)?;
|
||||||
|
|
||||||
|
let function_end = function_compiler.previous_position.1;
|
||||||
|
|
||||||
self.previous_token = function_compiler.previous_token;
|
self.previous_token = function_compiler.previous_token;
|
||||||
self.previous_position = function_compiler.previous_position;
|
self.previous_position = function_compiler.previous_position;
|
||||||
@ -1426,7 +1436,6 @@ impl<'src> Compiler<'src> {
|
|||||||
let function =
|
let function =
|
||||||
ConcreteValue::Function(function_compiler.finish(None, value_parameters.clone()));
|
ConcreteValue::Function(function_compiler.finish(None, value_parameters.clone()));
|
||||||
let constant_index = self.push_or_get_constant(function);
|
let constant_index = self.push_or_get_constant(function);
|
||||||
let function_end = self.current_position.1;
|
|
||||||
let register = self.next_register();
|
let register = self.next_register();
|
||||||
let function_type = FunctionType {
|
let function_type = FunctionType {
|
||||||
type_parameters: None,
|
type_parameters: None,
|
||||||
@ -1514,6 +1523,7 @@ impl<'src> Compiler<'src> {
|
|||||||
let start = self.current_position.0;
|
let start = self.current_position.0;
|
||||||
|
|
||||||
self.advance()?;
|
self.advance()?;
|
||||||
|
self.expect(Token::LeftParenthesis)?;
|
||||||
|
|
||||||
let mut argument_count = 0;
|
let mut argument_count = 0;
|
||||||
|
|
||||||
@ -1595,6 +1605,42 @@ 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::Integer, Type::Integer)
|
||||||
|
| (Type::Float, Type::Float)
|
||||||
|
| (Type::String, Type::String),
|
||||||
|
) {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(CompileError::CannotAddArguments {
|
||||||
|
left_type: left.clone(),
|
||||||
|
right_type: right.clone(),
|
||||||
|
position: Span(left_position.0, right_position.1),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Operator precedence levels.
|
/// Operator precedence levels.
|
||||||
@ -1977,6 +2023,24 @@ pub enum CompileError {
|
|||||||
},
|
},
|
||||||
|
|
||||||
// Type errors
|
// Type errors
|
||||||
|
CannotAddType {
|
||||||
|
argument_type: Type,
|
||||||
|
position: Span,
|
||||||
|
},
|
||||||
|
CannotAddArguments {
|
||||||
|
left_type: Type,
|
||||||
|
right_type: Type,
|
||||||
|
position: Span,
|
||||||
|
},
|
||||||
|
CannotSubtractLeft {
|
||||||
|
argument_type: Type,
|
||||||
|
position: Span,
|
||||||
|
},
|
||||||
|
CannotSubtract {
|
||||||
|
left_type: Type,
|
||||||
|
right_type: Type,
|
||||||
|
position: Span,
|
||||||
|
},
|
||||||
CannotResolveRegisterType {
|
CannotResolveRegisterType {
|
||||||
register_index: usize,
|
register_index: usize,
|
||||||
position: Span,
|
position: Span,
|
||||||
@ -2034,10 +2098,14 @@ impl AnnotatedError for CompileError {
|
|||||||
|
|
||||||
fn description(&self) -> &'static str {
|
fn description(&self) -> &'static str {
|
||||||
match self {
|
match self {
|
||||||
|
Self::CannotAddArguments { .. } => "Cannot add these types",
|
||||||
|
Self::CannotAddType { .. } => "Cannot add to this type",
|
||||||
Self::CannotChainComparison { .. } => "Cannot chain comparison operations",
|
Self::CannotChainComparison { .. } => "Cannot chain comparison operations",
|
||||||
Self::CannotMutateImmutableVariable { .. } => "Cannot mutate immutable variable",
|
Self::CannotMutateImmutableVariable { .. } => "Cannot mutate immutable variable",
|
||||||
Self::CannotResolveRegisterType { .. } => "Cannot resolve register type",
|
Self::CannotResolveRegisterType { .. } => "Cannot resolve register type",
|
||||||
Self::CannotResolveVariableType { .. } => "Cannot resolve type",
|
Self::CannotResolveVariableType { .. } => "Cannot resolve type",
|
||||||
|
Self::CannotSubtract { .. } => "Cannot subtract these types",
|
||||||
|
Self::CannotSubtractLeft { .. } => "Cannot subtract from this type",
|
||||||
Self::ConstantIndexOutOfBounds { .. } => "Constant index out of bounds",
|
Self::ConstantIndexOutOfBounds { .. } => "Constant index out of bounds",
|
||||||
Self::ExpectedExpression { .. } => "Expected an expression",
|
Self::ExpectedExpression { .. } => "Expected an expression",
|
||||||
Self::ExpectedFunction { .. } => "Expected a function",
|
Self::ExpectedFunction { .. } => "Expected a function",
|
||||||
@ -2133,10 +2201,14 @@ impl AnnotatedError for CompileError {
|
|||||||
|
|
||||||
fn position(&self) -> Span {
|
fn position(&self) -> Span {
|
||||||
match self {
|
match self {
|
||||||
|
Self::CannotAddArguments { position, .. } => *position,
|
||||||
|
Self::CannotAddType { position, .. } => *position,
|
||||||
Self::CannotChainComparison { position } => *position,
|
Self::CannotChainComparison { position } => *position,
|
||||||
Self::CannotMutateImmutableVariable { position, .. } => *position,
|
Self::CannotMutateImmutableVariable { position, .. } => *position,
|
||||||
Self::CannotResolveRegisterType { position, .. } => *position,
|
Self::CannotResolveRegisterType { position, .. } => *position,
|
||||||
Self::CannotResolveVariableType { position, .. } => *position,
|
Self::CannotResolveVariableType { position, .. } => *position,
|
||||||
|
Self::CannotSubtract { position, .. } => *position,
|
||||||
|
Self::CannotSubtractLeft { position, .. } => *position,
|
||||||
Self::ConstantIndexOutOfBounds { position, .. } => *position,
|
Self::ConstantIndexOutOfBounds { position, .. } => *position,
|
||||||
Self::ExpectedExpression { position, .. } => *position,
|
Self::ExpectedExpression { position, .. } => *position,
|
||||||
Self::ExpectedFunction { position, .. } => *position,
|
Self::ExpectedFunction { position, .. } => *position,
|
||||||
|
@ -26,7 +26,7 @@ fn function() {
|
|||||||
),
|
),
|
||||||
Span(30, 31)
|
Span(30, 31)
|
||||||
),
|
),
|
||||||
(Instruction::r#return(true), Span(35, 35)),
|
(Instruction::r#return(true), Span(34, 35)),
|
||||||
],
|
],
|
||||||
vec![ConcreteValue::string("a"), ConcreteValue::string("b"),],
|
vec![ConcreteValue::string("a"), ConcreteValue::string("b"),],
|
||||||
vec![
|
vec![
|
||||||
@ -53,7 +53,7 @@ fn function_call() {
|
|||||||
vec![
|
vec![
|
||||||
(
|
(
|
||||||
Instruction::load_constant(Destination::Register(0), 0, false),
|
Instruction::load_constant(Destination::Register(0), 0, false),
|
||||||
Span(0, 36)
|
Span(0, 35)
|
||||||
),
|
),
|
||||||
(
|
(
|
||||||
Instruction::load_constant(Destination::Register(1), 1, false),
|
Instruction::load_constant(Destination::Register(1), 1, false),
|
||||||
@ -86,7 +86,7 @@ fn function_call() {
|
|||||||
),
|
),
|
||||||
Span(30, 31)
|
Span(30, 31)
|
||||||
),
|
),
|
||||||
(Instruction::r#return(true), Span(35, 36)),
|
(Instruction::r#return(true), Span(34, 35)),
|
||||||
],
|
],
|
||||||
vec![ConcreteValue::string("a"), ConcreteValue::string("b"),],
|
vec![ConcreteValue::string("a"), ConcreteValue::string("b"),],
|
||||||
vec![
|
vec![
|
||||||
@ -142,7 +142,7 @@ fn function_declaration() {
|
|||||||
),
|
),
|
||||||
Span(35, 36)
|
Span(35, 36)
|
||||||
),
|
),
|
||||||
(Instruction::r#return(true), Span(40, 40)),
|
(Instruction::r#return(true), Span(39, 40)),
|
||||||
],
|
],
|
||||||
vec![ConcreteValue::string("a"), ConcreteValue::string("b")],
|
vec![ConcreteValue::string("a"), ConcreteValue::string("b")],
|
||||||
vec![
|
vec![
|
||||||
|
160
dust-lang/tests/type_errors_add.rs
Normal file
160
dust-lang/tests/type_errors_add.rs
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
use dust_lang::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_boolean_left() {
|
||||||
|
let source = "true + 1";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::Boolean,
|
||||||
|
position: Span(0, 4)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_boolean_right() {
|
||||||
|
let source = "1 + true";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::Boolean,
|
||||||
|
position: Span(4, 8)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_function_left() {
|
||||||
|
let source = "fn(){} + 1";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::Function(FunctionType {
|
||||||
|
type_parameters: None,
|
||||||
|
value_parameters: None,
|
||||||
|
return_type: Box::new(Type::None)
|
||||||
|
}),
|
||||||
|
position: Span(0, 6)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_function_right() {
|
||||||
|
let source = "1 + fn(){}";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::Function(FunctionType {
|
||||||
|
type_parameters: None,
|
||||||
|
value_parameters: None,
|
||||||
|
return_type: Box::new(Type::None)
|
||||||
|
}),
|
||||||
|
position: Span(4, 10)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_list_left() {
|
||||||
|
let source = "[1, 2] + 1";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::List(Box::new(Type::Integer)),
|
||||||
|
position: Span(0, 6)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_list_right() {
|
||||||
|
let source = "1 + [1, 2]";
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
compile(source),
|
||||||
|
Err(DustError::Compile {
|
||||||
|
error: CompileError::CannotAddType {
|
||||||
|
argument_type: Type::List(Box::new(Type::Integer)),
|
||||||
|
position: Span(4, 10)
|
||||||
|
},
|
||||||
|
source,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn add_range_left() {
|
||||||
|
// todo!("Add ranges")
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn add_range_right() {
|
||||||
|
// todo!("Add ranges")
|
||||||
|
// }
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_character_and_float() {
|
||||||
|
let source = "'a' + 1.0";
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_character_and_integer() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_character_and_string() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_float_and_character() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_float_and_integer() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_float_and_string() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_integer_and_character() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_integer_and_float() {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_integer_and_string() {
|
||||||
|
todo!()
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user