2021-10-20 22:16:26 +02:00
|
|
|
package prog8tests
|
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
import io.kotest.assertions.withClue
|
|
|
|
import io.kotest.core.spec.style.FunSpec
|
2022-01-27 23:32:55 +01:00
|
|
|
import io.kotest.matchers.ints.shouldBeGreaterThanOrEqual
|
2021-11-08 00:16:58 +01:00
|
|
|
import io.kotest.matchers.maps.shouldNotContainKey
|
|
|
|
import io.kotest.matchers.shouldBe
|
2022-02-01 23:09:52 +01:00
|
|
|
import io.kotest.matchers.string.shouldContain
|
2022-02-01 00:24:31 +01:00
|
|
|
import prog8.ast.Program
|
2021-10-20 22:16:26 +02:00
|
|
|
import prog8.ast.statements.Block
|
|
|
|
import prog8.ast.statements.Subroutine
|
2021-12-28 14:23:36 +01:00
|
|
|
import prog8.codegen.target.C64Target
|
2021-10-29 05:00:30 +02:00
|
|
|
import prog8.compilerinterface.CallGraph
|
2022-02-01 00:24:31 +01:00
|
|
|
import prog8.parser.Prog8Parser.parseModule
|
|
|
|
import prog8.parser.SourceCode
|
|
|
|
import prog8tests.helpers.*
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
class TestCallgraph: FunSpec({
|
|
|
|
test("testGraphForEmptySubs") {
|
2021-10-20 22:16:26 +02:00
|
|
|
val sourcecode = """
|
|
|
|
%import string
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
}
|
|
|
|
sub empty() {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
2022-03-07 21:41:12 +01:00
|
|
|
val result = compileText(C64Target(), false, sourcecode)!!
|
2021-10-30 00:25:34 +02:00
|
|
|
val graph = CallGraph(result.program)
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.imports.size shouldBe 1
|
|
|
|
graph.importedBy.size shouldBe 1
|
2021-10-30 00:25:34 +02:00
|
|
|
val toplevelModule = result.program.toplevelModule
|
2021-10-20 22:16:26 +02:00
|
|
|
val importedModule = graph.imports.getValue(toplevelModule).single()
|
2021-11-08 00:16:58 +01:00
|
|
|
importedModule.name shouldBe "string"
|
2021-10-20 22:16:26 +02:00
|
|
|
val importedBy = graph.importedBy.getValue(importedModule).single()
|
2021-11-08 00:16:58 +01:00
|
|
|
importedBy.name.startsWith("on_the_fly_test") shouldBe true
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.unused(toplevelModule) shouldBe false
|
|
|
|
graph.unused(importedModule) shouldBe false
|
2021-10-20 22:16:26 +02:00
|
|
|
|
|
|
|
val mainBlock = toplevelModule.statements.filterIsInstance<Block>().single()
|
|
|
|
for(stmt in mainBlock.statements) {
|
|
|
|
val sub = stmt as Subroutine
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.calls shouldNotContainKey sub
|
|
|
|
graph.calledBy shouldNotContainKey sub
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-10-30 00:25:34 +02:00
|
|
|
if(sub === result.program.entrypoint)
|
2021-11-08 00:16:58 +01:00
|
|
|
withClue("start() should always be marked as used to avoid having it removed") {
|
|
|
|
graph.unused(sub) shouldBe false
|
|
|
|
}
|
2021-10-20 22:16:26 +02:00
|
|
|
else
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.unused(sub) shouldBe true
|
2021-10-20 22:16:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-01 23:09:52 +01:00
|
|
|
test("reference to empty sub") {
|
2021-10-20 22:16:26 +02:00
|
|
|
val sourcecode = """
|
|
|
|
%import string
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword xx = &empty
|
|
|
|
xx++
|
|
|
|
}
|
|
|
|
sub empty() {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
2022-03-07 21:41:12 +01:00
|
|
|
val result = compileText(C64Target(), false, sourcecode)!!
|
2021-10-30 00:25:34 +02:00
|
|
|
val graph = CallGraph(result.program)
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.imports.size shouldBe 1
|
|
|
|
graph.importedBy.size shouldBe 1
|
2021-10-30 00:25:34 +02:00
|
|
|
val toplevelModule = result.program.toplevelModule
|
2021-10-20 22:16:26 +02:00
|
|
|
val importedModule = graph.imports.getValue(toplevelModule).single()
|
2021-11-08 00:16:58 +01:00
|
|
|
importedModule.name shouldBe "string"
|
2021-10-20 22:16:26 +02:00
|
|
|
val importedBy = graph.importedBy.getValue(importedModule).single()
|
2021-11-08 00:16:58 +01:00
|
|
|
importedBy.name.startsWith("on_the_fly_test") shouldBe true
|
2021-10-20 22:16:26 +02:00
|
|
|
|
2021-11-08 00:16:58 +01:00
|
|
|
graph.unused(toplevelModule) shouldBe false
|
|
|
|
graph.unused(importedModule) shouldBe false
|
2021-10-20 22:16:26 +02:00
|
|
|
|
|
|
|
val mainBlock = toplevelModule.statements.filterIsInstance<Block>().single()
|
|
|
|
val startSub = mainBlock.statements.filterIsInstance<Subroutine>().single{it.name=="start"}
|
|
|
|
val emptySub = mainBlock.statements.filterIsInstance<Subroutine>().single{it.name=="empty"}
|
|
|
|
|
2022-02-01 23:09:52 +01:00
|
|
|
graph.calls shouldNotContainKey startSub
|
|
|
|
graph.calledBy shouldNotContainKey emptySub
|
2021-11-08 00:16:58 +01:00
|
|
|
withClue("empty doesn't call anything") {
|
|
|
|
graph.calls shouldNotContainKey emptySub
|
|
|
|
}
|
|
|
|
withClue( "start doesn't get called (except as entrypoint ofc.)") {
|
|
|
|
graph.calledBy shouldNotContainKey startSub
|
|
|
|
}
|
2021-10-20 22:16:26 +02:00
|
|
|
}
|
2022-01-27 23:32:55 +01:00
|
|
|
|
|
|
|
test("allIdentifiers separates for different positions of the IdentifierReferences") {
|
|
|
|
val sourcecode = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
uword x1 = &empty
|
|
|
|
uword x2 = &empty
|
|
|
|
empty()
|
|
|
|
}
|
|
|
|
sub empty() {
|
|
|
|
%asm {{
|
|
|
|
nop
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"""
|
2022-03-07 21:41:12 +01:00
|
|
|
val result = compileText(C64Target(), false, sourcecode)!!
|
2022-01-27 23:32:55 +01:00
|
|
|
val graph = CallGraph(result.program)
|
|
|
|
graph.allIdentifiers.size shouldBeGreaterThanOrEqual 9
|
|
|
|
val empties = graph.allIdentifiers.keys.filter { it.nameInSource==listOf("empty") }
|
|
|
|
empties.size shouldBe 3
|
|
|
|
empties[0].position.line shouldBe 4
|
|
|
|
empties[1].position.line shouldBe 5
|
|
|
|
empties[2].position.line shouldBe 6
|
|
|
|
}
|
2022-02-01 00:24:31 +01:00
|
|
|
|
|
|
|
test("checking block and subroutine names usage in assembly code") {
|
|
|
|
val source = """
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
%asm {{
|
|
|
|
lda #<blockname
|
|
|
|
lda #<blockname.subroutine
|
|
|
|
correctlabel:
|
|
|
|
nop
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
blockname {
|
|
|
|
sub subroutine() {
|
|
|
|
@(1000) = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
sub correctlabel() {
|
|
|
|
@(1000) = 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
; all block and subroutines below should NOT be found in asm because they're only substrings of the names in there
|
|
|
|
locknam {
|
|
|
|
sub rout() {
|
|
|
|
@(1000) = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
sub orrectlab() {
|
|
|
|
@(1000) = 0
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val module = parseModule(SourceCode.Text(source))
|
|
|
|
val program = Program("test", DummyFunctions, DummyMemsizer, DummyStringEncoder)
|
|
|
|
program.addModule(module)
|
|
|
|
val callgraph = CallGraph(program)
|
|
|
|
val blockMain = program.allBlocks.single { it.name=="main" }
|
|
|
|
val blockBlockname = program.allBlocks.single { it.name=="blockname" }
|
|
|
|
val blockLocknam = program.allBlocks.single { it.name=="locknam" }
|
|
|
|
val subStart = blockMain.statements.filterIsInstance<Subroutine>().single { it.name == "start" }
|
|
|
|
val subSubroutine = blockBlockname.statements.filterIsInstance<Subroutine>().single { it.name == "subroutine" }
|
|
|
|
val subCorrectlabel = blockBlockname.statements.filterIsInstance<Subroutine>().single { it.name == "correctlabel" }
|
|
|
|
val subRout = blockLocknam.statements.filterIsInstance<Subroutine>().single { it.name == "rout" }
|
|
|
|
val subOrrectlab = blockLocknam.statements.filterIsInstance<Subroutine>().single { it.name == "orrectlab" }
|
|
|
|
callgraph.unused(blockMain) shouldBe false
|
|
|
|
callgraph.unused(blockBlockname) shouldBe false
|
|
|
|
callgraph.unused(blockLocknam) shouldBe true
|
|
|
|
callgraph.unused(subStart) shouldBe false
|
|
|
|
callgraph.unused(subSubroutine) shouldBe false
|
|
|
|
callgraph.unused(subCorrectlabel) shouldBe false
|
|
|
|
callgraph.unused(subRout) shouldBe true
|
|
|
|
callgraph.unused(subOrrectlab) shouldBe true
|
|
|
|
}
|
2022-02-01 23:09:52 +01:00
|
|
|
|
|
|
|
test("recursion detection") {
|
|
|
|
val source="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
recurse1()
|
|
|
|
}
|
|
|
|
sub recurse1() {
|
|
|
|
recurse2()
|
|
|
|
}
|
|
|
|
sub recurse2() {
|
|
|
|
start()
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val module = parseModule(SourceCode.Text(source))
|
|
|
|
val program = Program("test", DummyFunctions, DummyMemsizer, DummyStringEncoder)
|
|
|
|
program.addModule(module)
|
|
|
|
val callgraph = CallGraph(program)
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
callgraph.checkRecursiveCalls(errors)
|
|
|
|
errors.errors.size shouldBe 0
|
|
|
|
errors.warnings.size shouldBe 4
|
|
|
|
errors.warnings[0] shouldContain "contains recursive subroutine calls"
|
|
|
|
errors.warnings[1] shouldContain "start at"
|
|
|
|
errors.warnings[2] shouldContain "recurse1 at"
|
|
|
|
errors.warnings[3] shouldContain "recurse2 at"
|
|
|
|
}
|
|
|
|
|
|
|
|
test("no recursion warning if reference isn't a call") {
|
|
|
|
val source="""
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
recurse1()
|
|
|
|
}
|
|
|
|
sub recurse1() {
|
|
|
|
recurse2()
|
|
|
|
}
|
|
|
|
sub recurse2() {
|
|
|
|
uword @shared address = &start
|
|
|
|
}
|
|
|
|
}"""
|
|
|
|
val module = parseModule(SourceCode.Text(source))
|
|
|
|
val program = Program("test", DummyFunctions, DummyMemsizer, DummyStringEncoder)
|
|
|
|
program.addModule(module)
|
|
|
|
val callgraph = CallGraph(program)
|
|
|
|
val errors = ErrorReporterForTests()
|
|
|
|
callgraph.checkRecursiveCalls(errors)
|
|
|
|
errors.errors.size shouldBe 0
|
|
|
|
errors.warnings.size shouldBe 0
|
|
|
|
}
|
2021-11-08 00:16:58 +01:00
|
|
|
})
|