1
0
mirror of https://gitlab.com/camelot/kickc.git synced 2025-01-23 09:33:30 +00:00

Implemented immediate variables for preserving the value of load/store variables used in simple conditions that are inlined. #359

This commit is contained in:
jespergravgaard 2020-02-23 20:57:53 +01:00
parent 6862698fb7
commit 19b145c207
8 changed files with 155 additions and 92 deletions

View File

@ -92,7 +92,25 @@ public class ControlFlowBlock implements Serializable {
return;
}
}
throw new RuntimeException("No call statement in block " + getLabel().getFullName());
throw new InternalError("No call statement in block " + getLabel().getFullName());
}
/**
* Adds a new statement after an existing predecessor statement
* @param newStatement The new statement to add
* @param predecessor The existing predecessor statement
*/
public void addStatementAfter(Statement newStatement, Statement predecessor) {
ListIterator<Statement> listIterator = statements.listIterator();
while(listIterator.hasNext()) {
Statement statement = listIterator.next();
if(statement.equals(predecessor)) {
listIterator.previous();
listIterator.add(newStatement);
return;
}
}
throw new InternalError("Predecessor not found in block " +getLabel().getFullName() + " predecessor: "+ predecessor.toString());
}
public LabelRef getDefaultSuccessor() {

View File

@ -199,7 +199,7 @@ public class ControlFlowGraph implements Serializable {
public Statement getStatementByIndex(int statementIdx) {
for(ControlFlowBlock block : getAllBlocks()) {
for(Statement statement : block.getStatements()) {
if(statementIdx == statement.getIndex()) {
if(statement.getIndex()!=null && statementIdx == statement.getIndex()) {
return statement;
}
}

View File

@ -1,18 +1,27 @@
package dk.camelot64.kickc.passes;
import dk.camelot64.kickc.model.Comment;
import dk.camelot64.kickc.model.ControlFlowBlock;
import dk.camelot64.kickc.model.Program;
import dk.camelot64.kickc.model.VariableReferenceInfos;
import dk.camelot64.kickc.model.iterator.ProgramValue;
import dk.camelot64.kickc.model.iterator.ProgramValueIterator;
import dk.camelot64.kickc.model.operators.Operator;
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.statements.StatementInfos;
import dk.camelot64.kickc.model.symbols.Scope;
import dk.camelot64.kickc.model.symbols.Variable;
import dk.camelot64.kickc.model.values.RValue;
import dk.camelot64.kickc.model.values.ScopeRef;
import dk.camelot64.kickc.model.values.SymbolRef;
import dk.camelot64.kickc.model.values.VariableRef;
import dk.camelot64.kickc.passes.utils.AliasReplacer;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.stream.Collectors;
@ -27,15 +36,38 @@ public class Pass2ConditionalJumpSimplification extends Pass2SsaOptimization {
@Override
public boolean step() {
final List<VariableRef> simpleConditionVars = getSimpleConditions();
final List<VariableRef> simpleConditionVars = new ArrayList<>();
List<SimpleCondition> simpleConditions = getSimpleConditionTodos();
for(SimpleCondition simpleCondition : simpleConditions) {
rewriteSimpleCondition(simpleCondition);
simpleConditionVars.add(simpleCondition.conditionVar);
}
removeAssignments(getGraph(), simpleConditionVars);
deleteSymbols(getScope(), simpleConditionVars);
return (simpleConditionVars.size() > 0);
}
private List<VariableRef> getSimpleConditions() {
/** An identified conditional jump with a one-variable condition that uses a comparison condition (less-than, greater-than, ...) and the assignment for that condition. */
static class SimpleCondition {
StatementConditionalJump conditionalJump;
StatementAssignment conditionAssignment;
VariableRef conditionVar;
SimpleCondition(StatementConditionalJump conditionalJump, StatementAssignment conditionAssignment, VariableRef conditionVar) {
this.conditionalJump = conditionalJump;
this.conditionAssignment = conditionAssignment;
this.conditionVar = conditionVar;
}
}
/**
* Find all simple conditions that can be rewritten.
* Simple conditions are conditional jumps with a single variable as condition. The condition must be the result of an assignment with a comparison-operator.
* @return The simple conditions to rewrite
*/
private List<SimpleCondition> getSimpleConditionTodos() {
final VariableReferenceInfos variableReferenceInfos = getProgram().getVariableReferenceInfos();
final List<VariableRef> simpleConditionVars = new ArrayList<>();
List<SimpleCondition> todos = new ArrayList<>();
for(ControlFlowBlock block : getGraph().getAllBlocks()) {
for(Statement statement : block.getStatements()) {
if(statement instanceof StatementConditionalJump) {
@ -60,35 +92,7 @@ public class Pass2ConditionalJumpSimplification extends Pass2SsaOptimization {
case "=<":
case ">=":
case "=>":
final Collection<Variable> referencedLoadStoreVariables = getReferencedLoadStoreVariables(conditionAssignment.getrValue1());
referencedLoadStoreVariables.addAll(getReferencedLoadStoreVariables(conditionAssignment.getrValue2()));
boolean isSimple = true;
if(referencedLoadStoreVariables.size() > 0) {
// Found referenced load/store variables
// Examine all statements between the conditionAssignment and conditionalJump for modifications
final StatementInfos statementInfos = getProgram().getStatementInfos();
Collection<Statement> statementsBetween = getGraph().getStatementsBetween(conditionAssignment, conditionalJump, statementInfos);
for(Statement statementBetween : statementsBetween) {
for(Variable referencedLoadStoreVariable : referencedLoadStoreVariables) {
if(variableReferenceInfos.getDefinedVars(statementBetween).contains(referencedLoadStoreVariable.getVariableRef())) {
// A referenced load/store-variable is modified in a statement between the assignment and the condition!
isSimple = false;
getLog().append("Condition not simple " + conditionVar.toString(getProgram()) + " " + conditionalJump.toString(getProgram(), false));
// TODO: Introduce intermediate variable copy of the load/store-variable and use that in the condition!
}
}
}
}
if(isSimple) {
conditionalJump.setrValue1(conditionAssignment.getrValue1());
conditionalJump.setOperator(conditionAssignment.getOperator());
conditionalJump.setrValue2(conditionAssignment.getrValue2());
simpleConditionVars.add(conditionVar);
getLog().append("Simple Condition " + conditionVar.toString(getProgram()) + " " + conditionalJump.toString(getProgram(), false));
break;
}
todos.add(new SimpleCondition(conditionalJump, conditionAssignment, conditionVar));
default:
}
}
@ -98,7 +102,64 @@ public class Pass2ConditionalJumpSimplification extends Pass2SsaOptimization {
}
}
}
return simpleConditionVars;
return todos;
}
/**
* Rewrite a simple condition - by inlining the comparison into the conditional jump.
* If the condition uses any load/store-variables that are modified between the condition assignment and the jump the values of these is preserved in a new intermediate variable.
*
* @param simpleCondition The simple condition to rewrite
*/
private void rewriteSimpleCondition(SimpleCondition simpleCondition) {
final VariableReferenceInfos variableReferenceInfos = getProgram().getVariableReferenceInfos();
// For each condition/assignment pair - do a rewrite to inline the condition if possible
final Operator operator = simpleCondition.conditionAssignment.getOperator();
RValue rValue1 = simpleCondition.conditionAssignment.getrValue1();
RValue rValue2 = simpleCondition.conditionAssignment.getrValue2();
final Collection<Variable> referencedLoadStoreVariables = getReferencedLoadStoreVariables(rValue1);
referencedLoadStoreVariables.addAll(getReferencedLoadStoreVariables(rValue2));
if(referencedLoadStoreVariables.size() > 0) {
// Found referenced load/store variables
// Examine all statements between the conditionAssignment and conditionalJump for modifications
final StatementInfos statementInfos = getProgram().getStatementInfos();
Collection<Statement> statementsBetween = getGraph().getStatementsBetween(simpleCondition.conditionAssignment, simpleCondition.conditionalJump, statementInfos);
for(Statement statementBetween : statementsBetween) {
for(Variable referencedLoadStoreVariable : referencedLoadStoreVariables) {
if(variableReferenceInfos.getDefinedVars(statementBetween).contains(referencedLoadStoreVariable.getVariableRef())) {
// A referenced load/store-variable is modified in a statement between the assignment and the condition!
getLog().append("Condition not simple " + simpleCondition.conditionVar.toString(getProgram()) + " " + simpleCondition.conditionalJump.toString(getProgram(), false));
// Create an intermediate variable copy of the load/store-variable at the position of the condition-variable to preserve the value
final ControlFlowBlock conditionDefineBlock = statementInfos.getBlock(simpleCondition.conditionAssignment);
final ScopeRef conditionDefineScopeRef = conditionDefineBlock.getScope();
final Scope conditionDefineScope = getScope().getScope(conditionDefineScopeRef);
final Variable intermediateLoadStoreVar = conditionDefineScope.addVariableIntermediate();
intermediateLoadStoreVar.setType(referencedLoadStoreVariable.getType());
final StatementAssignment intermediateLoadStoreAssignment = new StatementAssignment(intermediateLoadStoreVar.getVariableRef(), referencedLoadStoreVariable.getRef(), true, simpleCondition.conditionAssignment.getSource(), Comment.NO_COMMENTS);
conditionDefineBlock.addStatementAfter(intermediateLoadStoreAssignment, simpleCondition.conditionAssignment);
// Replace all references to the load/store variable in the expressions with the new intermediate
final LinkedHashMap<SymbolRef, RValue> aliases = new LinkedHashMap<>();
aliases.put(referencedLoadStoreVariable.getRef(), intermediateLoadStoreVar.getRef());
{
final ProgramValue.GenericValue programValue1 = new ProgramValue.GenericValue(rValue1);
ProgramValueIterator.execute(programValue1, new AliasReplacer(aliases), null, null, null);
rValue1 = (RValue) programValue1.get();
}
{
final ProgramValue.GenericValue programValue2 = new ProgramValue.GenericValue(rValue2);
ProgramValueIterator.execute(programValue2, new AliasReplacer(aliases), null, null, null);
rValue2 = (RValue) programValue2.get();
}
getLog().append("Introduced intermediate condition variable " + intermediateLoadStoreAssignment.toString(getProgram(), false));
}
}
}
}
// Perform the condition rewrite
simpleCondition.conditionalJump.setrValue1(rValue1);
simpleCondition.conditionalJump.setOperator(operator);
simpleCondition.conditionalJump.setrValue2(rValue2);
getLog().append("Simple Condition " + simpleCondition.conditionVar.toString(getProgram()) + " " + simpleCondition.conditionalJump.toString(getProgram(), false));
}
/**

View File

@ -475,7 +475,7 @@ public class Unroller {
RValue rValueNew = valueToNew(origPhiRValue.getrValue(), varsOriginalToCopied);
newPhiVariable.setrValue(blocksOriginalToCopied.get(predecessor), rValueNew);
// - Then an entry from the existing predecessor block
RValue rValue = valueToOrig(origPhiRValue.getrValue());
RValue rValue = copyValue(origPhiRValue.getrValue());
newPhiVariable.setrValue(predecessor, rValue);
// Finally remove the phi entry into the original block (since both will hit the new block)
origPhiRValuesIt.remove();
@ -530,7 +530,7 @@ public class Unroller {
*/
private static RValue valueToNew(RValue rValue, Map<SymbolVariableRef, SymbolVariableRef> definedToNewVar) {
if(rValue == null) return null;
RValue rValueCopy = valueToOrig(rValue);
RValue rValueCopy = copyValue(rValue);
ProgramValue.GenericValue genericValue = new ProgramValue.GenericValue(rValueCopy);
ProgramValueIterator.execute(genericValue, (programValue, currentStmt, stmtIt, currentBlock) -> {
Value rVal = programValue.get();
@ -550,7 +550,7 @@ public class Unroller {
* @param rValue The value to copy
* @return An exact copy of the value
*/
private static RValue valueToOrig(RValue rValue) {
public static RValue copyValue(RValue rValue) {
if(rValue == null) return null;
ProgramValue.GenericValue genericValue = new ProgramValue.GenericValue(rValue);
ProgramValueIterator.execute(genericValue, (programValue, currentStmt, stmtIt, currentBlock) -> {

View File

@ -15,14 +15,10 @@ main: {
sta SCREEN,y
// i++<4
tya
cmp #4
lda #0
rol
eor #1
// while(i++<4)
inc i
cmp #0
bne __b1
cmp #4
bcc __b1
// }
rts
i: .byte 0

View File

@ -14,9 +14,9 @@ main: scope:[main] from @1
to:main::@1
main::@1: scope:[main] from main main::@1
[5] *((const byte*) SCREEN + (byte) main::i) ← (byte) '*'
[6] (bool~) main::$0 ← (byte) main::i < (byte) 4
[6] (byte~) main::$1 ← (byte) main::i
[7] (byte) main::i ← ++ (byte) main::i
[8] if((bool~) main::$0) goto main::@1
[8] if((byte~) main::$1<(byte) 4) goto main::@1
to:main::@return
main::@return: scope:[main] from main::@1
[9] return

View File

@ -44,8 +44,9 @@ Successful SSA optimization PassNCastSimplification
Finalized unsigned number type (byte) 4
Successful SSA optimization PassNFinalizeNumberTypeConversions
Condition not simple (bool~) main::$0 [4] if((bool~) main::$0) goto main::@1
Condition not simple (bool~) main::$0 [4] if((bool~) main::$0) goto main::@1
Condition not simple (bool~) main::$0 [4] if((bool~) main::$0) goto main::@1
Introduced intermediate condition variable (byte~) main::$1 ← (byte) main::i
Simple Condition (bool~) main::$0 [4] if((byte~) main::$1<(byte) 4) goto main::@1
Successful SSA optimization Pass2ConditionalJumpSimplification
Adding NOP phi() at start of @begin
Adding NOP phi() at start of @1
Adding NOP phi() at start of @2
@ -77,9 +78,9 @@ main: scope:[main] from @1
to:main::@1
main::@1: scope:[main] from main main::@1
[5] *((const byte*) SCREEN + (byte) main::i) ← (byte) '*'
[6] (bool~) main::$0 ← (byte) main::i < (byte) 4
[6] (byte~) main::$1 ← (byte) main::i
[7] (byte) main::i ← ++ (byte) main::i
[8] if((bool~) main::$0) goto main::@1
[8] if((byte~) main::$1<(byte) 4) goto main::@1
to:main::@return
main::@return: scope:[main] from main::@1
[9] return
@ -88,17 +89,17 @@ main::@return: scope:[main] from main::@1
VARIABLE REGISTER WEIGHTS
(void()) main()
(bool~) main::$0 11.0
(byte~) main::$1 11.0
(byte) main::i loadstore 9.200000000000001
Initial phi equivalence classes
Added variable main::i to live range equivalence class [ main::i ]
Added variable main::$0 to live range equivalence class [ main::$0 ]
Added variable main::$1 to live range equivalence class [ main::$1 ]
Complete equivalence classes
[ main::i ]
[ main::$0 ]
[ main::$1 ]
Allocated mem[1] [ main::i ]
Allocated zp[1]:2 [ main::$0 ]
Allocated zp[1]:2 [ main::$1 ]
INITIAL ASM
Target platform is c64basic / MOS6502X
@ -127,7 +128,7 @@ __bend_from___b1:
__bend:
// main
main: {
.label __0 = 2
.label __1 = 2
// [4] (byte) main::i ← (byte) 0 -- vbum1=vbuc1
lda #0
sta i
@ -138,19 +139,15 @@ main: {
lda #'*'
ldy i
sta SCREEN,y
// [6] (bool~) main::$0 ← (byte) main::i < (byte) 4 -- vboz1=vbum2_lt_vbuc1
// [6] (byte~) main::$1 ← (byte) main::i -- vbuz1=vbum2
lda i
cmp #4
lda #0
rol
eor #1
sta.z __0
sta.z __1
// [7] (byte) main::i ← ++ (byte) main::i -- vbum1=_inc_vbum1
inc i
// [8] if((bool~) main::$0) goto main::@1 -- vboz1_then_la1
lda.z __0
cmp #0
bne __b1
// [8] if((byte~) main::$1<(byte) 4) goto main::@1 -- vbuz1_lt_vbuc1_then_la1
lda.z __1
cmp #4
bcc __b1
jmp __breturn
// main::@return
__breturn:
@ -163,18 +160,17 @@ main: {
REGISTER UPLIFT POTENTIAL REGISTERS
Statement [4] (byte) main::i ← (byte) 0 [ main::i ] ( main:2 [ main::i ] ) always clobbers reg byte a
Statement [5] *((const byte*) SCREEN + (byte) main::i) ← (byte) '*' [ main::i ] ( main:2 [ main::i ] ) always clobbers reg byte a reg byte y
Statement [6] (bool~) main::$0 ← (byte) main::i < (byte) 4 [ main::i main::$0 ] ( main:2 [ main::i main::$0 ] ) always clobbers reg byte a
Potential registers mem[1] [ main::i ] : mem[1] ,
Potential registers zp[1]:2 [ main::$0 ] : zp[1]:2 , reg byte a , reg byte x , reg byte y ,
Potential registers zp[1]:2 [ main::$1 ] : zp[1]:2 , reg byte a , reg byte x , reg byte y ,
REGISTER UPLIFT SCOPES
Uplift Scope [main] 11: zp[1]:2 [ main::$0 ] 9.2: mem[1] [ main::i ]
Uplift Scope [main] 11: zp[1]:2 [ main::$1 ] 9.2: mem[1] [ main::i ]
Uplift Scope []
Uplifting [main] best 392 combination reg byte a [ main::$0 ] mem[1] [ main::i ]
Uplifting [] best 392 combination
Uplifting [main] best 312 combination reg byte a [ main::$1 ] mem[1] [ main::i ]
Uplifting [] best 312 combination
Attempting to uplift remaining variables inmem[1] [ main::i ]
Uplifting [main] best 392 combination mem[1] [ main::i ]
Uplifting [main] best 312 combination mem[1] [ main::i ]
ASSEMBLER BEFORE OPTIMIZATION
// File Comments
@ -212,17 +208,13 @@ main: {
lda #'*'
ldy i
sta SCREEN,y
// [6] (bool~) main::$0 ← (byte) main::i < (byte) 4 -- vboaa=vbum1_lt_vbuc1
// [6] (byte~) main::$1 ← (byte) main::i -- vbuaa=vbum1
lda i
cmp #4
lda #0
rol
eor #1
// [7] (byte) main::i ← ++ (byte) main::i -- vbum1=_inc_vbum1
inc i
// [8] if((bool~) main::$0) goto main::@1 -- vboaa_then_la1
cmp #0
bne __b1
// [8] if((byte~) main::$1<(byte) 4) goto main::@1 -- vbuaa_lt_vbuc1_then_la1
cmp #4
bcc __b1
jmp __breturn
// main::@return
__breturn:
@ -259,17 +251,17 @@ FINAL SYMBOL TABLE
(label) @end
(const byte*) SCREEN = (byte*) 1024
(void()) main()
(bool~) main::$0 reg byte a 11.0
(byte~) main::$1 reg byte a 11.0
(label) main::@1
(label) main::@return
(byte) main::i loadstore mem[1] 9.200000000000001
mem[1] [ main::i ]
reg byte a [ main::$0 ]
reg byte a [ main::$1 ]
FINAL ASSEMBLER
Score: 327
Score: 247
// File Comments
// Test memory model
@ -300,18 +292,14 @@ main: {
ldy i
sta SCREEN,y
// i++<4
// [6] (bool~) main::$0 ← (byte) main::i < (byte) 4 -- vboaa=vbum1_lt_vbuc1
// [6] (byte~) main::$1 ← (byte) main::i -- vbuaa=vbum1
tya
cmp #4
lda #0
rol
eor #1
// while(i++<4)
// [7] (byte) main::i ← ++ (byte) main::i -- vbum1=_inc_vbum1
inc i
// [8] if((bool~) main::$0) goto main::@1 -- vboaa_then_la1
cmp #0
bne __b1
// [8] if((byte~) main::$1<(byte) 4) goto main::@1 -- vbuaa_lt_vbuc1_then_la1
cmp #4
bcc __b1
// main::@return
// }
// [9] return

View File

@ -3,10 +3,10 @@
(label) @end
(const byte*) SCREEN = (byte*) 1024
(void()) main()
(bool~) main::$0 reg byte a 11.0
(byte~) main::$1 reg byte a 11.0
(label) main::@1
(label) main::@return
(byte) main::i loadstore mem[1] 9.200000000000001
mem[1] [ main::i ]
reg byte a [ main::$0 ]
reg byte a [ main::$1 ]