diff --git a/src/main/java/dk/camelot64/kickc/passes/Pass2ConstantStringConsolidation.java b/src/main/java/dk/camelot64/kickc/passes/Pass2ConstantStringConsolidation.java index a507a5734..9e11fffd2 100644 --- a/src/main/java/dk/camelot64/kickc/passes/Pass2ConstantStringConsolidation.java +++ b/src/main/java/dk/camelot64/kickc/passes/Pass2ConstantStringConsolidation.java @@ -134,6 +134,7 @@ public class Pass2ConstantStringConsolidation extends Pass2SsaOptimization { if(getScope().getLocalSymbol(candidateName) == null) { return candidateName; } + i++; } while(true); } diff --git a/src/test/java/dk/camelot64/kickc/test/TestPrograms.java b/src/test/java/dk/camelot64/kickc/test/TestPrograms.java index ce8636d3f..cc54987f7 100644 --- a/src/test/java/dk/camelot64/kickc/test/TestPrograms.java +++ b/src/test/java/dk/camelot64/kickc/test/TestPrograms.java @@ -467,6 +467,11 @@ public class TestPrograms { compileAndCompare("examples/nes/nes-demo.c"); } + //@Test + //public void testCx16Vera() throws IOException, URISyntaxException { + // compileAndCompare("examples/cx16/cx16-vera.c"); + //} + @Test public void testCx16VeraLayers() throws IOException, URISyntaxException { compileAndCompare("examples/cx16/cx16-veralayers.c"); @@ -3546,6 +3551,11 @@ public class TestPrograms { compileAndCompare("string-const-consolidation.c"); } + @Test + public void testStringConstConsolidationRoot() throws IOException, URISyntaxException { + compileAndCompare("string-const-consolidation-root.c"); + } + @Test public void testCommentsGlobalInit() throws IOException, URISyntaxException { compileAndCompare("test-comments-global.c"); diff --git a/src/test/kc/string-const-consolidation-root.c b/src/test/kc/string-const-consolidation-root.c new file mode 100644 index 000000000..cbbf04573 --- /dev/null +++ b/src/test/kc/string-const-consolidation-root.c @@ -0,0 +1,30 @@ +// Tests that multiple different identical strings are consolidated to a root variable + +void main() { + print1(); + print2(); +} + +char* s = "string3"; +char* s1 = "string4"; + + +void print1() { + print("string1"); + print("string2"); + print(s); +} + +void print2() { + print("string1"); + print("string2"); + print(s1); +} + +char* screen = 0x400; + +void print(char* s) { + while(*s) { + *screen++ = *s++; + } +} \ No newline at end of file diff --git a/src/test/ref/string-const-consolidation-root.asm b/src/test/ref/string-const-consolidation-root.asm new file mode 100644 index 000000000..e64559d85 --- /dev/null +++ b/src/test/ref/string-const-consolidation-root.asm @@ -0,0 +1,103 @@ +// Tests that multiple different identical strings are consolidated to a root variable + // Commodore 64 PRG executable file +.file [name="string-const-consolidation-root.prg", type="prg", segments="Program"] +.segmentdef Program [segments="Basic, Code, Data"] +.segmentdef Basic [start=$0801] +.segmentdef Code [start=$80d] +.segmentdef Data [startAfter="Code"] +.segment Basic +:BasicUpstart(main) + .label screen = 2 +.segment Code +main: { + // print1() + jsr print1 + // print2() + jsr print2 + // } + rts +} +print1: { + // print("string1") + lda #<$400 + sta.z screen + lda #>$400 + sta.z screen+1 + lda #string_0 + sta.z print.s+1 + jsr print + // print("string2") + lda #string_1 + sta.z print.s+1 + jsr print + // print(s) + lda #s + sta.z print.s+1 + jsr print + // } + rts +} +print2: { + // print("string1") + lda #string_0 + sta.z print.s+1 + jsr print + // print("string2") + lda #string_1 + sta.z print.s+1 + jsr print + // print(s1) + lda #s1 + sta.z print.s+1 + jsr print + // } + rts +} +// print(byte* zp(4) s) +print: { + .label s = 4 + __b1: + // while(*s) + ldy #0 + lda (s),y + cmp #0 + bne __b2 + // } + rts + __b2: + // *screen++ = *s++ + ldy #0 + lda (s),y + sta (screen),y + // *screen++ = *s++; + inc.z screen + bne !+ + inc.z screen+1 + !: + inc.z s + bne !+ + inc.z s+1 + !: + jmp __b1 +} +.segment Data + s: .text "string3" + .byte 0 + s1: .text "string4" + .byte 0 + string_0: .text "string1" + .byte 0 + string_1: .text "string2" + .byte 0 diff --git a/src/test/ref/string-const-consolidation-root.cfg b/src/test/ref/string-const-consolidation-root.cfg new file mode 100644 index 000000000..f90ae9900 --- /dev/null +++ b/src/test/ref/string-const-consolidation-root.cfg @@ -0,0 +1,66 @@ + +void main() +main: scope:[main] from + [0] phi() + [1] call print1 + to:main::@1 +main::@1: scope:[main] from main + [2] phi() + [3] call print2 + to:main::@return +main::@return: scope:[main] from main::@1 + [4] return + to:@return + +void print1() +print1: scope:[print1] from main + [5] phi() + [6] call print + to:print1::@1 +print1::@1: scope:[print1] from print1 + [7] phi() + [8] call print + to:print1::@2 +print1::@2: scope:[print1] from print1::@1 + [9] phi() + [10] call print + to:print1::@return +print1::@return: scope:[print1] from print1::@2 + [11] return + to:@return + +void print2() +print2: scope:[print2] from main::@1 + [12] phi() + [13] call print + to:print2::@1 +print2::@1: scope:[print2] from print2 + [14] phi() + [15] call print + to:print2::@2 +print2::@2: scope:[print2] from print2::@1 + [16] phi() + [17] call print + to:print2::@return +print2::@return: scope:[print2] from print2::@2 + [18] return + to:@return + +void print(byte* print::s) +print: scope:[print] from print1 print1::@1 print1::@2 print2 print2::@1 print2::@2 + [19] screen#36 = phi( print1/(byte*) 1024, print1::@1/screen#12, print1::@2/screen#12, print2/screen#12, print2::@1/screen#12, print2::@2/screen#12 ) + [19] print::s#9 = phi( print1/string_0, print1::@1/string_1, print1::@2/s, print2/string_0, print2::@1/string_1, print2::@2/s1 ) + to:print::@1 +print::@1: scope:[print] from print print::@2 + [20] screen#12 = phi( print/screen#36, print::@2/screen#11 ) + [20] print::s#7 = phi( print/print::s#9, print::@2/print::s#6 ) + [21] if(0!=*print::s#7) goto print::@2 + to:print::@return +print::@return: scope:[print] from print::@1 + [22] return + to:@return +print::@2: scope:[print] from print::@1 + [23] *screen#12 = *print::s#7 + [24] screen#11 = ++ screen#12 + [25] print::s#6 = ++ print::s#7 + to:print::@1 diff --git a/src/test/ref/string-const-consolidation-root.log b/src/test/ref/string-const-consolidation-root.log new file mode 100644 index 000000000..88e043360 --- /dev/null +++ b/src/test/ref/string-const-consolidation-root.log @@ -0,0 +1,825 @@ +Inlined call call __init + +CONTROL FLOW GRAPH SSA + +void main() +main: scope:[main] from __start::@1 + screen#31 = phi( __start::@1/screen#35 ) + call print1 + to:main::@1 +main::@1: scope:[main] from main + screen#16 = phi( main/screen#6 ) + screen#0 = screen#16 + call print2 + to:main::@2 +main::@2: scope:[main] from main::@1 + screen#17 = phi( main::@1/screen#10 ) + screen#1 = screen#17 + to:main::@return +main::@return: scope:[main] from main::@2 + screen#18 = phi( main::@2/screen#1 ) + screen#2 = screen#18 + return + to:@return + +void print1() +print1: scope:[print1] from main + screen#32 = phi( main/screen#31 ) + print::s#0 = print1::s + call print + to:print1::@1 +print1::@1: scope:[print1] from print1 + screen#19 = phi( print1/screen#12 ) + screen#3 = screen#19 + print::s#1 = print1::s1 + call print + to:print1::@2 +print1::@2: scope:[print1] from print1::@1 + screen#20 = phi( print1::@1/screen#12 ) + screen#4 = screen#20 + print::s#2 = s + call print + to:print1::@3 +print1::@3: scope:[print1] from print1::@2 + screen#21 = phi( print1::@2/screen#12 ) + screen#5 = screen#21 + to:print1::@return +print1::@return: scope:[print1] from print1::@3 + screen#22 = phi( print1::@3/screen#5 ) + screen#6 = screen#22 + return + to:@return + +void print2() +print2: scope:[print2] from main::@1 + screen#33 = phi( main::@1/screen#0 ) + print::s#3 = print2::s + call print + to:print2::@1 +print2::@1: scope:[print2] from print2 + screen#23 = phi( print2/screen#12 ) + screen#7 = screen#23 + print::s#4 = print2::s1 + call print + to:print2::@2 +print2::@2: scope:[print2] from print2::@1 + screen#24 = phi( print2::@1/screen#12 ) + screen#8 = screen#24 + print::s#5 = s1 + call print + to:print2::@3 +print2::@3: scope:[print2] from print2::@2 + screen#25 = phi( print2::@2/screen#12 ) + screen#9 = screen#25 + to:print2::@return +print2::@return: scope:[print2] from print2::@3 + screen#26 = phi( print2::@3/screen#9 ) + screen#10 = screen#26 + return + to:@return + +void print(byte* print::s) +print: scope:[print] from print1 print1::@1 print1::@2 print2 print2::@1 print2::@2 + screen#36 = phi( print1/screen#32, print1::@1/screen#3, print1::@2/screen#4, print2/screen#33, print2::@1/screen#7, print2::@2/screen#8 ) + print::s#9 = phi( print1/print::s#0, print1::@1/print::s#1, print1::@2/print::s#2, print2/print::s#3, print2::@1/print::s#4, print2::@2/print::s#5 ) + to:print::@1 +print::@1: scope:[print] from print print::@2 + screen#34 = phi( print/screen#36, print::@2/screen#11 ) + print::s#7 = phi( print/print::s#9, print::@2/print::s#6 ) + print::$0 = 0 != *print::s#7 + if(print::$0) goto print::@2 + to:print::@return +print::@2: scope:[print] from print::@1 + screen#27 = phi( print::@1/screen#34 ) + print::s#8 = phi( print::@1/print::s#7 ) + *screen#27 = *print::s#8 + screen#11 = ++ screen#27 + print::s#6 = ++ print::s#8 + to:print::@1 +print::@return: scope:[print] from print::@1 + screen#28 = phi( print::@1/screen#34 ) + screen#12 = screen#28 + return + to:@return + +void __start() +__start: scope:[__start] from + to:__start::__init1 +__start::__init1: scope:[__start] from __start + screen#13 = (byte*)$400 + to:__start::@1 +__start::@1: scope:[__start] from __start::__init1 + screen#35 = phi( __start::__init1/screen#13 ) + call main + to:__start::@2 +__start::@2: scope:[__start] from __start::@1 + screen#29 = phi( __start::@1/screen#2 ) + screen#14 = screen#29 + to:__start::@return +__start::@return: scope:[__start] from __start::@2 + screen#30 = phi( __start::@2/screen#14 ) + screen#15 = screen#30 + return + to:@return + +SYMBOL TABLE SSA +void __start() +void main() +void print(byte* print::s) +bool~ print::$0 +byte* print::s +byte* print::s#0 +byte* print::s#1 +byte* print::s#2 +byte* print::s#3 +byte* print::s#4 +byte* print::s#5 +byte* print::s#6 +byte* print::s#7 +byte* print::s#8 +byte* print::s#9 +void print1() +const byte* print1::s[8] = "string1" +const byte* print1::s1[8] = "string2" +void print2() +const byte* print2::s[8] = "string1" +const byte* print2::s1[8] = "string2" +const byte* s = "string3" +const byte* s1 = "string4" +byte* screen +byte* screen#0 +byte* screen#1 +byte* screen#10 +byte* screen#11 +byte* screen#12 +byte* screen#13 +byte* screen#14 +byte* screen#15 +byte* screen#16 +byte* screen#17 +byte* screen#18 +byte* screen#19 +byte* screen#2 +byte* screen#20 +byte* screen#21 +byte* screen#22 +byte* screen#23 +byte* screen#24 +byte* screen#25 +byte* screen#26 +byte* screen#27 +byte* screen#28 +byte* screen#29 +byte* screen#3 +byte* screen#30 +byte* screen#31 +byte* screen#32 +byte* screen#33 +byte* screen#34 +byte* screen#35 +byte* screen#36 +byte* screen#4 +byte* screen#5 +byte* screen#6 +byte* screen#7 +byte* screen#8 +byte* screen#9 + +Adding number conversion cast (unumber) 0 in print::$0 = 0 != *print::s#7 +Successful SSA optimization PassNAddNumberTypeConversions +Simplifying constant integer cast 0 +Simplifying constant pointer cast (byte*) 1024 +Successful SSA optimization PassNCastSimplification +Finalized unsigned number type (byte) 0 +Successful SSA optimization PassNFinalizeNumberTypeConversions +Alias screen#0 = screen#16 +Alias screen#1 = screen#17 screen#18 screen#2 +Alias screen#19 = screen#3 +Alias screen#20 = screen#4 +Alias screen#21 = screen#5 screen#22 screen#6 +Alias screen#23 = screen#7 +Alias screen#24 = screen#8 +Alias screen#10 = screen#9 screen#25 screen#26 +Alias print::s#7 = print::s#8 +Alias screen#12 = screen#27 screen#34 screen#28 +Alias screen#13 = screen#35 +Alias screen#14 = screen#29 screen#30 screen#15 +Successful SSA optimization Pass2AliasElimination +Identical Phi Values screen#31 screen#13 +Identical Phi Values screen#0 screen#21 +Identical Phi Values screen#1 screen#10 +Identical Phi Values screen#32 screen#31 +Identical Phi Values screen#19 screen#12 +Identical Phi Values screen#20 screen#12 +Identical Phi Values screen#21 screen#12 +Identical Phi Values screen#33 screen#0 +Identical Phi Values screen#23 screen#12 +Identical Phi Values screen#24 screen#12 +Identical Phi Values screen#10 screen#12 +Identical Phi Values screen#14 screen#1 +Successful SSA optimization Pass2IdenticalPhiElimination +Simple Condition print::$0 [31] if(0!=*print::s#7) goto print::@2 +Successful SSA optimization Pass2ConditionalJumpSimplification +Constant print::s#0 = print1::s +Constant print::s#1 = print1::s1 +Constant print::s#2 = s +Constant print::s#3 = print2::s +Constant print::s#4 = print2::s1 +Constant print::s#5 = s1 +Constant screen#13 = (byte*) 1024 +Successful SSA optimization Pass2ConstantIdentification +Consolidated constant strings into string_0 +Consolidated constant strings into string_1 +Successful SSA optimization Pass2ConstantStringConsolidation +Removing unused procedure __start +Removing unused procedure block __start +Removing unused procedure block __start::__init1 +Removing unused procedure block __start::@1 +Removing unused procedure block __start::@2 +Removing unused procedure block __start::@return +Successful SSA optimization PassNEliminateEmptyStart +Inlining constant with var siblings print::s#0 +Inlining constant with var siblings print::s#1 +Inlining constant with var siblings print::s#2 +Inlining constant with var siblings print::s#3 +Inlining constant with var siblings print::s#4 +Inlining constant with var siblings print::s#5 +Inlining constant with var siblings screen#13 +Constant inlined print::s#0 = string_0 +Constant inlined print1::s1 = string_1 +Constant inlined print2::s1 = string_1 +Constant inlined print::s#4 = string_1 +Constant inlined print::s#3 = string_0 +Constant inlined print::s#2 = s +Constant inlined print::s#1 = string_1 +Constant inlined print2::s = string_0 +Constant inlined screen#13 = (byte*) 1024 +Constant inlined print::s#5 = s1 +Constant inlined print1::s = string_0 +Successful SSA optimization Pass2ConstantInlining +Adding NOP phi() at start of main +Adding NOP phi() at start of main::@1 +Adding NOP phi() at start of main::@2 +Adding NOP phi() at start of print1 +Adding NOP phi() at start of print1::@3 +Adding NOP phi() at start of print2::@3 +CALL GRAPH +Calls in [main] to print1:1 print2:3 +Calls in [print1] to print:7 print:9 print:11 +Calls in [print2] to print:15 print:17 print:19 + +Created 4 initial phi equivalence classes +Coalesced [8] screen#37 = screen#12 +Coalesced (already) [10] screen#38 = screen#12 +Coalesced (already) [14] screen#39 = screen#12 +Coalesced (already) [16] screen#40 = screen#12 +Coalesced (already) [18] screen#41 = screen#12 +Coalesced [23] print::s#10 = print::s#9 +Coalesced (already) [24] screen#42 = screen#36 +Coalesced [31] print::s#11 = print::s#6 +Coalesced [32] screen#43 = screen#11 +Coalesced down to 2 phi equivalence classes +Culled Empty Block label main::@2 +Culled Empty Block label print1::@3 +Culled Empty Block label print2::@3 +Adding NOP phi() at start of main +Adding NOP phi() at start of main::@1 +Adding NOP phi() at start of print1 +Adding NOP phi() at start of print1::@1 +Adding NOP phi() at start of print1::@2 +Adding NOP phi() at start of print2 +Adding NOP phi() at start of print2::@1 +Adding NOP phi() at start of print2::@2 + +FINAL CONTROL FLOW GRAPH + +void main() +main: scope:[main] from + [0] phi() + [1] call print1 + to:main::@1 +main::@1: scope:[main] from main + [2] phi() + [3] call print2 + to:main::@return +main::@return: scope:[main] from main::@1 + [4] return + to:@return + +void print1() +print1: scope:[print1] from main + [5] phi() + [6] call print + to:print1::@1 +print1::@1: scope:[print1] from print1 + [7] phi() + [8] call print + to:print1::@2 +print1::@2: scope:[print1] from print1::@1 + [9] phi() + [10] call print + to:print1::@return +print1::@return: scope:[print1] from print1::@2 + [11] return + to:@return + +void print2() +print2: scope:[print2] from main::@1 + [12] phi() + [13] call print + to:print2::@1 +print2::@1: scope:[print2] from print2 + [14] phi() + [15] call print + to:print2::@2 +print2::@2: scope:[print2] from print2::@1 + [16] phi() + [17] call print + to:print2::@return +print2::@return: scope:[print2] from print2::@2 + [18] return + to:@return + +void print(byte* print::s) +print: scope:[print] from print1 print1::@1 print1::@2 print2 print2::@1 print2::@2 + [19] screen#36 = phi( print1/(byte*) 1024, print1::@1/screen#12, print1::@2/screen#12, print2/screen#12, print2::@1/screen#12, print2::@2/screen#12 ) + [19] print::s#9 = phi( print1/string_0, print1::@1/string_1, print1::@2/s, print2/string_0, print2::@1/string_1, print2::@2/s1 ) + to:print::@1 +print::@1: scope:[print] from print print::@2 + [20] screen#12 = phi( print/screen#36, print::@2/screen#11 ) + [20] print::s#7 = phi( print/print::s#9, print::@2/print::s#6 ) + [21] if(0!=*print::s#7) goto print::@2 + to:print::@return +print::@return: scope:[print] from print::@1 + [22] return + to:@return +print::@2: scope:[print] from print::@1 + [23] *screen#12 = *print::s#7 + [24] screen#11 = ++ screen#12 + [25] print::s#6 = ++ print::s#7 + to:print::@1 + + +VARIABLE REGISTER WEIGHTS +void main() +void print(byte* print::s) +byte* print::s +byte* print::s#6 2002.0 +byte* print::s#7 1026.25 +byte* print::s#9 101.0 +void print1() +void print2() +byte* screen +byte* screen#11 1001.0 +byte* screen#12 185.8235294117647 +byte* screen#36 156.0 + +Initial phi equivalence classes +[ screen#36 screen#12 screen#11 ] +[ print::s#7 print::s#9 print::s#6 ] +Complete equivalence classes +[ screen#36 screen#12 screen#11 ] +[ print::s#7 print::s#9 print::s#6 ] +Allocated zp[2]:2 [ screen#36 screen#12 screen#11 ] +Allocated zp[2]:4 [ print::s#7 print::s#9 print::s#6 ] +REGISTER UPLIFT POTENTIAL REGISTERS +Statement [21] if(0!=*print::s#7) goto print::@2 [ screen#12 print::s#7 ] ( print1:1::print:6 [ screen#12 print::s#7 ] { } print1:1::print:8 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print1:1::print:10 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:13 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:15 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:17 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } ) always clobbers reg byte a reg byte y +Statement [23] *screen#12 = *print::s#7 [ screen#12 print::s#7 ] ( print1:1::print:6 [ screen#12 print::s#7 ] { } print1:1::print:8 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print1:1::print:10 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:13 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:15 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } print2:3::print:17 [ screen#12 print::s#7 ] { { screen#12 = screen#36 } } ) always clobbers reg byte a reg byte y +Potential registers zp[2]:2 [ screen#36 screen#12 screen#11 ] : zp[2]:2 , +Potential registers zp[2]:4 [ print::s#7 print::s#9 print::s#6 ] : zp[2]:4 , + +REGISTER UPLIFT SCOPES +Uplift Scope [print] 3,129.25: zp[2]:4 [ print::s#7 print::s#9 print::s#6 ] +Uplift Scope [] 1,342.82: zp[2]:2 [ screen#36 screen#12 screen#11 ] +Uplift Scope [main] +Uplift Scope [print1] +Uplift Scope [print2] + +Uplifting [print] best 781 combination zp[2]:4 [ print::s#7 print::s#9 print::s#6 ] +Uplifting [] best 781 combination zp[2]:2 [ screen#36 screen#12 screen#11 ] +Uplifting [main] best 781 combination +Uplifting [print1] best 781 combination +Uplifting [print2] best 781 combination + +ASSEMBLER BEFORE OPTIMIZATION + // File Comments +// Tests that multiple different identical strings are consolidated to a root variable + // Upstart + // Commodore 64 PRG executable file +.file [name="string-const-consolidation-root.prg", type="prg", segments="Program"] +.segmentdef Program [segments="Basic, Code, Data"] +.segmentdef Basic [start=$0801] +.segmentdef Code [start=$80d] +.segmentdef Data [startAfter="Code"] +.segment Basic +:BasicUpstart(main) + // Global Constants & labels + .label screen = 2 +.segment Code + // main +main: { + // [1] call print1 + // [5] phi from main to print1 [phi:main->print1] + print1_from_main: + jsr print1 + // [2] phi from main to main::@1 [phi:main->main::@1] + __b1_from_main: + jmp __b1 + // main::@1 + __b1: + // [3] call print2 + // [12] phi from main::@1 to print2 [phi:main::@1->print2] + print2_from___b1: + jsr print2 + jmp __breturn + // main::@return + __breturn: + // [4] return + rts +} + // print1 +print1: { + // [6] call print + // [19] phi from print1 to print [phi:print1->print] + print_from_print1: + // [19] phi screen#36 = (byte*) 1024 [phi:print1->print#0] -- pbuz1=pbuc1 + lda #<$400 + sta.z screen + lda #>$400 + sta.z screen+1 + // [19] phi print::s#9 = string_0 [phi:print1->print#1] -- pbuz1=pbuc1 + lda #string_0 + sta.z print.s+1 + jsr print + // [7] phi from print1 to print1::@1 [phi:print1->print1::@1] + __b1_from_print1: + jmp __b1 + // print1::@1 + __b1: + // [8] call print + // [19] phi from print1::@1 to print [phi:print1::@1->print] + print_from___b1: + // [19] phi screen#36 = screen#12 [phi:print1::@1->print#0] -- register_copy + // [19] phi print::s#9 = string_1 [phi:print1::@1->print#1] -- pbuz1=pbuc1 + lda #string_1 + sta.z print.s+1 + jsr print + // [9] phi from print1::@1 to print1::@2 [phi:print1::@1->print1::@2] + __b2_from___b1: + jmp __b2 + // print1::@2 + __b2: + // [10] call print + // [19] phi from print1::@2 to print [phi:print1::@2->print] + print_from___b2: + // [19] phi screen#36 = screen#12 [phi:print1::@2->print#0] -- register_copy + // [19] phi print::s#9 = s [phi:print1::@2->print#1] -- pbuz1=pbuc1 + lda #s + sta.z print.s+1 + jsr print + jmp __breturn + // print1::@return + __breturn: + // [11] return + rts +} + // print2 +print2: { + // [13] call print + // [19] phi from print2 to print [phi:print2->print] + print_from_print2: + // [19] phi screen#36 = screen#12 [phi:print2->print#0] -- register_copy + // [19] phi print::s#9 = string_0 [phi:print2->print#1] -- pbuz1=pbuc1 + lda #string_0 + sta.z print.s+1 + jsr print + // [14] phi from print2 to print2::@1 [phi:print2->print2::@1] + __b1_from_print2: + jmp __b1 + // print2::@1 + __b1: + // [15] call print + // [19] phi from print2::@1 to print [phi:print2::@1->print] + print_from___b1: + // [19] phi screen#36 = screen#12 [phi:print2::@1->print#0] -- register_copy + // [19] phi print::s#9 = string_1 [phi:print2::@1->print#1] -- pbuz1=pbuc1 + lda #string_1 + sta.z print.s+1 + jsr print + // [16] phi from print2::@1 to print2::@2 [phi:print2::@1->print2::@2] + __b2_from___b1: + jmp __b2 + // print2::@2 + __b2: + // [17] call print + // [19] phi from print2::@2 to print [phi:print2::@2->print] + print_from___b2: + // [19] phi screen#36 = screen#12 [phi:print2::@2->print#0] -- register_copy + // [19] phi print::s#9 = s1 [phi:print2::@2->print#1] -- pbuz1=pbuc1 + lda #s1 + sta.z print.s+1 + jsr print + jmp __breturn + // print2::@return + __breturn: + // [18] return + rts +} + // print +// print(byte* zp(4) s) +print: { + .label s = 4 + // [20] phi from print print::@2 to print::@1 [phi:print/print::@2->print::@1] + __b1_from_print: + __b1_from___b2: + // [20] phi screen#12 = screen#36 [phi:print/print::@2->print::@1#0] -- register_copy + // [20] phi print::s#7 = print::s#9 [phi:print/print::@2->print::@1#1] -- register_copy + jmp __b1 + // print::@1 + __b1: + // [21] if(0!=*print::s#7) goto print::@2 -- 0_neq__deref_pbuz1_then_la1 + ldy #0 + lda (s),y + cmp #0 + bne __b2 + jmp __breturn + // print::@return + __breturn: + // [22] return + rts + // print::@2 + __b2: + // [23] *screen#12 = *print::s#7 -- _deref_pbuz1=_deref_pbuz2 + ldy #0 + lda (s),y + ldy #0 + sta (screen),y + // [24] screen#11 = ++ screen#12 -- pbuz1=_inc_pbuz1 + inc.z screen + bne !+ + inc.z screen+1 + !: + // [25] print::s#6 = ++ print::s#7 -- pbuz1=_inc_pbuz1 + inc.z s + bne !+ + inc.z s+1 + !: + jmp __b1_from___b2 +} + // File Data +.segment Data + s: .text "string3" + .byte 0 + s1: .text "string4" + .byte 0 + string_0: .text "string1" + .byte 0 + string_1: .text "string2" + .byte 0 + +ASSEMBLER OPTIMIZATIONS +Removing instruction jmp __b1 +Removing instruction jmp __breturn +Removing instruction jmp __b1 +Removing instruction jmp __b2 +Removing instruction jmp __breturn +Removing instruction jmp __b1 +Removing instruction jmp __b2 +Removing instruction jmp __breturn +Removing instruction jmp __b1 +Removing instruction jmp __breturn +Succesful ASM optimization Pass5NextJumpElimination +Removing instruction ldy #0 +Succesful ASM optimization Pass5UnnecesaryLoadElimination +Replacing label __b1_from___b2 with __b1 +Removing instruction __b1_from_main: +Removing instruction print2_from___b1: +Removing instruction __b1_from_print1: +Removing instruction print_from___b1: +Removing instruction __b2_from___b1: +Removing instruction print_from___b2: +Removing instruction __b1_from_print2: +Removing instruction print_from___b1: +Removing instruction __b2_from___b1: +Removing instruction print_from___b2: +Removing instruction __b1_from_print: +Removing instruction __b1_from___b2: +Succesful ASM optimization Pass5RedundantLabelElimination +Removing instruction print1_from_main: +Removing instruction __b1: +Removing instruction __breturn: +Removing instruction print_from_print1: +Removing instruction __b1: +Removing instruction __b2: +Removing instruction __breturn: +Removing instruction print_from_print2: +Removing instruction __b1: +Removing instruction __b2: +Removing instruction __breturn: +Removing instruction __breturn: +Succesful ASM optimization Pass5UnusedLabelElimination + +FINAL SYMBOL TABLE +void main() +void print(byte* print::s) +byte* print::s +byte* print::s#6 s zp[2]:4 2002.0 +byte* print::s#7 s zp[2]:4 1026.25 +byte* print::s#9 s zp[2]:4 101.0 +void print1() +void print2() +const byte* s = "string3" +const byte* s1 = "string4" +byte* screen +byte* screen#11 screen zp[2]:2 1001.0 +byte* screen#12 screen zp[2]:2 185.8235294117647 +byte* screen#36 screen zp[2]:2 156.0 +const byte* string_0[8] = "string1" +const byte* string_1[8] = "string2" + +zp[2]:2 [ screen#36 screen#12 screen#11 ] +zp[2]:4 [ print::s#7 print::s#9 print::s#6 ] + + +FINAL ASSEMBLER +Score: 677 + + // File Comments +// Tests that multiple different identical strings are consolidated to a root variable + // Upstart + // Commodore 64 PRG executable file +.file [name="string-const-consolidation-root.prg", type="prg", segments="Program"] +.segmentdef Program [segments="Basic, Code, Data"] +.segmentdef Basic [start=$0801] +.segmentdef Code [start=$80d] +.segmentdef Data [startAfter="Code"] +.segment Basic +:BasicUpstart(main) + // Global Constants & labels + .label screen = 2 +.segment Code + // main +main: { + // print1() + // [1] call print1 + // [5] phi from main to print1 [phi:main->print1] + jsr print1 + // [2] phi from main to main::@1 [phi:main->main::@1] + // main::@1 + // print2() + // [3] call print2 + // [12] phi from main::@1 to print2 [phi:main::@1->print2] + jsr print2 + // main::@return + // } + // [4] return + rts +} + // print1 +print1: { + // print("string1") + // [6] call print + // [19] phi from print1 to print [phi:print1->print] + // [19] phi screen#36 = (byte*) 1024 [phi:print1->print#0] -- pbuz1=pbuc1 + lda #<$400 + sta.z screen + lda #>$400 + sta.z screen+1 + // [19] phi print::s#9 = string_0 [phi:print1->print#1] -- pbuz1=pbuc1 + lda #string_0 + sta.z print.s+1 + jsr print + // [7] phi from print1 to print1::@1 [phi:print1->print1::@1] + // print1::@1 + // print("string2") + // [8] call print + // [19] phi from print1::@1 to print [phi:print1::@1->print] + // [19] phi screen#36 = screen#12 [phi:print1::@1->print#0] -- register_copy + // [19] phi print::s#9 = string_1 [phi:print1::@1->print#1] -- pbuz1=pbuc1 + lda #string_1 + sta.z print.s+1 + jsr print + // [9] phi from print1::@1 to print1::@2 [phi:print1::@1->print1::@2] + // print1::@2 + // print(s) + // [10] call print + // [19] phi from print1::@2 to print [phi:print1::@2->print] + // [19] phi screen#36 = screen#12 [phi:print1::@2->print#0] -- register_copy + // [19] phi print::s#9 = s [phi:print1::@2->print#1] -- pbuz1=pbuc1 + lda #s + sta.z print.s+1 + jsr print + // print1::@return + // } + // [11] return + rts +} + // print2 +print2: { + // print("string1") + // [13] call print + // [19] phi from print2 to print [phi:print2->print] + // [19] phi screen#36 = screen#12 [phi:print2->print#0] -- register_copy + // [19] phi print::s#9 = string_0 [phi:print2->print#1] -- pbuz1=pbuc1 + lda #string_0 + sta.z print.s+1 + jsr print + // [14] phi from print2 to print2::@1 [phi:print2->print2::@1] + // print2::@1 + // print("string2") + // [15] call print + // [19] phi from print2::@1 to print [phi:print2::@1->print] + // [19] phi screen#36 = screen#12 [phi:print2::@1->print#0] -- register_copy + // [19] phi print::s#9 = string_1 [phi:print2::@1->print#1] -- pbuz1=pbuc1 + lda #string_1 + sta.z print.s+1 + jsr print + // [16] phi from print2::@1 to print2::@2 [phi:print2::@1->print2::@2] + // print2::@2 + // print(s1) + // [17] call print + // [19] phi from print2::@2 to print [phi:print2::@2->print] + // [19] phi screen#36 = screen#12 [phi:print2::@2->print#0] -- register_copy + // [19] phi print::s#9 = s1 [phi:print2::@2->print#1] -- pbuz1=pbuc1 + lda #s1 + sta.z print.s+1 + jsr print + // print2::@return + // } + // [18] return + rts +} + // print +// print(byte* zp(4) s) +print: { + .label s = 4 + // [20] phi from print print::@2 to print::@1 [phi:print/print::@2->print::@1] + // [20] phi screen#12 = screen#36 [phi:print/print::@2->print::@1#0] -- register_copy + // [20] phi print::s#7 = print::s#9 [phi:print/print::@2->print::@1#1] -- register_copy + // print::@1 + __b1: + // while(*s) + // [21] if(0!=*print::s#7) goto print::@2 -- 0_neq__deref_pbuz1_then_la1 + ldy #0 + lda (s),y + cmp #0 + bne __b2 + // print::@return + // } + // [22] return + rts + // print::@2 + __b2: + // *screen++ = *s++ + // [23] *screen#12 = *print::s#7 -- _deref_pbuz1=_deref_pbuz2 + ldy #0 + lda (s),y + sta (screen),y + // *screen++ = *s++; + // [24] screen#11 = ++ screen#12 -- pbuz1=_inc_pbuz1 + inc.z screen + bne !+ + inc.z screen+1 + !: + // [25] print::s#6 = ++ print::s#7 -- pbuz1=_inc_pbuz1 + inc.z s + bne !+ + inc.z s+1 + !: + jmp __b1 +} + // File Data +.segment Data + s: .text "string3" + .byte 0 + s1: .text "string4" + .byte 0 + string_0: .text "string1" + .byte 0 + string_1: .text "string2" + .byte 0 + diff --git a/src/test/ref/string-const-consolidation-root.sym b/src/test/ref/string-const-consolidation-root.sym new file mode 100644 index 000000000..6eb468cac --- /dev/null +++ b/src/test/ref/string-const-consolidation-root.sym @@ -0,0 +1,19 @@ +void main() +void print(byte* print::s) +byte* print::s +byte* print::s#6 s zp[2]:4 2002.0 +byte* print::s#7 s zp[2]:4 1026.25 +byte* print::s#9 s zp[2]:4 101.0 +void print1() +void print2() +const byte* s = "string3" +const byte* s1 = "string4" +byte* screen +byte* screen#11 screen zp[2]:2 1001.0 +byte* screen#12 screen zp[2]:2 185.8235294117647 +byte* screen#36 screen zp[2]:2 156.0 +const byte* string_0[8] = "string1" +const byte* string_1[8] = "string2" + +zp[2]:2 [ screen#36 screen#12 screen#11 ] +zp[2]:4 [ print::s#7 print::s#9 print::s#6 ]