2021-07-17 10:37:58 +02:00
|
|
|
package prog8tests
|
|
|
|
|
2021-10-11 00:22:04 +02:00
|
|
|
import org.junit.jupiter.api.*
|
2021-07-17 10:37:58 +02:00
|
|
|
import org.junit.jupiter.api.DynamicTest.dynamicTest
|
|
|
|
import prog8.ast.expressions.AddressOf
|
|
|
|
import prog8.ast.expressions.IdentifierReference
|
|
|
|
import prog8.ast.expressions.StringLiteralValue
|
|
|
|
import prog8.ast.statements.FunctionCallStatement
|
|
|
|
import prog8.ast.statements.Label
|
|
|
|
import prog8.compiler.target.Cx16Target
|
2021-10-29 05:00:30 +02:00
|
|
|
import prog8tests.ast.helpers.*
|
2021-10-29 05:28:02 +02:00
|
|
|
import prog8tests.helpers.assertFailure
|
|
|
|
import prog8tests.helpers.assertSuccess
|
|
|
|
import prog8tests.helpers.compileFile
|
2021-10-11 00:22:04 +02:00
|
|
|
import kotlin.io.path.name
|
|
|
|
import kotlin.test.assertEquals
|
|
|
|
import kotlin.test.assertNotEquals
|
2021-07-17 10:37:58 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ATTENTION: this is just kludge!
|
|
|
|
* They are not really unit tests, but rather tests of the whole process,
|
|
|
|
* from source file loading all the way through to running 64tass.
|
|
|
|
*/
|
|
|
|
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
|
|
|
|
class TestCompilerOnImportsAndIncludes {
|
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
@Nested
|
|
|
|
inner class Import {
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun testImportFromSameFolder() {
|
|
|
|
val filepath = assumeReadableFile(fixturesDir, "importFromSameFolder.p8")
|
|
|
|
assumeReadableFile(fixturesDir, "foo_bar.p8")
|
|
|
|
|
|
|
|
val platform = Cx16Target
|
|
|
|
val result = compileFile(platform, optimize = false, fixturesDir, filepath.name)
|
|
|
|
.assertSuccess()
|
|
|
|
|
2021-10-30 00:25:34 +02:00
|
|
|
val program = result.program
|
2021-10-11 00:01:26 +02:00
|
|
|
val startSub = program.entrypoint
|
2021-08-02 08:57:09 +02:00
|
|
|
val strLits = startSub.statements
|
|
|
|
.filterIsInstance<FunctionCallStatement>()
|
|
|
|
.map { it.args[0] as IdentifierReference }
|
|
|
|
.map { it.targetVarDecl(program)!!.value as StringLiteralValue }
|
|
|
|
|
|
|
|
assertEquals("main.bar", strLits[0].value)
|
|
|
|
assertEquals("foo.bar", strLits[1].value)
|
2021-10-11 00:01:26 +02:00
|
|
|
assertEquals("main", strLits[0].definingScope.name)
|
|
|
|
assertEquals("foo", strLits[1].definingScope.name)
|
2021-08-02 08:57:09 +02:00
|
|
|
}
|
2021-07-17 10:37:58 +02:00
|
|
|
}
|
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
@Nested
|
|
|
|
inner class AsmInclude {
|
|
|
|
@Test
|
|
|
|
fun testAsmIncludeFromSameFolder() {
|
|
|
|
val filepath = assumeReadableFile(fixturesDir, "asmIncludeFromSameFolder.p8")
|
|
|
|
assumeReadableFile(fixturesDir, "foo_bar.asm")
|
|
|
|
|
|
|
|
val platform = Cx16Target
|
|
|
|
val result = compileFile(platform, optimize = false, fixturesDir, filepath.name)
|
|
|
|
.assertSuccess()
|
|
|
|
|
2021-10-30 00:25:34 +02:00
|
|
|
val program = result.program
|
2021-10-11 00:01:26 +02:00
|
|
|
val startSub = program.entrypoint
|
2021-08-02 08:57:09 +02:00
|
|
|
val args = startSub.statements
|
|
|
|
.filterIsInstance<FunctionCallStatement>()
|
|
|
|
.map { it.args[0] }
|
|
|
|
|
|
|
|
val str0 = (args[0] as IdentifierReference).targetVarDecl(program)!!.value as StringLiteralValue
|
|
|
|
assertEquals("main.bar", str0.value)
|
2021-10-11 00:01:26 +02:00
|
|
|
assertEquals("main", str0.definingScope.name)
|
2021-08-02 08:57:09 +02:00
|
|
|
|
|
|
|
val id1 = (args[1] as AddressOf).identifier
|
|
|
|
val lbl1 = id1.targetStatement(program) as Label
|
|
|
|
assertEquals("foo_bar", lbl1.name)
|
2021-10-11 00:01:26 +02:00
|
|
|
assertEquals("start", lbl1.definingScope.name)
|
2021-08-02 08:57:09 +02:00
|
|
|
}
|
2021-07-17 10:37:58 +02:00
|
|
|
}
|
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
@Nested
|
|
|
|
inner class Asmbinary {
|
|
|
|
@Test
|
|
|
|
fun testAsmbinaryDirectiveWithNonExistingFile() {
|
2021-10-09 17:59:40 +02:00
|
|
|
val p8Path = assumeReadableFile(fixturesDir, "asmBinaryNonExisting.p8")
|
2021-08-02 08:57:09 +02:00
|
|
|
assumeNotExists(fixturesDir, "i_do_not_exist.bin")
|
2021-07-17 10:37:58 +02:00
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
compileFile(Cx16Target, false, p8Path.parent, p8Path.name, outputDir)
|
|
|
|
.assertFailure()
|
|
|
|
}
|
2021-07-17 10:37:58 +02:00
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
@Test
|
|
|
|
fun testAsmbinaryDirectiveWithNonReadableFile() {
|
2021-10-09 17:59:40 +02:00
|
|
|
val p8Path = assumeReadableFile(fixturesDir, "asmBinaryNonReadable.p8")
|
2021-08-02 08:57:09 +02:00
|
|
|
assumeDirectory(fixturesDir, "subFolder")
|
2021-07-17 10:37:58 +02:00
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
compileFile(Cx16Target, false, p8Path.parent, p8Path.name, outputDir)
|
|
|
|
.assertFailure()
|
|
|
|
}
|
2021-07-17 10:37:58 +02:00
|
|
|
|
2021-08-02 08:57:09 +02:00
|
|
|
@TestFactory
|
|
|
|
fun asmbinaryDirectiveWithExistingBinFile(): Iterable<DynamicTest> =
|
|
|
|
listOf(
|
|
|
|
Triple("same ", "asmBinaryFromSameFolder.p8", "do_nothing1.bin"),
|
|
|
|
Triple("sub", "asmBinaryFromSubFolder.p8", "subFolder/do_nothing2.bin"),
|
|
|
|
).map {
|
|
|
|
val (where, p8Str, binStr) = it
|
|
|
|
dynamicTest("%asmbinary from ${where}folder") {
|
|
|
|
val p8Path = assumeReadableFile(fixturesDir, p8Str)
|
2021-10-29 00:20:33 +02:00
|
|
|
// val binPath = assumeReadableFile(fixturesDir, binStr)
|
|
|
|
assertNotEquals( // the bug we're testing for (#54) was hidden if outputDir == workingDir
|
2021-08-02 08:57:09 +02:00
|
|
|
workingDir.normalize().toAbsolutePath(),
|
|
|
|
outputDir.normalize().toAbsolutePath(),
|
|
|
|
"sanity check: workingDir and outputDir should not be the same folder"
|
2021-07-17 10:37:58 +02:00
|
|
|
)
|
2021-08-02 08:57:09 +02:00
|
|
|
|
|
|
|
compileFile(Cx16Target, false, p8Path.parent, p8Path.name, outputDir)
|
|
|
|
.assertSuccess(
|
|
|
|
"argument to assembler directive .binary " +
|
|
|
|
"should be relative to the generated .asm file (in output dir), " +
|
|
|
|
"NOT relative to .p8 neither current working dir"
|
|
|
|
)
|
|
|
|
}
|
2021-07-17 10:37:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-10-09 17:59:40 +02:00
|
|
|
}
|