2022-10-27 19:58:37 +00:00
|
|
|
package prog8tests.codegeneration
|
|
|
|
|
|
|
|
import io.kotest.core.spec.style.FunSpec
|
2023-05-28 21:19:01 +00:00
|
|
|
import io.kotest.matchers.shouldBe
|
2022-10-27 19:58:37 +00:00
|
|
|
import io.kotest.matchers.shouldNotBe
|
2023-05-28 21:19:01 +00:00
|
|
|
import io.kotest.matchers.string.shouldContain
|
2024-10-13 15:46:41 +00:00
|
|
|
import io.kotest.matchers.types.instanceOf
|
2024-10-13 02:20:57 +00:00
|
|
|
import prog8.code.ast.*
|
2022-10-27 19:58:37 +00:00
|
|
|
import prog8.code.target.C64Target
|
2022-10-27 21:31:57 +00:00
|
|
|
import prog8.code.target.VMTarget
|
2024-07-20 20:36:19 +00:00
|
|
|
import prog8tests.helpers.ErrorReporterForTests
|
2022-10-27 21:08:40 +00:00
|
|
|
import prog8tests.helpers.compileText
|
2023-11-02 23:39:43 +00:00
|
|
|
import kotlin.io.path.readText
|
2022-10-27 19:58:37 +00:00
|
|
|
|
2023-05-28 21:19:01 +00:00
|
|
|
class TestArrayThings: FunSpec({
|
2022-10-27 19:58:37 +00:00
|
|
|
test("assign prefix var to array should compile fine and is not split into inplace array modification") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
byte[5] array
|
|
|
|
byte bb
|
|
|
|
array[1] = -bb
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 21:31:57 +00:00
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 19:58:37 +00:00
|
|
|
}
|
2022-10-27 21:08:40 +00:00
|
|
|
|
|
|
|
test("array in-place negation (integer types)") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
byte[10] foo
|
|
|
|
ubyte[10] foou
|
|
|
|
word[10] foow
|
|
|
|
uword[10] foowu
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
foo[1] = 42
|
|
|
|
foo[1] = -foo[1]
|
|
|
|
|
|
|
|
foow[1] = 4242
|
|
|
|
foow[1] = -foow[1]
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 21:31:57 +00:00
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 21:08:40 +00:00
|
|
|
}
|
|
|
|
|
2022-10-29 15:04:39 +00:00
|
|
|
test("array in-place negation (float type) vm target") {
|
2022-10-27 21:08:40 +00:00
|
|
|
val text = """
|
|
|
|
%import floats
|
|
|
|
|
|
|
|
main {
|
|
|
|
float[10] flt
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
flt[1] = 42.42
|
|
|
|
flt[1] = -flt[1]
|
|
|
|
}
|
|
|
|
}"""
|
2022-10-27 21:31:57 +00:00
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-29 15:04:39 +00:00
|
|
|
}
|
|
|
|
|
2023-02-03 23:02:50 +00:00
|
|
|
test("array in-place negation (float type) 6502 target") {
|
2022-10-29 15:04:39 +00:00
|
|
|
val text = """
|
|
|
|
%import floats
|
|
|
|
|
|
|
|
main {
|
|
|
|
float[10] flt
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
flt[1] = 42.42
|
|
|
|
flt[1] = -flt[1]
|
|
|
|
}
|
|
|
|
}"""
|
2022-10-27 21:08:40 +00:00
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
}
|
|
|
|
|
|
|
|
test("array in-place invert") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
ubyte[10] foo
|
|
|
|
uword[10] foow
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
foo[1] = 42
|
|
|
|
foo[1] = ~foo[1]
|
|
|
|
|
|
|
|
foow[1] = 4242
|
|
|
|
foow[1] = ~foow[1]
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 21:31:57 +00:00
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
2022-10-27 21:08:40 +00:00
|
|
|
}
|
2023-05-28 21:19:01 +00:00
|
|
|
|
|
|
|
test("split only for word arrays") {
|
2023-12-26 17:49:01 +00:00
|
|
|
val srcGood = """
|
2023-05-28 21:19:01 +00:00
|
|
|
main {
|
|
|
|
uword[10] @split sw
|
|
|
|
word[10] @split sw2
|
2023-12-26 17:49:01 +00:00
|
|
|
|
|
|
|
sub start() {
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(C64Target(), false, srcGood, writeAssembly = false) shouldNotBe null
|
|
|
|
|
|
|
|
val srcWrong1 = """
|
|
|
|
main {
|
|
|
|
ubyte[10] @split sb
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
}
|
|
|
|
}"""
|
2024-09-06 20:51:26 +00:00
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(C64Target(), false, srcWrong1, writeAssembly = false, errors=errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 1
|
|
|
|
errors.errors[0] shouldContain "split can only be used on word arrays"
|
2023-12-26 17:49:01 +00:00
|
|
|
|
|
|
|
val srcWrong2 = """
|
2024-09-06 20:51:26 +00:00
|
|
|
%option enable_floats
|
2023-12-26 17:49:01 +00:00
|
|
|
main {
|
2023-05-28 21:19:01 +00:00
|
|
|
float[10] @split sf
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
}
|
|
|
|
}"""
|
2024-09-06 20:51:26 +00:00
|
|
|
errors.clear()
|
|
|
|
compileText(C64Target(), false, srcWrong2, writeAssembly = false, errors=errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 1
|
|
|
|
errors.errors[0] shouldContain "split can only be used on word arrays"
|
2023-05-28 21:19:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
test("split word arrays in asm as lsb/msb") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
uword[10] @split @shared uw
|
|
|
|
word[10] @split @shared sw
|
|
|
|
uword[10] @shared normal
|
|
|
|
|
|
|
|
sub start() {
|
|
|
|
%asm {{
|
2023-12-20 21:20:12 +00:00
|
|
|
lda p8v_normal
|
|
|
|
lda p8v_uw_lsb
|
|
|
|
lda p8v_uw_msb
|
|
|
|
lda p8v_sw_lsb
|
|
|
|
lda p8v_sw_msb
|
2023-05-28 21:19:01 +00:00
|
|
|
}}
|
|
|
|
}
|
|
|
|
}"""
|
2023-05-30 19:54:19 +00:00
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
}
|
|
|
|
|
2023-10-26 20:28:07 +00:00
|
|
|
test("array target with expression for index") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
ubyte[] array = [1,2,3]
|
|
|
|
array[cx16.r0L+1] += 42
|
|
|
|
cx16.r0L = array[cx16.r0L+1]
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(VMTarget(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
}
|
2023-11-02 23:39:43 +00:00
|
|
|
|
|
|
|
test("split array in zeropage is okay") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword[3] @zp @split @shared thearray
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val result = compileText(C64Target(), false, text, writeAssembly = true)!!
|
|
|
|
val assemblyFile = result.compilationOptions.outputDir.resolve(result.compilerAst.name + ".asm")
|
|
|
|
val assembly = assemblyFile.readText()
|
|
|
|
assembly shouldContain "thearray_lsb"
|
|
|
|
assembly shouldContain "thearray_msb"
|
|
|
|
}
|
2023-11-14 20:46:11 +00:00
|
|
|
|
|
|
|
test("indexing str or pointervar with expression") {
|
|
|
|
val text = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
str name = "thing"
|
|
|
|
modify(name)
|
|
|
|
|
|
|
|
sub modify(str arg) {
|
|
|
|
ubyte n=1
|
|
|
|
uword pointervar
|
|
|
|
arg[n+1] = arg[1]
|
|
|
|
pointervar[n+1] = pointervar[1]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(C64Target(), false, text, writeAssembly = true) shouldNotBe null
|
|
|
|
}
|
2024-01-16 19:41:53 +00:00
|
|
|
|
|
|
|
test("address of a uword pointer array expression") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
set_state(12345, 1)
|
|
|
|
}
|
|
|
|
sub set_state(uword buffer, ubyte i) {
|
|
|
|
uword addr = &buffer[i]
|
|
|
|
addr++
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(C64Target(), false, src, writeAssembly = true) shouldNotBe null
|
|
|
|
}
|
2024-07-20 20:36:19 +00:00
|
|
|
|
|
|
|
test("negative array index variables are not allowed, but ptr indexing is allowed") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword @shared pointer
|
|
|
|
ubyte[10] array
|
|
|
|
str name = "hello"
|
|
|
|
|
|
|
|
byte sindex
|
|
|
|
|
|
|
|
array[sindex] = 10
|
|
|
|
cx16.r0L = array[sindex]
|
|
|
|
name[sindex] = 0
|
|
|
|
cx16.r0L = name[sindex]
|
|
|
|
pointer[sindex] = 10
|
|
|
|
cx16.r0L=pointer[sindex]
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false, errors = errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 4
|
|
|
|
errors.errors[0] shouldContain "signed variables"
|
|
|
|
errors.errors[1] shouldContain "signed variables"
|
|
|
|
errors.errors[2] shouldContain "signed variables"
|
|
|
|
errors.errors[3] shouldContain "signed variables"
|
|
|
|
}
|
|
|
|
|
|
|
|
test("bounds checking for both positive and negative indexes, correct cases") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
ubyte[10] array
|
|
|
|
array[0] = 0
|
|
|
|
array[9] = 0
|
|
|
|
array[-1] = 0
|
|
|
|
array[-10] = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false) shouldNotBe null
|
|
|
|
}
|
|
|
|
|
|
|
|
test("bounds checking on strings, correct cases") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
str name = "1234567890"
|
|
|
|
name[0] = 0
|
|
|
|
name[9] = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false) shouldNotBe null
|
|
|
|
}
|
|
|
|
|
|
|
|
test("bounds checking for positive indexes, invalid case") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
ubyte[10] array
|
|
|
|
array[10] = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false, errors = errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 1
|
|
|
|
errors.errors[0] shouldContain "out of bounds"
|
|
|
|
}
|
|
|
|
|
|
|
|
test("bounds checking for negative indexes, invalid case") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
ubyte[10] array
|
|
|
|
array[-11] = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false, errors = errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 1
|
|
|
|
errors.errors[0] shouldContain "out of bounds"
|
|
|
|
}
|
|
|
|
|
|
|
|
test("bounds checking on strings invalid cases") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
str name = "1234567890"
|
|
|
|
name[10] = 0
|
|
|
|
name[-1] = 0
|
|
|
|
name[-11] = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(VMTarget(), false, src, writeAssembly = false, errors = errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 3
|
|
|
|
errors.errors[0] shouldContain "out of bounds"
|
|
|
|
errors.errors[1] shouldContain "out of bounds"
|
|
|
|
errors.errors[2] shouldContain "out of bounds"
|
|
|
|
}
|
2024-10-12 00:56:36 +00:00
|
|
|
|
2024-10-13 02:20:57 +00:00
|
|
|
test("array and string initializer with multiplication") {
|
|
|
|
val src="""
|
|
|
|
%option enable_floats
|
|
|
|
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
str name = "xyz" * 3
|
|
|
|
bool[3] boolarray = [true] * 3
|
|
|
|
ubyte[3] bytearray = [42] * 3
|
|
|
|
uword[3] wordarray = [5555] * 3
|
|
|
|
float[3] floatarray = [123.45] * 3
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val result = compileText(C64Target(), false, src, writeAssembly = true)!!
|
|
|
|
val x = result.codegenAst!!.entrypoint()!!
|
|
|
|
x.children.size shouldBe 6
|
|
|
|
((x.children[0] as PtVariable).value as PtString).value shouldBe "xyzxyzxyz"
|
|
|
|
val array1 = (x.children[1] as PtVariable).value as PtArray
|
|
|
|
val array2 = (x.children[2] as PtVariable).value as PtArray
|
|
|
|
val array3 = (x.children[3] as PtVariable).value as PtArray
|
|
|
|
val array4 = (x.children[4] as PtVariable).value as PtArray
|
|
|
|
array1.children.map { (it as PtBool).value } shouldBe listOf(true, true, true)
|
|
|
|
array2.children.map { (it as PtNumber).number } shouldBe listOf(42, 42, 42)
|
|
|
|
array3.children.map { (it as PtNumber).number } shouldBe listOf(5555, 5555, 5555)
|
|
|
|
array4.children.map { (it as PtNumber).number } shouldBe listOf(123.45, 123.45, 123.45)
|
|
|
|
}
|
|
|
|
|
|
|
|
test("array initializer with range") {
|
|
|
|
val src="""
|
|
|
|
%option enable_floats
|
|
|
|
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
ubyte[3] bytearray2 = 10 to 12
|
|
|
|
uword[3] wordarray2 = 5000 to 5002
|
|
|
|
float[3] floatarray2 = 100 to 102
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val result = compileText(C64Target(), false, src, writeAssembly = true)!!
|
|
|
|
val x = result.codegenAst!!.entrypoint()!!
|
|
|
|
x.children.size shouldBe 4
|
|
|
|
val array1 = (x.children[0] as PtVariable).value as PtArray
|
|
|
|
val array2 = (x.children[1] as PtVariable).value as PtArray
|
|
|
|
val array3 = (x.children[2] as PtVariable).value as PtArray
|
|
|
|
array1.children.map { (it as PtNumber).number } shouldBe listOf(10, 11, 12)
|
|
|
|
array2.children.map { (it as PtNumber).number } shouldBe listOf(5000, 5001, 5002)
|
|
|
|
array3.children.map { (it as PtNumber).number } shouldBe listOf(100, 101, 102)
|
|
|
|
}
|
|
|
|
|
2024-10-13 15:46:41 +00:00
|
|
|
test("identifiers in array literals getting implicit address-of") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
label:
|
|
|
|
str @shared name = "name"
|
|
|
|
uword[] @shared array1 = [name, label, start, main]
|
|
|
|
uword[] @shared array2 = [&name, &label, &start, &main]
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val result = compileText(C64Target(), false, src, writeAssembly = true)!!
|
|
|
|
val x = result.codegenAst!!.entrypoint()!!
|
|
|
|
x.children.size shouldBe 5
|
|
|
|
val array1 = (x.children[1] as PtVariable).value as PtArray
|
|
|
|
val array2 = (x.children[2] as PtVariable).value as PtArray
|
|
|
|
array1.children.forEach {
|
|
|
|
it shouldBe instanceOf<PtAddressOf>()
|
|
|
|
}
|
|
|
|
array2.children.forEach {
|
|
|
|
it shouldBe instanceOf<PtAddressOf>()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test("variable identifiers in array literals not getting implicit address-of") {
|
|
|
|
val src="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
label:
|
|
|
|
str @shared name = "name"
|
|
|
|
ubyte @shared bytevar
|
|
|
|
uword[] @shared array1 = [cx16.r0] ; error, is variables
|
|
|
|
uword[] @shared array2 = [bytevar] ; error, is variables
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
compileText(C64Target(), false, src, writeAssembly = true, errors=errors) shouldBe null
|
|
|
|
errors.errors.size shouldBe 2
|
|
|
|
errors.errors[0] shouldContain "contains non-constant"
|
|
|
|
errors.errors[1] shouldContain "contains non-constant"
|
|
|
|
}
|
2022-10-27 19:58:37 +00:00
|
|
|
})
|
|
|
|
|