1
0
mirror of https://gitlab.com/camelot/kickc.git synced 2025-04-03 08:30:49 +00:00

Fixed struct address-of handling.

This commit is contained in:
jespergravgaard 2020-02-06 15:40:27 +01:00
parent 345b154264
commit 1a978bc82d
22 changed files with 822 additions and 894 deletions

View File

@ -176,7 +176,7 @@ public class Compiler {
getLog().append(program.getScope().toString(program, false));
}
new Pass1AddressOfVolatile(program).execute();
new Pass1AddressOfHandling(program).execute();
new Pass1FixLValuesLoHi(program).execute();
new Pass1AssertNoLValueIntermediate(program).execute();
new PassNAddTypeConversionAssignment(program, false).execute();
@ -190,7 +190,7 @@ public class Compiler {
new Pass1PointerSizeofFix(program).execute(); // After this point in the code all pointer math is byte-based
new PassNSizeOfSimplification(program).execute(); // Needed to eliminate sizeof() referencing pointer value variables
new Pass2AssertTypeMatch(program).check();
new PassNAssertTypeMatch(program).check();
new Pass1ConstantifyRValue(program).execute();
new Pass1UnwindStructVariables(program).execute();
@ -262,7 +262,7 @@ public class Compiler {
List<Pass2SsaAssertion> assertions = new ArrayList<>();
//assertions.add(new Pass2AssertNoLValueObjectEquality(program));
assertions.add(new PassNAssertTypeDeref(program));
assertions.add(new Pass2AssertTypeMatch(program));
assertions.add(new PassNAssertTypeMatch(program));
assertions.add(new Pass2AssertSymbols(program));
assertions.add(new Pass2AssertBlocks(program));
assertions.add(new Pass2AssertNoCallParameters(program));

View File

@ -39,7 +39,6 @@ public class StructVariableMemberUnwinding {
return structVariables.get(ref);
}
/** Information about how a single struct variable was unwound. */
public static class VariableUnwinding {

View File

@ -304,6 +304,10 @@ public class Variable implements Symbol {
return kind;
}
public void setKind(Kind kind) {
this.kind = kind;
}
public boolean isKindConstant() {
return Kind.CONSTANT.equals(getKind());
}

View File

@ -7,14 +7,15 @@ import dk.camelot64.kickc.model.iterator.ProgramValueIterator;
import dk.camelot64.kickc.model.operators.Operators;
import dk.camelot64.kickc.model.symbols.Symbol;
import dk.camelot64.kickc.model.symbols.Variable;
import dk.camelot64.kickc.model.types.SymbolTypeStruct;
import dk.camelot64.kickc.model.values.*;
/**
* Add inferred volatile to all variables, where address-of is used
* Update variables properly if address-of is used
*/
public class Pass1AddressOfVolatile extends Pass2SsaOptimization {
public class Pass1AddressOfHandling extends Pass2SsaOptimization {
public Pass1AddressOfVolatile(Program program) {
public Pass1AddressOfHandling(Program program) {
super(program);
}
@ -26,8 +27,9 @@ public class Pass1AddressOfVolatile extends Pass2SsaOptimization {
if(rValue instanceof SymbolVariableRef) {
Symbol toSymbol = getScope().getSymbol((SymbolVariableRef) rValue);
if(toSymbol instanceof Variable) {
((Variable) toSymbol).setInferredVolatile(true);
getLog().append("Setting inferred volatile on symbol affected by address-of " + currentStmt.toString(getProgram(), false));
final Variable variable = (Variable) toSymbol;
final String stmtStr = currentStmt.toString(getProgram(), false);
updateAddressOfVariable(variable, stmtStr);
}
}
}
@ -38,9 +40,9 @@ public class Pass1AddressOfVolatile extends Pass2SsaOptimization {
if(value instanceof SymbolVariableRef) {
Symbol toSymbol = getScope().getSymbol((SymbolVariableRef) value);
if(toSymbol instanceof Variable) {
((Variable) toSymbol).setInferredVolatile(true);
final Variable variable = (Variable) toSymbol;
final String stmtStr = currentStmt==null?toSymbol.toString(getProgram()):currentStmt.toString(getProgram(), false);
getLog().append("Setting inferred volatile on symbol affected by address-of " + stmtStr);
updateAddressOfVariable(variable, stmtStr);
}
}
}
@ -48,5 +50,15 @@ public class Pass1AddressOfVolatile extends Pass2SsaOptimization {
return false;
}
private void updateAddressOfVariable(Variable variable, String stmtStr) {
if(variable.getType() instanceof SymbolTypeStruct) {
variable.setKind(Variable.Kind.LOAD_STORE);
getLog().append("Setting struct to load/store in variable affected by address-of " + stmtStr);
} else {
variable.setInferredVolatile(true);
getLog().append("Setting inferred volatile on symbol affected by address-of " + stmtStr);
}
}
}

View File

@ -5,20 +5,18 @@ import dk.camelot64.kickc.model.ControlFlowBlock;
import dk.camelot64.kickc.model.Program;
import dk.camelot64.kickc.model.statements.Statement;
import dk.camelot64.kickc.model.statements.StatementAssignment;
import dk.camelot64.kickc.model.statements.StatementConditionalJump;
import dk.camelot64.kickc.model.types.SymbolType;
import dk.camelot64.kickc.model.types.SymbolTypeConversion;
import dk.camelot64.kickc.model.types.SymbolTypeInference;
import dk.camelot64.kickc.model.values.AssignmentRValue;
import dk.camelot64.kickc.model.values.LValue;
import dk.camelot64.kickc.model.values.RValue;
/**
* Asserts that types match in all assignments and calculations
*/
public class Pass2AssertTypeMatch extends Pass2SsaAssertion {
public class PassNAssertTypeMatch extends Pass2SsaAssertion {
public Pass2AssertTypeMatch(Program program) {
public PassNAssertTypeMatch(Program program) {
super(program);
}

View File

@ -59,8 +59,7 @@ public class ValueSourceStructValueList extends ValueSourceBase {
public ValueSource getMemberUnwinding(String memberName, Program program, ProgramScope programScope, Statement currentStmt, ListIterator<Statement> stmtIt, ControlFlowBlock currentBlock) {
int memberIndex = getMemberNames(programScope).indexOf(memberName);
final RValue memberValue = valueList.getList().get(memberIndex);
final ValueSource valueSource = ValueSourceFactory.getValueSource(memberValue, program, programScope, currentStmt, stmtIt, currentBlock);
return valueSource;
return ValueSourceFactory.getValueSource(memberValue, program, programScope, currentStmt, stmtIt, currentBlock);
}
}

View File

@ -976,12 +976,10 @@ public class TestPrograms {
// compileAndCompare("struct-ptr-29");
//}
// TODO: Fix problem with stack-allocated structs that contain arrays!
// https://gitlab.com/camelot/kickc/issues/314
//@Test
//public void testStructPtr28() throws IOException, URISyntaxException {
// compileAndCompare("struct-ptr-28");
//}
@Test
public void testStructPtr28() throws IOException, URISyntaxException {
compileAndCompare("struct-ptr-28");
}
@Test
public void testStructPtr26() throws IOException, URISyntaxException {

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct foo*) main::barp ← &(struct foo) bar
Setting struct to load/store in variable affected by address-of (struct foo*) main::barp ← &(struct foo) bar
Replacing struct member reference *((struct foo*) main::barp).thing1 with member unwinding reference *((byte*~) main::$0)
Replacing struct member reference *((struct foo*) main::barp).thing2 with member unwinding reference *((byte*~) main::$1)
Identified constant variable (struct foo*) main::barp

View File

@ -2,7 +2,7 @@ Fixing struct type size struct foo to 14
Fixing struct type size struct foo to 14
Fixing struct type SIZE_OF struct foo to 14
Fixing struct type SIZE_OF struct foo to 14
Setting inferred volatile on symbol affected by address-of (struct foo*) main::barp ← &(struct foo) bar
Setting struct to load/store in variable affected by address-of (struct foo*) main::barp ← &(struct foo) bar
Replacing struct member reference *((struct foo*) main::barp).thing1 with member unwinding reference *((byte*~) main::$1)
Replacing struct member reference *((struct foo*) main::barp).thing2 with member unwinding reference *((byte*~) main::$2)
Replacing struct member reference *((struct foo*) main::barp).thing3 with member unwinding reference (byte*~) main::$3

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct A*) main::a ← &(struct A) aa
Setting struct to load/store in variable affected by address-of (struct A*) main::a ← &(struct A) aa
Replacing struct member reference *((struct A*) main::a).b with member unwinding reference *((byte*~) main::$2)
Warning! Adding boolean cast to non-boolean sub-expression *((byte*~) main::$2)
Identified constant variable (struct A*) main::a

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct Point*) main::ptr ← &(struct Point) main::point1
Setting struct to load/store in variable affected by address-of (struct Point*) main::ptr ← &(struct Point) main::point1
Adding value bulk copy *(&(struct Point) main::point1) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
Replacing struct member reference *((struct Point*) main::ptr).x with member unwinding reference *((byte*~) main::$0)
Replacing struct member reference *((struct Point*) main::ptr).y with member unwinding reference *((byte*~) main::$1)

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct Point*) main::p2 ← &(struct Point) point2
Setting struct to load/store in variable affected by address-of (struct Point*) main::p2 ← &(struct Point) point2
Adding value bulk copy *((struct Point*) main::p2) ← memcpy(*(&(struct Point) point1), struct Point, (const byte) SIZEOF_STRUCT_POINT)
Replacing struct member reference (struct Point) point2.x with member unwinding reference *((byte*)&(struct Point) point2+(const byte) OFFSET_STRUCT_POINT_X)
Replacing struct member reference (struct Point) point2.y with member unwinding reference *((byte*)&(struct Point) point2+(const byte) OFFSET_STRUCT_POINT_Y)

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct Point*) main::q ← &(struct Point) main::p
Setting struct to load/store in variable affected by address-of (struct Point*) main::q ← &(struct Point) main::p
Adding value bulk copy *(&(struct Point) main::p) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
Replacing struct member reference *((struct Point*) main::q).x with member unwinding reference *((byte*~) main::$0)
Replacing struct member reference *((struct Point*) main::q).y with member unwinding reference *((byte*~) main::$1)

View File

@ -1,4 +1,4 @@
Setting inferred volatile on symbol affected by address-of (struct Point*) main::q ← &(struct Point) main::p
Setting struct to load/store in variable affected by address-of (struct Point*) main::q ← &(struct Point) main::p
Adding value bulk copy *(&(struct Point) main::p) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
Replacing struct member reference *((struct Point*) main::q).x with member unwinding reference *((byte*~) main::$1)
Replacing struct member reference *((struct Point*) main::q).y with member unwinding reference *((byte*~) main::$2)

View File

@ -3,18 +3,20 @@
:BasicUpstart(main)
.pc = $80d "Program"
.label SCREEN = $400
.const SIZEOF_STRUCT_POINT = 2
.const OFFSET_STRUCT_POINT_Y = 1
.label idx = 2
main: {
.label ptr = point_x
.label point_x = 3
.label point_y = 4
lda #1
sta.z point_x
lda #2
sta.z point_y
ldy.z point_x
tax
.label ptr = point
.label point = 3
ldy #SIZEOF_STRUCT_POINT
!:
lda __0-1,y
sta point-1,y
dey
bne !-
ldy.z point
ldx point+OFFSET_STRUCT_POINT_Y
lda #0
sta.z idx
jsr print
@ -35,3 +37,4 @@ print: {
sty.z idx
rts
}
__0: .byte 1, 2

View File

@ -10,31 +10,30 @@
(void()) main()
main: scope:[main] from @1
[4] (byte) main::point_x#0 ← (byte) 1
[5] (byte) main::point_y#0 ← (byte) 2
[6] (byte) print::p_x#0 ← (byte) main::point_x#0
[7] (byte) print::p_y#0 ← (byte) main::point_y#0
[8] call print
[4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
[5] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point)
[6] (byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y)
[7] call print
to:main::@1
main::@1: scope:[main] from main
[9] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr)
[10] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y)
[11] call print
[8] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr)
[9] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y)
[10] call print
to:main::@return
main::@return: scope:[main] from main::@1
[12] return
[11] return
to:@return
(void()) print((byte) print::p_x , (byte) print::p_y)
print: scope:[print] from main main::@1
[13] (byte) print::p_y#2 ← phi( main/(byte) print::p_y#0 main::@1/(byte) print::p_y#1 )
[13] (byte) idx#11 ← phi( main/(byte) 0 main::@1/(byte) idx#12 )
[13] (byte) print::p_x#2 ← phi( main/(byte) print::p_x#0 main::@1/(byte) print::p_x#1 )
[14] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2
[15] (byte) idx#4 ← ++ (byte) idx#11
[16] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2
[17] (byte) idx#12 ← ++ (byte) idx#4
[12] (byte) print::p_y#2 ← phi( main/(byte) print::p_y#0 main::@1/(byte) print::p_y#1 )
[12] (byte) idx#11 ← phi( main/(byte) 0 main::@1/(byte) idx#12 )
[12] (byte) print::p_x#2 ← phi( main/(byte) print::p_x#0 main::@1/(byte) print::p_x#1 )
[13] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2
[14] (byte) idx#4 ← ++ (byte) idx#11
[15] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2
[16] (byte) idx#12 ← ++ (byte) idx#4
to:print::@return
print::@return: scope:[print] from print
[18] return
[17] return
to:@return

View File

@ -1,15 +1,10 @@
Setting inferred volatile on symbol affected by address-of (struct Point*) main::ptr ← &(struct Point) main::point
Created struct value member variable (byte) main::point_x
Created struct value member variable (byte) main::point_y
Converted struct value to member variables (struct Point) main::point
Setting struct to load/store in variable affected by address-of (struct Point*) main::ptr ← &(struct Point) main::point
Created struct value member variable (byte) print::p_x
Created struct value member variable (byte) print::p_y
Converted struct value to member variables (struct Point) print::p
Converted procedure struct value parameter to member unwinding (void()) print((byte) print::p_x , (byte) print::p_y)
Unwinding value copy (struct Point) main::point ← { x: (byte) 1, y: (byte) 2 }
Adding value simple copy (byte) main::point_x ← (byte) 1
Adding value simple copy (byte) main::point_y ← (byte) 2
Converted procedure struct value parameter to member unwinding in call (void~) main::$0 ← call print (byte) main::point_x (byte) main::point_y
Adding value bulk copy *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
Converted procedure struct value parameter to member unwinding in call (void~) main::$0 ← call print *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_X) *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y)
Converted procedure struct value parameter to member unwinding in call (void~) main::$1 ← call print *((byte*~) main::$2) *((byte*~) main::$3)
Replacing struct member reference (struct Point) print::p.x with member unwinding reference (byte) print::p_x
Replacing struct member reference (struct Point) print::p.y with member unwinding reference (byte) print::p_y
@ -24,11 +19,10 @@ CONTROL FLOW GRAPH SSA
(void()) main()
main: scope:[main] from @2
(byte) idx#14 ← phi( @2/(byte) idx#15 )
(byte) main::point_x#0 ← (byte) 1
(byte) main::point_y#0 ← (byte) 2
(struct Point) main::point#0 ← struct-unwound {(byte) main::point_x#0, (byte) main::point_y#0}
(byte) print::p_x#0 ← (byte) main::point_x#0
(byte) print::p_y#0 ← (byte) main::point_y#0
*(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
(struct Point) main::point ← struct-unwound {*(&(struct Point) main::point)}
(byte) print::p_x#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_X)
(byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y)
call print
to:main::@1
main::@1: scope:[main] from main
@ -76,6 +70,7 @@ print::@return: scope:[print] from print
@end: scope:[] from @3
SYMBOL TABLE SSA
(const struct Point) $0 = { x: (byte) 1, y: (byte) 2 }
(label) @2
(label) @3
(label) @begin
@ -85,6 +80,7 @@ SYMBOL TABLE SSA
(byte) Point::x
(byte) Point::y
(const byte*) SCREEN = (byte*)(number) $400
(const byte) SIZEOF_STRUCT_POINT = (byte) 2
(byte) idx
(byte) idx#0
(byte) idx#1
@ -108,12 +104,7 @@ SYMBOL TABLE SSA
(label) main::@1
(label) main::@2
(label) main::@return
(struct Point) main::point
(struct Point) main::point#0
(byte) main::point_x
(byte) main::point_x#0
(byte) main::point_y
(byte) main::point_y#0
(struct Point) main::point loadstore
(const struct Point*) main::ptr = &(struct Point) main::point
(void()) print((byte) print::p_x , (byte) print::p_y)
(label) print::@return
@ -140,18 +131,17 @@ Identical Phi Values (byte) idx#1 (byte) idx#12
Identical Phi Values (byte) idx#10 (byte) idx#12
Identical Phi Values (byte) idx#13 (byte) idx#10
Successful SSA optimization Pass2IdenticalPhiElimination
Rewriting struct address-of to first member &(struct Point) main::point
Successful SSA optimization PassNStructAddressOfRewriting
Constant right-side identified [10] (byte*~) main::$2 ← (byte*)(const struct Point*) main::ptr + (const byte) OFFSET_STRUCT_POINT_X
Constant right-side identified [11] (byte*~) main::$3 ← (byte*)(const struct Point*) main::ptr + (const byte) OFFSET_STRUCT_POINT_Y
Removing C-classic struct-unwound assignment [3] (struct Point) main::point ← struct-unwound {*(&(struct Point) main::point)}
Constant right-side identified [9] (byte*~) main::$2 ← (byte*)(const struct Point*) main::ptr + (const byte) OFFSET_STRUCT_POINT_X
Constant right-side identified [10] (byte*~) main::$3 ← (byte*)(const struct Point*) main::ptr + (const byte) OFFSET_STRUCT_POINT_Y
Successful SSA optimization Pass2ConstantRValueConsolidation
Constant (const byte) idx#0 = 0
Constant (const byte*) main::$2 = (byte*)main::ptr+OFFSET_STRUCT_POINT_X
Constant (const byte*) main::$3 = (byte*)main::ptr+OFFSET_STRUCT_POINT_Y
Successful SSA optimization Pass2ConstantIdentification
Simplifying expression containing zero (byte*)main::ptr in
Simplifying expression containing zero (byte*)&main::point in [4] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_X)
Successful SSA optimization PassNSimplifyExpressionWithZero
Eliminating unused variable (struct Point) main::point#0 and assignment [2] (struct Point) main::point#0 ← struct-unwound {(byte) main::point_x#0, (byte) main::point_y#0}
Eliminating unused constant (const byte) OFFSET_STRUCT_POINT_X
Successful SSA optimization PassNEliminateUnusedVars
Inlining constant with var siblings (const byte) idx#0
@ -166,14 +156,14 @@ Adding NOP phi() at start of @end
Adding NOP phi() at start of main::@2
CALL GRAPH
Calls in [] to main:2
Calls in [main] to print:11 print:17
Calls in [main] to print:10 print:16
Created 3 initial phi equivalence classes
Coalesced [9] print::p_x#3 ← print::p_x#0
Coalesced [10] print::p_y#3 ← print::p_y#0
Coalesced [14] print::p_x#4 ← print::p_x#1
Coalesced [15] idx#16 ← idx#12
Coalesced [16] print::p_y#4 ← print::p_y#1
Coalesced [8] print::p_x#3 ← print::p_x#0
Coalesced [9] print::p_y#3 ← print::p_y#0
Coalesced [13] print::p_x#4 ← print::p_x#1
Coalesced [14] idx#16 ← idx#12
Coalesced [15] print::p_y#4 ← print::p_y#1
Coalesced down to 3 phi equivalence classes
Culled Empty Block (label) @3
Culled Empty Block (label) main::@2
@ -195,33 +185,32 @@ FINAL CONTROL FLOW GRAPH
(void()) main()
main: scope:[main] from @1
[4] (byte) main::point_x#0 ← (byte) 1
[5] (byte) main::point_y#0 ← (byte) 2
[6] (byte) print::p_x#0 ← (byte) main::point_x#0
[7] (byte) print::p_y#0 ← (byte) main::point_y#0
[8] call print
[4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT)
[5] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point)
[6] (byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y)
[7] call print
to:main::@1
main::@1: scope:[main] from main
[9] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr)
[10] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y)
[11] call print
[8] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr)
[9] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y)
[10] call print
to:main::@return
main::@return: scope:[main] from main::@1
[12] return
[11] return
to:@return
(void()) print((byte) print::p_x , (byte) print::p_y)
print: scope:[print] from main main::@1
[13] (byte) print::p_y#2 ← phi( main/(byte) print::p_y#0 main::@1/(byte) print::p_y#1 )
[13] (byte) idx#11 ← phi( main/(byte) 0 main::@1/(byte) idx#12 )
[13] (byte) print::p_x#2 ← phi( main/(byte) print::p_x#0 main::@1/(byte) print::p_x#1 )
[14] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2
[15] (byte) idx#4 ← ++ (byte) idx#11
[16] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2
[17] (byte) idx#12 ← ++ (byte) idx#4
[12] (byte) print::p_y#2 ← phi( main/(byte) print::p_y#0 main::@1/(byte) print::p_y#1 )
[12] (byte) idx#11 ← phi( main/(byte) 0 main::@1/(byte) idx#12 )
[12] (byte) print::p_x#2 ← phi( main/(byte) print::p_x#0 main::@1/(byte) print::p_x#1 )
[13] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2
[14] (byte) idx#4 ← ++ (byte) idx#11
[15] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2
[16] (byte) idx#12 ← ++ (byte) idx#4
to:print::@return
print::@return: scope:[print] from print
[18] return
[17] return
to:@return
@ -233,11 +222,7 @@ VARIABLE REGISTER WEIGHTS
(byte) idx#12 0.8
(byte) idx#4 3.0
(void()) main()
(struct Point) main::point
(byte) main::point_x
(byte) main::point_x#0 2.0
(byte) main::point_y
(byte) main::point_y#0 2.0
(struct Point) main::point loadstore
(void()) print((byte) print::p_x , (byte) print::p_y)
(struct Point) print::p
(byte) print::p_x
@ -254,19 +239,18 @@ Initial phi equivalence classes
[ idx#11 idx#12 ]
[ print::p_y#2 print::p_y#0 print::p_y#1 ]
Added variable idx#4 to live range equivalence class [ idx#4 ]
Added variable main::point to live range equivalence class [ main::point ]
Complete equivalence classes
[ print::p_x#2 print::p_x#0 print::p_x#1 ]
[ idx#11 idx#12 ]
[ print::p_y#2 print::p_y#0 print::p_y#1 ]
[ main::point_x#0 ]
[ main::point_y#0 ]
[ idx#4 ]
[ main::point ]
Allocated zp[1]:2 [ print::p_x#2 print::p_x#0 print::p_x#1 ]
Allocated zp[1]:3 [ idx#11 idx#12 ]
Allocated zp[1]:4 [ print::p_y#2 print::p_y#0 print::p_y#1 ]
Allocated zp[1]:5 [ main::point_x#0 ]
Allocated zp[1]:6 [ main::point_y#0 ]
Allocated zp[1]:7 [ idx#4 ]
Allocated zp[1]:5 [ idx#4 ]
Allocated zp[2]:6 [ main::point ]
INITIAL ASM
Target platform is c64basic / MOS6502X
@ -278,8 +262,9 @@ Target platform is c64basic / MOS6502X
.pc = $80d "Program"
// Global Constants & labels
.label SCREEN = $400
.const SIZEOF_STRUCT_POINT = 2
.const OFFSET_STRUCT_POINT_Y = 1
.label idx = 7
.label idx = 5
.label idx_1 = 3
// @begin
__bbegin:
@ -297,50 +282,50 @@ __bend_from___b1:
__bend:
// main
main: {
.label ptr = point_x
.label point_x = 5
.label point_y = 6
// [4] (byte) main::point_x#0 ← (byte) 1 -- vbuz1=vbuc1
lda #1
sta.z point_x
// [5] (byte) main::point_y#0 ← (byte) 2 -- vbuz1=vbuc1
lda #2
sta.z point_y
// [6] (byte) print::p_x#0 ← (byte) main::point_x#0 -- vbuz1=vbuz2
lda.z point_x
.label ptr = point
.label point = 6
// [4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT) -- _deref_pssc1=_deref_pssc2_memcpy_vbuc3
ldy #SIZEOF_STRUCT_POINT
!:
lda __0-1,y
sta point-1,y
dey
bne !-
// [5] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point) -- vbuz1=_deref_pbuc1
lda.z point
sta.z print.p_x
// [7] (byte) print::p_y#0 ← (byte) main::point_y#0 -- vbuz1=vbuz2
lda.z point_y
// [6] (byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuz1=_deref_pbuc1
lda point+OFFSET_STRUCT_POINT_Y
sta.z print.p_y
// [8] call print
// [13] phi from main to print [phi:main->print]
// [7] call print
// [12] phi from main to print [phi:main->print]
print_from_main:
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
lda #0
sta.z idx_1
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
jsr print
jmp __b1
// main::@1
__b1:
// [9] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuz1=_deref_pbuc1
// [8] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuz1=_deref_pbuc1
lda.z ptr
sta.z print.p_x
// [10] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuz1=_deref_pbuc1
// [9] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuz1=_deref_pbuc1
lda ptr+OFFSET_STRUCT_POINT_Y
sta.z print.p_y
// [11] call print
// [13] phi from main::@1 to print [phi:main::@1->print]
// [10] call print
// [12] phi from main::@1 to print [phi:main::@1->print]
print_from___b1:
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
jsr print
jmp __breturn
// main::@return
__breturn:
// [12] return
// [11] return
rts
}
// print
@ -348,59 +333,53 @@ main: {
print: {
.label p_x = 2
.label p_y = 4
// [14] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuz2
// [13] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuz2
lda.z p_x
ldy.z idx_1
sta SCREEN,y
// [15] (byte) idx#4 ← ++ (byte) idx#11 -- vbuz1=_inc_vbuz2
// [14] (byte) idx#4 ← ++ (byte) idx#11 -- vbuz1=_inc_vbuz2
ldy.z idx_1
iny
sty.z idx
// [16] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuz1=vbuz2
// [15] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuz1=vbuz2
lda.z p_y
ldy.z idx
sta SCREEN,y
// [17] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuz2
// [16] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuz2
ldy.z idx
iny
sty.z idx_1
jmp __breturn
// print::@return
__breturn:
// [18] return
// [17] return
rts
}
// File Data
__0: .byte 1, 2
REGISTER UPLIFT POTENTIAL REGISTERS
Statement [4] (byte) main::point_x#0 ← (byte) 1 [ main::point_x#0 ] ( main:2 [ main::point_x#0 ] ) always clobbers reg byte a
Statement [5] (byte) main::point_y#0 ← (byte) 2 [ main::point_x#0 main::point_y#0 ] ( main:2 [ main::point_x#0 main::point_y#0 ] ) always clobbers reg byte a
Statement [4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT) [ main::point ] ( main:2 [ main::point ] ) always clobbers reg byte a reg byte y
Potential registers zp[1]:2 [ print::p_x#2 print::p_x#0 print::p_x#1 ] : zp[1]:2 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[1]:3 [ idx#11 idx#12 ] : zp[1]:3 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[1]:4 [ print::p_y#2 print::p_y#0 print::p_y#1 ] : zp[1]:4 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[1]:5 [ main::point_x#0 ] : zp[1]:5 ,
Potential registers zp[1]:6 [ main::point_y#0 ] : zp[1]:6 ,
Potential registers zp[1]:7 [ idx#4 ] : zp[1]:7 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[1]:5 [ idx#4 ] : zp[1]:5 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[2]:6 [ main::point ] : zp[2]:6 ,
REGISTER UPLIFT SCOPES
Uplift Scope [print] 10: zp[1]:2 [ print::p_x#2 print::p_x#0 print::p_x#1 ] 10: zp[1]:4 [ print::p_y#2 print::p_y#0 print::p_y#1 ]
Uplift Scope [] 3.8: zp[1]:3 [ idx#11 idx#12 ] 3: zp[1]:7 [ idx#4 ]
Uplift Scope [main] 2: zp[1]:5 [ main::point_x#0 ] 2: zp[1]:6 [ main::point_y#0 ]
Uplift Scope [] 3.8: zp[1]:3 [ idx#11 idx#12 ] 3: zp[1]:5 [ idx#4 ]
Uplift Scope [Point]
Uplift Scope [main] 0: zp[2]:6 [ main::point ]
Uplifting [print] best 109 combination reg byte y [ print::p_x#2 print::p_x#0 print::p_x#1 ] reg byte x [ print::p_y#2 print::p_y#0 print::p_y#1 ]
Uplifting [] best 100 combination zp[1]:3 [ idx#11 idx#12 ] reg byte y [ idx#4 ]
Uplifting [main] best 100 combination zp[1]:5 [ main::point_x#0 ] zp[1]:6 [ main::point_y#0 ]
Uplifting [Point] best 100 combination
Uplifting [print] best 116 combination reg byte y [ print::p_x#2 print::p_x#0 print::p_x#1 ] reg byte x [ print::p_y#2 print::p_y#0 print::p_y#1 ]
Uplifting [] best 107 combination zp[1]:3 [ idx#11 idx#12 ] reg byte y [ idx#4 ]
Uplifting [Point] best 107 combination
Uplifting [main] best 107 combination zp[2]:6 [ main::point ]
Attempting to uplift remaining variables inzp[1]:3 [ idx#11 idx#12 ]
Uplifting [] best 100 combination zp[1]:3 [ idx#11 idx#12 ]
Attempting to uplift remaining variables inzp[1]:5 [ main::point_x#0 ]
Uplifting [main] best 100 combination zp[1]:5 [ main::point_x#0 ]
Attempting to uplift remaining variables inzp[1]:6 [ main::point_y#0 ]
Uplifting [main] best 100 combination zp[1]:6 [ main::point_y#0 ]
Uplifting [] best 107 combination zp[1]:3 [ idx#11 idx#12 ]
Allocated (was zp[1]:3) zp[1]:2 [ idx#11 idx#12 ]
Allocated (was zp[1]:5) zp[1]:3 [ main::point_x#0 ]
Allocated (was zp[1]:6) zp[1]:4 [ main::point_y#0 ]
Allocated (was zp[2]:6) zp[2]:3 [ main::point ]
ASSEMBLER BEFORE OPTIMIZATION
// File Comments
@ -411,6 +390,7 @@ ASSEMBLER BEFORE OPTIMIZATION
.pc = $80d "Program"
// Global Constants & labels
.label SCREEN = $400
.const SIZEOF_STRUCT_POINT = 2
.const OFFSET_STRUCT_POINT_Y = 1
.label idx = 2
// @begin
@ -429,71 +409,72 @@ __bend_from___b1:
__bend:
// main
main: {
.label ptr = point_x
.label point_x = 3
.label point_y = 4
// [4] (byte) main::point_x#0 ← (byte) 1 -- vbuz1=vbuc1
lda #1
sta.z point_x
// [5] (byte) main::point_y#0 ← (byte) 2 -- vbuz1=vbuc1
lda #2
sta.z point_y
// [6] (byte) print::p_x#0 ← (byte) main::point_x#0 -- vbuyy=vbuz1
ldy.z point_x
// [7] (byte) print::p_y#0 ← (byte) main::point_y#0 -- vbuxx=vbuz1
ldx.z point_y
// [8] call print
// [13] phi from main to print [phi:main->print]
.label ptr = point
.label point = 3
// [4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT) -- _deref_pssc1=_deref_pssc2_memcpy_vbuc3
ldy #SIZEOF_STRUCT_POINT
!:
lda __0-1,y
sta point-1,y
dey
bne !-
// [5] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point) -- vbuyy=_deref_pbuc1
ldy.z point
// [6] (byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
ldx point+OFFSET_STRUCT_POINT_Y
// [7] call print
// [12] phi from main to print [phi:main->print]
print_from_main:
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
lda #0
sta.z idx
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
jsr print
jmp __b1
// main::@1
__b1:
// [9] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuyy=_deref_pbuc1
// [8] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuyy=_deref_pbuc1
ldy.z ptr
// [10] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
// [9] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
ldx ptr+OFFSET_STRUCT_POINT_Y
// [11] call print
// [13] phi from main::@1 to print [phi:main::@1->print]
// [10] call print
// [12] phi from main::@1 to print [phi:main::@1->print]
print_from___b1:
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
jsr print
jmp __breturn
// main::@return
__breturn:
// [12] return
// [11] return
rts
}
// print
// print(byte register(Y) p_x, byte register(X) p_y)
print: {
// [14] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuyy
// [13] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuyy
tya
ldy.z idx
sta SCREEN,y
// [15] (byte) idx#4 ← ++ (byte) idx#11 -- vbuyy=_inc_vbuz1
// [14] (byte) idx#4 ← ++ (byte) idx#11 -- vbuyy=_inc_vbuz1
ldy.z idx
iny
// [16] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuyy=vbuxx
// [15] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuyy=vbuxx
txa
sta SCREEN,y
// [17] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuyy
// [16] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuyy
iny
sty.z idx
jmp __breturn
// print::@return
__breturn:
// [18] return
// [17] return
rts
}
// File Data
__0: .byte 1, 2
ASSEMBLER OPTIMIZATIONS
Removing instruction jmp __b1
@ -502,7 +483,6 @@ Removing instruction jmp __b1
Removing instruction jmp __breturn
Removing instruction jmp __breturn
Succesful ASM optimization Pass5NextJumpElimination
Replacing instruction ldx.z point_y with TAX
Removing instruction ldy.z idx
Succesful ASM optimization Pass5UnnecesaryLoadElimination
Replacing label __bbegin with __b1
@ -524,6 +504,7 @@ Removing instruction __b1:
Succesful ASM optimization Pass5UnusedLabelElimination
FINAL SYMBOL TABLE
(const struct Point) $0 = { x: (byte) 1, y: (byte) 2 }
(label) @1
(label) @begin
(label) @end
@ -531,6 +512,7 @@ FINAL SYMBOL TABLE
(byte) Point::x
(byte) Point::y
(const byte*) SCREEN = (byte*) 1024
(const byte) SIZEOF_STRUCT_POINT = (byte) 2
(byte) idx
(byte) idx#11 idx zp[1]:2 3.0
(byte) idx#12 idx zp[1]:2 0.8
@ -538,12 +520,8 @@ FINAL SYMBOL TABLE
(void()) main()
(label) main::@1
(label) main::@return
(struct Point) main::point
(byte) main::point_x
(byte) main::point_x#0 point_x zp[1]:3 2.0
(byte) main::point_y
(byte) main::point_y#0 point_y zp[1]:4 2.0
(const struct Point*) main::ptr = (struct Point*)&(byte) main::point_x#0
(struct Point) main::point loadstore zp[2]:3
(const struct Point*) main::ptr = &(struct Point) main::point
(void()) print((byte) print::p_x , (byte) print::p_y)
(label) print::@return
(struct Point) print::p
@ -559,13 +537,12 @@ FINAL SYMBOL TABLE
reg byte y [ print::p_x#2 print::p_x#0 print::p_x#1 ]
zp[1]:2 [ idx#11 idx#12 ]
reg byte x [ print::p_y#2 print::p_y#0 print::p_y#1 ]
zp[1]:3 [ main::point_x#0 ]
zp[1]:4 [ main::point_y#0 ]
reg byte y [ idx#4 ]
zp[2]:3 [ main::point ]
FINAL ASSEMBLER
Score: 75
Score: 83
// File Comments
// Demonstrates problem with passing struct pointer deref as parameter to call
@ -575,6 +552,7 @@ Score: 75
.pc = $80d "Program"
// Global Constants & labels
.label SCREEN = $400
.const SIZEOF_STRUCT_POINT = 2
.const OFFSET_STRUCT_POINT_Y = 1
.label idx = 2
// @begin
@ -585,69 +563,70 @@ Score: 75
// @end
// main
main: {
.label ptr = point_x
.label point_x = 3
.label point_y = 4
.label ptr = point
.label point = 3
// point = { 1, 2 }
// [4] (byte) main::point_x#0 ← (byte) 1 -- vbuz1=vbuc1
lda #1
sta.z point_x
// [5] (byte) main::point_y#0 ← (byte) 2 -- vbuz1=vbuc1
lda #2
sta.z point_y
// [4] *(&(struct Point) main::point) ← memcpy(*(&(const struct Point) $0), struct Point, (const byte) SIZEOF_STRUCT_POINT) -- _deref_pssc1=_deref_pssc2_memcpy_vbuc3
ldy #SIZEOF_STRUCT_POINT
!:
lda __0-1,y
sta point-1,y
dey
bne !-
// print(point)
// [6] (byte) print::p_x#0 ← (byte) main::point_x#0 -- vbuyy=vbuz1
ldy.z point_x
// [7] (byte) print::p_y#0 ← (byte) main::point_y#0 -- vbuxx=vbuz1
tax
// [8] call print
// [13] phi from main to print [phi:main->print]
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
// [5] (byte) print::p_x#0 ← *((byte*)&(struct Point) main::point) -- vbuyy=_deref_pbuc1
ldy.z point
// [6] (byte) print::p_y#0 ← *((byte*)&(struct Point) main::point+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
ldx point+OFFSET_STRUCT_POINT_Y
// [7] call print
// [12] phi from main to print [phi:main->print]
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#0 [phi:main->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) 0 [phi:main->print#1] -- vbuz1=vbuc1
lda #0
sta.z idx
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#0 [phi:main->print#2] -- register_copy
jsr print
// main::@1
// print(*ptr)
// [9] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuyy=_deref_pbuc1
// [8] (byte) print::p_x#1 ← *((byte*)(const struct Point*) main::ptr) -- vbuyy=_deref_pbuc1
ldy.z ptr
// [10] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
// [9] (byte) print::p_y#1 ← *((byte*)(const struct Point*) main::ptr+(const byte) OFFSET_STRUCT_POINT_Y) -- vbuxx=_deref_pbuc1
ldx ptr+OFFSET_STRUCT_POINT_Y
// [11] call print
// [13] phi from main::@1 to print [phi:main::@1->print]
// [13] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [13] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [13] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
// [10] call print
// [12] phi from main::@1 to print [phi:main::@1->print]
// [12] phi (byte) print::p_y#2 = (byte) print::p_y#1 [phi:main::@1->print#0] -- register_copy
// [12] phi (byte) idx#11 = (byte) idx#12 [phi:main::@1->print#1] -- register_copy
// [12] phi (byte) print::p_x#2 = (byte) print::p_x#1 [phi:main::@1->print#2] -- register_copy
jsr print
// main::@return
// }
// [12] return
// [11] return
rts
}
// print
// print(byte register(Y) p_x, byte register(X) p_y)
print: {
// SCREEN[idx++] = p.x
// [14] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuyy
// [13] *((const byte*) SCREEN + (byte) idx#11) ← (byte) print::p_x#2 -- pbuc1_derefidx_vbuz1=vbuyy
tya
ldy.z idx
sta SCREEN,y
// SCREEN[idx++] = p.x;
// [15] (byte) idx#4 ← ++ (byte) idx#11 -- vbuyy=_inc_vbuz1
// [14] (byte) idx#4 ← ++ (byte) idx#11 -- vbuyy=_inc_vbuz1
iny
// SCREEN[idx++] = p.y
// [16] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuyy=vbuxx
// [15] *((const byte*) SCREEN + (byte) idx#4) ← (byte) print::p_y#2 -- pbuc1_derefidx_vbuyy=vbuxx
txa
sta SCREEN,y
// SCREEN[idx++] = p.y;
// [17] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuyy
// [16] (byte) idx#12 ← ++ (byte) idx#4 -- vbuz1=_inc_vbuyy
iny
sty.z idx
// print::@return
// }
// [18] return
// [17] return
rts
}
// File Data
__0: .byte 1, 2

View File

@ -1,3 +1,4 @@
(const struct Point) $0 = { x: (byte) 1, y: (byte) 2 }
(label) @1
(label) @begin
(label) @end
@ -5,6 +6,7 @@
(byte) Point::x
(byte) Point::y
(const byte*) SCREEN = (byte*) 1024
(const byte) SIZEOF_STRUCT_POINT = (byte) 2
(byte) idx
(byte) idx#11 idx zp[1]:2 3.0
(byte) idx#12 idx zp[1]:2 0.8
@ -12,12 +14,8 @@
(void()) main()
(label) main::@1
(label) main::@return
(struct Point) main::point
(byte) main::point_x
(byte) main::point_x#0 point_x zp[1]:3 2.0
(byte) main::point_y
(byte) main::point_y#0 point_y zp[1]:4 2.0
(const struct Point*) main::ptr = (struct Point*)&(byte) main::point_x#0
(struct Point) main::point loadstore zp[2]:3
(const struct Point*) main::ptr = &(struct Point) main::point
(void()) print((byte) print::p_x , (byte) print::p_y)
(label) print::@return
(struct Point) print::p
@ -33,6 +31,5 @@
reg byte y [ print::p_x#2 print::p_x#0 print::p_x#1 ]
zp[1]:2 [ idx#11 idx#12 ]
reg byte x [ print::p_y#2 print::p_y#0 print::p_y#1 ]
zp[1]:3 [ main::point_x#0 ]
zp[1]:4 [ main::point_y#0 ]
reg byte y [ idx#4 ]
zp[2]:3 [ main::point ]

View File

@ -1,48 +1,44 @@
// Example of a struct containing an array
// It works on the surface - but illustrates the problem with structs containing arrays treating them like pointers.
// https://gitlab.com/camelot/kickc/issues/312
// https://gitlab.com/camelot/kickc/issues/314
.pc = $801 "Basic"
:BasicUpstart(main)
.pc = $80d "Program"
.const OFFSET_STRUCT_PERSON_NAME = 1
.label SCREEN = $400
.const SIZEOF_STRUCT_PERSON = $11
.const OFFSET_STRUCT_PERSON_NAME = 1
main: {
.label jesper_id = 5
.label jesper_name = 6
.label henriette_id = 8
.label henriette_name = 9
lda #4
sta.z jesper_id
lda #<_4
sta.z jesper_name
lda #>_4
sta.z jesper_name+1
.label jesper = 8
.label henriette = $19
ldy #SIZEOF_STRUCT_PERSON
!:
lda __0-1,y
sta jesper-1,y
dey
bne !-
ldx #0
lda #<jesper_id
lda #<jesper
sta.z print_person.person
lda #>jesper_id
lda #>jesper
sta.z print_person.person+1
jsr print_person
lda #7
sta.z henriette_id
lda #<_5
sta.z henriette_name
lda #>_5
sta.z henriette_name+1
lda #<henriette_id
ldy #SIZEOF_STRUCT_PERSON
!:
lda __1-1,y
sta henriette-1,y
dey
bne !-
lda #<henriette
sta.z print_person.person
lda #>henriette_id
lda #>henriette
sta.z print_person.person+1
jsr print_person
rts
_4: .text "jesper"
.byte 0
_5: .text "henriette"
.byte 0
}
// print_person(struct Person* zeropage(2) person)
// print_person(struct Person* zp(2) person)
print_person: {
.label i = 4
.label __1 = 4
.label __2 = 6
.label person = 2
ldy #0
lda (person),y
@ -53,36 +49,43 @@ print_person: {
lda #' '
sta SCREEN,x
inx
ldy #0
__b1:
lda #OFFSET_STRUCT_PERSON_NAME
clc
adc.z person
sta.z __1
lda #0
sta.z i
b1:
ldy #OFFSET_STRUCT_PERSON_NAME
lda (person),y
sta.z $fe
iny
lda (person),y
sta.z $ff
ldy.z i
lda ($fe),y
adc.z person+1
sta.z __1+1
lda (__1),y
cmp #0
bne b2
bne __b2
lda #' '
sta SCREEN,x
inx
rts
b2:
ldy #OFFSET_STRUCT_PERSON_NAME
lda (person),y
sta.z $fe
iny
lda (person),y
sta.z $ff
ldy.z i
lda ($fe),y
__b2:
lda #OFFSET_STRUCT_PERSON_NAME
clc
adc.z person
sta.z __2
lda #0
adc.z person+1
sta.z __2+1
lda (__2),y
sta SCREEN,x
inx
inc.z i
jmp b1
iny
jmp __b1
}
DIGIT: .text "0123456789"
.byte 0
__0: .byte 4
.text "jesper"
.byte 0
.fill 9, 0
__1: .byte 7
.text "henriette"
.byte 0
.fill 6, 0

View File

@ -7,41 +7,45 @@
to:@end
@end: scope:[] from @1
[3] phi()
(void()) main()
main: scope:[main] from @1
[4] (byte) main::jesper_id#0 ← (byte) 4
[5] (byte[$10]) main::jesper_name#0 ← (const string) main::$4
[6] call print_person
[4] *(&(struct Person) main::jesper) ← memcpy(*(&(const struct Person) $0), struct Person, (const byte) SIZEOF_STRUCT_PERSON)
[5] call print_person
to:main::@1
main::@1: scope:[main] from main
[7] (byte) main::henriette_id#0 ← (byte) 7
[8] (byte[$10]) main::henriette_name#0 ← (const string) main::$5
[9] call print_person
[6] *(&(struct Person) main::henriette) ← memcpy(*(&(const struct Person) $1), struct Person, (const byte) SIZEOF_STRUCT_PERSON)
[7] call print_person
to:main::@return
main::@return: scope:[main] from main::@1
[10] return
[8] return
to:@return
(void()) print_person((struct Person*) print_person::person)
print_person: scope:[print_person] from main main::@1
[11] (byte) idx#13 ← phi( main/(byte) 0 main::@1/(byte) idx#16 )
[11] (struct Person*) print_person::person#2 ← phi( main/(struct Person*)&(byte) main::jesper_id#0 main::@1/(struct Person*)&(byte) main::henriette_id#0 )
[12] *((const byte*) SCREEN#0 + (byte) idx#13) ← *((const byte[]) DIGIT#0 + *((byte*)(struct Person*) print_person::person#2))
[13] (byte) idx#4 ← ++ (byte) idx#13
[14] *((const byte*) SCREEN#0 + (byte) idx#4) ← (byte) ' '
[15] (byte) idx#5 ← ++ (byte) idx#4
[9] (byte) idx#13 ← phi( main/(byte) 0 main::@1/(byte) idx#16 )
[9] (struct Person*) print_person::person#2 ← phi( main/&(struct Person) main::jesper main::@1/&(struct Person) main::henriette )
[10] *((const byte*) SCREEN + (byte) idx#13) ← *((const byte*) DIGIT + *((byte*)(struct Person*) print_person::person#2))
[11] (byte) idx#4 ← ++ (byte) idx#13
[12] *((const byte*) SCREEN + (byte) idx#4) ← (byte) ' '
[13] (byte) idx#5 ← ++ (byte) idx#4
to:print_person::@1
print_person::@1: scope:[print_person] from print_person print_person::@2
[16] (byte) idx#14 ← phi( print_person/(byte) idx#5 print_person::@2/(byte) idx#6 )
[16] (byte) print_person::i#2 ← phi( print_person/(byte) 0 print_person::@2/(byte) print_person::i#1 )
[17] if((byte) 0!=*(*((byte[$10]*)(struct Person*) print_person::person#2 + (const byte) OFFSET_STRUCT_PERSON_NAME) + (byte) print_person::i#2)) goto print_person::@2
[14] (byte) idx#14 ← phi( print_person/(byte) idx#5 print_person::@2/(byte) idx#6 )
[14] (byte) print_person::i#2 ← phi( print_person/(byte) 0 print_person::@2/(byte) print_person::i#1 )
[15] (byte*~) print_person::$1 ← (byte*)(struct Person*) print_person::person#2 + (const byte) OFFSET_STRUCT_PERSON_NAME
[16] if((byte) 0!=*((byte*~) print_person::$1 + (byte) print_person::i#2)) goto print_person::@2
to:print_person::@3
print_person::@3: scope:[print_person] from print_person::@1
[18] *((const byte*) SCREEN#0 + (byte) idx#14) ← (byte) ' '
[19] (byte) idx#16 ← ++ (byte) idx#14
[17] *((const byte*) SCREEN + (byte) idx#14) ← (byte) ' '
[18] (byte) idx#16 ← ++ (byte) idx#14
to:print_person::@return
print_person::@return: scope:[print_person] from print_person::@3
[20] return
[19] return
to:@return
print_person::@2: scope:[print_person] from print_person::@1
[21] *((const byte*) SCREEN#0 + (byte) idx#14) ← *(*((byte[$10]*)(struct Person*) print_person::person#2 + (const byte) OFFSET_STRUCT_PERSON_NAME) + (byte) print_person::i#2)
[20] (byte*~) print_person::$2 ← (byte*)(struct Person*) print_person::person#2 + (const byte) OFFSET_STRUCT_PERSON_NAME
[21] *((const byte*) SCREEN + (byte) idx#14) ← *((byte*~) print_person::$2 + (byte) print_person::i#2)
[22] (byte) idx#6 ← ++ (byte) idx#14
[23] (byte) print_person::i#1 ← ++ (byte) print_person::i#2
to:print_person::@1

File diff suppressed because it is too large Load Diff

View File

@ -1,52 +1,45 @@
(const struct Person) $0 = { id: (byte) 4, name: (string) "jesper" }
(const struct Person) $1 = { id: (byte) 7, name: (string) "henriette" }
(label) @1
(label) @begin
(label) @end
(byte[]) DIGIT
(const byte[]) DIGIT#0 DIGIT = (string) "0123456789"
(const byte) OFFSET_STRUCT_PERSON_NAME OFFSET_STRUCT_PERSON_NAME = (byte) 1
(const byte*) DIGIT[] = (string) "0123456789"
(const byte) OFFSET_STRUCT_PERSON_NAME = (byte) 1
(byte) Person::id
(byte[$10]) Person::name
(byte*) SCREEN
(const byte*) SCREEN#0 SCREEN = (byte*) 1024
(const byte*) Person::name[(number) $10] = { fill( $10, 0) }
(const byte*) SCREEN = (byte*) 1024
(const byte) SIZEOF_STRUCT_PERSON = (byte) $11
(byte) idx
(byte) idx#13 reg byte x 3.0
(byte) idx#14 reg byte x 9.75
(byte) idx#16 reg byte x 0.8
(byte) idx#14 reg byte x 6.5
(byte) idx#16 reg byte x 1.0
(byte) idx#4 reg byte x 3.0
(byte) idx#5 reg byte x 4.0
(byte) idx#6 reg byte x 11.0
(void()) main()
(const string) main::$4 $4 = (string) "jesper"
(const string) main::$5 $5 = (string) "henriette"
(label) main::@1
(label) main::@return
(struct Person) main::henriette
(byte) main::henriette_id
(byte) main::henriette_id#0 henriette_id zp ZP_BYTE:8 20.0
(byte[$10]) main::henriette_name
(byte[$10]) main::henriette_name#0 henriette_name zp ZP_WORD:9 20.0
(struct Person) main::jesper
(byte) main::jesper_id
(byte) main::jesper_id#0 jesper_id zp ZP_BYTE:5 20.0
(byte[$10]) main::jesper_name
(byte[$10]) main::jesper_name#0 jesper_name zp ZP_WORD:6 20.0
(struct Person) main::henriette loadstore zp[17]:25
(struct Person) main::jesper loadstore zp[17]:8
(void()) print_person((struct Person*) print_person::person)
(byte*~) print_person::$1 zp[2]:4 22.0
(byte*~) print_person::$2 zp[2]:6 22.0
(label) print_person::@1
(label) print_person::@2
(label) print_person::@3
(label) print_person::@return
(byte) print_person::i
(byte) print_person::i#1 i zp ZP_BYTE:4 22.0
(byte) print_person::i#2 i zp ZP_BYTE:4 11.0
(byte) print_person::i#1 reg byte y 22.0
(byte) print_person::i#2 reg byte y 7.333333333333333
(struct Person*) print_person::person
(struct Person*) print_person::person#2 person zp ZP_WORD:2
(struct Person*) print_person::person#2 person zp[2]:2
zp ZP_WORD:2 [ print_person::person#2 ]
zp[2]:2 [ print_person::person#2 ]
reg byte x [ idx#13 idx#16 ]
zp ZP_BYTE:4 [ print_person::i#2 print_person::i#1 ]
reg byte y [ print_person::i#2 print_person::i#1 ]
reg byte x [ idx#14 idx#5 idx#6 ]
zp ZP_BYTE:5 [ main::jesper_id#0 ]
zp ZP_WORD:6 [ main::jesper_name#0 ]
zp ZP_BYTE:8 [ main::henriette_id#0 ]
zp ZP_WORD:9 [ main::henriette_name#0 ]
reg byte x [ idx#4 ]
zp[2]:4 [ print_person::$1 ]
zp[2]:6 [ print_person::$2 ]
zp[17]:8 [ main::jesper ]
zp[17]:25 [ main::henriette ]