2021-10-29 05:00:30 +02:00
|
|
|
package prog8tests.ast
|
2021-10-21 22:06:21 +02:00
|
|
|
|
2023-12-10 21:15:50 +01:00
|
|
|
import io.kotest.core.spec.style.FunSpec
|
2021-11-07 21:18:18 +01:00
|
|
|
import io.kotest.matchers.shouldBe
|
2023-12-10 21:15:50 +01:00
|
|
|
import io.kotest.matchers.string.shouldContain
|
2021-10-21 22:06:21 +02:00
|
|
|
import prog8.ast.statements.Block
|
|
|
|
import prog8.ast.statements.Subroutine
|
2022-03-10 22:38:16 +01:00
|
|
|
import prog8.code.core.DataType
|
2022-03-21 01:01:21 +01:00
|
|
|
import prog8.code.core.SourceCode
|
2023-12-10 21:15:50 +01:00
|
|
|
import prog8.code.target.Cx16Target
|
2022-05-22 23:11:22 +02:00
|
|
|
import prog8.parser.Prog8Parser.parseModule
|
2023-12-10 21:15:50 +01:00
|
|
|
import prog8tests.helpers.ErrorReporterForTests
|
|
|
|
import prog8tests.helpers.compileText
|
2021-10-21 22:06:21 +02:00
|
|
|
|
|
|
|
|
2023-12-10 21:15:50 +01:00
|
|
|
class TestSubroutines: FunSpec({
|
2021-10-21 22:06:21 +02:00
|
|
|
|
2023-12-10 21:15:50 +01:00
|
|
|
test("stringParameter AcceptedInParser") {
|
2021-10-21 22:06:21 +02:00
|
|
|
// note: the *parser* should accept this as it is valid *syntax*,
|
|
|
|
// however, the compiler itself may or may not complain about it later.
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
asmsub asmfunc(str thing @AY) {
|
|
|
|
}
|
|
|
|
|
|
|
|
sub func(str thing) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
val src = SourceCode.Text(text)
|
|
|
|
val module = parseModule(src)
|
|
|
|
val mainBlock = module.statements.single() as Block
|
|
|
|
val asmfunc = mainBlock.statements.filterIsInstance<Subroutine>().single { it.name=="asmfunc"}
|
|
|
|
val func = mainBlock.statements.filterIsInstance<Subroutine>().single { it.name=="func"}
|
2021-11-07 21:18:18 +01:00
|
|
|
asmfunc.isAsmSubroutine shouldBe true
|
|
|
|
asmfunc.parameters.single().type shouldBe DataType.STR
|
|
|
|
asmfunc.statements.isEmpty() shouldBe true
|
|
|
|
func.isAsmSubroutine shouldBe false
|
|
|
|
func.parameters.single().type shouldBe DataType.STR
|
|
|
|
func.statements.isEmpty() shouldBe true
|
2021-10-21 22:06:21 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 21:15:50 +01:00
|
|
|
test("arrayParameter AcceptedInParser") {
|
2021-10-21 22:06:21 +02:00
|
|
|
// note: the *parser* should accept this as it is valid *syntax*,
|
|
|
|
// however, the compiler itself may or may not complain about it later.
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
asmsub asmfunc(ubyte[] thing @AY) {
|
|
|
|
}
|
|
|
|
|
|
|
|
sub func(ubyte[22] thing) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
val src = SourceCode.Text(text)
|
|
|
|
val module = parseModule(src)
|
|
|
|
val mainBlock = module.statements.single() as Block
|
|
|
|
val asmfunc = mainBlock.statements.filterIsInstance<Subroutine>().single { it.name=="asmfunc"}
|
|
|
|
val func = mainBlock.statements.filterIsInstance<Subroutine>().single { it.name=="func"}
|
2021-11-07 21:18:18 +01:00
|
|
|
asmfunc.isAsmSubroutine shouldBe true
|
|
|
|
asmfunc.parameters.single().type shouldBe DataType.ARRAY_UB
|
|
|
|
asmfunc.statements.isEmpty() shouldBe true
|
|
|
|
func.isAsmSubroutine shouldBe false
|
|
|
|
func.parameters.single().type shouldBe DataType.ARRAY_UB
|
|
|
|
func.statements.isEmpty() shouldBe true
|
2021-10-21 22:06:21 +02:00
|
|
|
}
|
2023-12-10 21:15:50 +01:00
|
|
|
|
|
|
|
test("cannot call a subroutine via pointer") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword func = 12345
|
|
|
|
func() ; error
|
|
|
|
func(1,2,3) ; error
|
|
|
|
cx16.r0 = func() ; error
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(Cx16Target(), false, src, errors, false) shouldBe null
|
|
|
|
errors.errors.size shouldBe 3
|
|
|
|
errors.errors[0] shouldContain "cannot call that"
|
|
|
|
errors.errors[1] shouldContain "cannot call that"
|
|
|
|
errors.errors[2] shouldContain "cannot call that"
|
|
|
|
}
|
|
|
|
|
|
|
|
test("can call a subroutine pointer using call") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword func = 12345
|
2023-12-22 22:24:11 +01:00
|
|
|
void call(func) ; ok
|
|
|
|
void call(12345) ; ok
|
|
|
|
cx16.r0 = call(func) ; ok
|
|
|
|
void call(&start) ; error
|
|
|
|
void call(start) ; error
|
2023-12-10 21:15:50 +01:00
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(Cx16Target(), false, src, errors, false) shouldBe null
|
2023-12-22 22:24:11 +01:00
|
|
|
errors.errors.size shouldBe 2
|
|
|
|
errors.errors[0] shouldContain "can't call this indirectly"
|
2023-12-10 21:15:50 +01:00
|
|
|
errors.errors[1] shouldContain "can't call this indirectly"
|
|
|
|
}
|
|
|
|
})
|