mirror of
https://gitlab.com/camelot/kickc.git
synced 2025-04-05 07:40:39 +00:00
Now supporting fibonacci array generation
This commit is contained in:
parent
6f56b1bcfa
commit
b3dfb503f5
@ -101,6 +101,8 @@ public class AsmFragment {
|
||||
private static String getOperatorFragmentName(Operator operator) {
|
||||
String op = operator.getOperator();
|
||||
switch (op) {
|
||||
case "*":
|
||||
return "_star_";
|
||||
case "+":
|
||||
return "_plus_";
|
||||
case "-":
|
||||
@ -198,8 +200,8 @@ public class AsmFragment {
|
||||
bindings.put(name, value);
|
||||
return name;
|
||||
}
|
||||
} else if (value instanceof PointerDereference) {
|
||||
PointerDereference deref = (PointerDereference) value;
|
||||
} else if (value instanceof PointerDereferenceVariable) {
|
||||
PointerDereferenceVariable deref = (PointerDereferenceVariable) value;
|
||||
Variable pointer = deref.getPointer();
|
||||
RegisterAllocation.Register register = symbols.getRegister(pointer);
|
||||
if (RegisterAllocation.RegisterType.ZP_PTR_BYTE.equals(register.getType())) {
|
||||
@ -207,13 +209,21 @@ public class AsmFragment {
|
||||
bindings.put(name, value);
|
||||
return name;
|
||||
}
|
||||
} else if (value instanceof PointerDereferenceConstant) {
|
||||
PointerDereferenceConstant deref = (PointerDereferenceConstant) value;
|
||||
Constant pointer = deref.getPointer();
|
||||
if(pointer instanceof ConstantInteger) {
|
||||
String name = "coptr"+ nextConstByteIdx++;
|
||||
bindings.put(name, value);
|
||||
return name;
|
||||
}
|
||||
} else if (value instanceof ConstantInteger) {
|
||||
ConstantInteger intValue = (ConstantInteger) value;
|
||||
if (intValue.getType().equals(SymbolTypeBasic.BYTE)) {
|
||||
if (SymbolTypeBasic.BYTE.equals(intValue.getType())) {
|
||||
String name = "coby" + nextConstByteIdx++;
|
||||
bindings.put(name, value);
|
||||
return name;
|
||||
} else if (intValue.getType().equals(SymbolTypeBasic.WORD)) {
|
||||
} else if (SymbolTypeBasic.WORD.equals(intValue.getType())) {
|
||||
String name = "cowo" + nextConstByteIdx++;
|
||||
bindings.put(name, value);
|
||||
return name;
|
||||
@ -234,6 +244,9 @@ public class AsmFragment {
|
||||
*/
|
||||
public String getBoundValue(String name) {
|
||||
Value boundValue = getBinding(name);
|
||||
if(boundValue==null) {
|
||||
throw new RuntimeException("Binding not found in fragment '" + name+"'");
|
||||
}
|
||||
String bound;
|
||||
if (boundValue instanceof Variable) {
|
||||
RegisterAllocation.Register register = symbols.getRegister((Variable) boundValue);
|
||||
@ -246,8 +259,8 @@ public class AsmFragment {
|
||||
} else {
|
||||
throw new RuntimeException("Register Type not implemented " + register);
|
||||
}
|
||||
} else if(boundValue instanceof PointerDereference) {
|
||||
PointerDereference deref = (PointerDereference) boundValue;
|
||||
} else if(boundValue instanceof PointerDereferenceVariable) {
|
||||
PointerDereferenceVariable deref = (PointerDereferenceVariable) boundValue;
|
||||
Variable pointer = deref.getPointer();
|
||||
RegisterAllocation.Register register = symbols.getRegister(pointer);
|
||||
if(register instanceof RegisterAllocation.RegisterZpPointerByte) {
|
||||
@ -255,6 +268,15 @@ public class AsmFragment {
|
||||
} else {
|
||||
throw new RuntimeException("Bound Value Type not implemented " + boundValue);
|
||||
}
|
||||
} else if(boundValue instanceof PointerDereferenceConstant) {
|
||||
PointerDereferenceConstant deref = (PointerDereferenceConstant) boundValue;
|
||||
Constant pointer = deref.getPointer();
|
||||
if (pointer instanceof ConstantInteger) {
|
||||
ConstantInteger intPointer = (ConstantInteger) pointer;
|
||||
bound = Integer.toString(intPointer.getNumber());
|
||||
} else {
|
||||
throw new RuntimeException("Bound Value Type not implemented " + boundValue);
|
||||
}
|
||||
} else if (boundValue instanceof ConstantInteger) {
|
||||
ConstantInteger boundInt = (ConstantInteger) boundValue;
|
||||
if (boundInt.getType().equals(SymbolTypeBasic.BYTE)) {
|
||||
|
2
src/dk/camelot64/kickc/asm/fragment/coptr1=coby2.asm
Normal file
2
src/dk/camelot64/kickc/asm/fragment/coptr1=coby2.asm
Normal file
@ -0,0 +1,2 @@
|
||||
lda #{coby2}
|
||||
sta {coptr1}
|
@ -0,0 +1,2 @@
|
||||
cpx #{coby1}
|
||||
bcc {la1}
|
@ -0,0 +1,2 @@
|
||||
lda {cowo1}
|
||||
sta {zpby1}
|
@ -0,0 +1,3 @@
|
||||
ldy #0
|
||||
lda ({zpptrby1}),y
|
||||
sta {zpby1}
|
@ -0,0 +1,3 @@
|
||||
dex
|
||||
stx {zpby1}
|
||||
inx
|
@ -0,0 +1,4 @@
|
||||
txa
|
||||
sec
|
||||
sbc #{coby1}
|
||||
sta {zpby1}
|
3
src/dk/camelot64/kickc/asm/fragment/zpiby1=zpby1.asm
Normal file
3
src/dk/camelot64/kickc/asm/fragment/zpiby1=zpby1.asm
Normal file
@ -0,0 +1,3 @@
|
||||
ldy #0
|
||||
lda {zpby1}
|
||||
sta ({zpiby1}),y
|
@ -0,0 +1,7 @@
|
||||
txa
|
||||
clc
|
||||
adc #<{cowo1}
|
||||
sta {zpptrby1}
|
||||
lda #0
|
||||
adc #>{cowo1}
|
||||
sta {zpptrby1}+1
|
@ -0,0 +1,7 @@
|
||||
lda #<{cowo1}
|
||||
clc
|
||||
adc {zpby1}
|
||||
sta {zpptrby1}
|
||||
lda #>{cowo1}
|
||||
adc #0
|
||||
sta {zpptrby1}+1
|
@ -165,7 +165,7 @@ public class Pass1GenerateStatementSequence extends KickCBaseVisitor<Object> {
|
||||
VariableIntermediate tmpVar = symbolTable.newIntermediateAssignment();
|
||||
Statement stmt = new StatementAssignment(tmpVar, lval, operator, index);
|
||||
sequence.addStatement(stmt);
|
||||
return new PointerDereference(tmpVar);
|
||||
return new PointerDereferenceVariable(tmpVar);
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -218,7 +218,10 @@ public class Pass1GenerateStatementSequence extends KickCBaseVisitor<Object> {
|
||||
VariableIntermediate tmpVar = symbolTable.newIntermediateAssignment();
|
||||
Statement stmt = new StatementAssignment(tmpVar, array, operator, index);
|
||||
sequence.addStatement(stmt);
|
||||
return new PointerDereference(tmpVar);
|
||||
VariableIntermediate tmpVar2 = symbolTable.newIntermediateAssignment();
|
||||
Statement stmt2 = new StatementAssignment(tmpVar2, null, new Operator("*"), tmpVar );
|
||||
sequence.addStatement(stmt2);
|
||||
return tmpVar2;
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -5,7 +5,7 @@ import java.util.Iterator;
|
||||
import java.util.Map;
|
||||
|
||||
/** Compiler Pass eliminating alias assignments */
|
||||
public class Pass2AliasElimination extends Pass2Optimization {
|
||||
public class Pass2AliasElimination extends Pass2SsaOptimization {
|
||||
|
||||
public Pass2AliasElimination(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
super(graph, symbolTable);
|
||||
|
@ -1,14 +1,13 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
/**
|
||||
* Compiler Pass simplifying conditional jumps that are simple comparisons
|
||||
*/
|
||||
public class Pass2ConditionalJumpSimplification extends Pass2Optimization {
|
||||
public class Pass2ConditionalJumpSimplification extends Pass2SsaOptimization {
|
||||
|
||||
private Map<Variable, List<Statement>> allUsages;
|
||||
|
||||
|
@ -4,7 +4,7 @@ import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
|
||||
/** Compiler Pass propagating constants in expressions eliminating constant variables */
|
||||
public class Pass2ConstantPropagation extends Pass2Optimization {
|
||||
public class Pass2ConstantPropagation extends Pass2SsaOptimization {
|
||||
|
||||
public Pass2ConstantPropagation(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
super(graph, symbolTable);
|
||||
@ -47,7 +47,9 @@ public class Pass2ConstantPropagation extends Pass2Optimization {
|
||||
} else {
|
||||
// Constant unary expression
|
||||
Constant constant = calculateUnary(assignment.getOperator(), (Constant) assignment.getRValue2());
|
||||
constants.put(variable, constant);
|
||||
if(constant!=null) {
|
||||
constants.put(variable, constant);
|
||||
}
|
||||
}
|
||||
} else if (assignment.getRValue1() instanceof Constant && assignment.getRValue2() instanceof Constant) {
|
||||
// Constant binary expression
|
||||
@ -141,6 +143,9 @@ public class Pass2ConstantPropagation extends Pass2Optimization {
|
||||
case "+": {
|
||||
return c;
|
||||
}
|
||||
case "*": { // pointer dereference
|
||||
return null;
|
||||
}
|
||||
default:
|
||||
throw new RuntimeException("Unhandled Unary Operator " + operator.getOperator());
|
||||
}
|
||||
|
@ -3,7 +3,7 @@ package dk.camelot64.kickc.icl;
|
||||
import java.util.*;
|
||||
|
||||
/** Pass that culls empty control flow blocks from the program */
|
||||
public class Pass2CullEmptyBlocks extends Pass2Optimization {
|
||||
public class Pass2CullEmptyBlocks extends Pass2SsaOptimization {
|
||||
|
||||
public Pass2CullEmptyBlocks(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
super(graph, symbolTable);
|
||||
|
@ -4,7 +4,7 @@ import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
|
||||
/** Compiler Pass eliminating redundant phi functions */
|
||||
public class Pass2RedundantPhiElimination extends Pass2Optimization{
|
||||
public class Pass2RedundantPhiElimination extends Pass2SsaOptimization {
|
||||
|
||||
public Pass2RedundantPhiElimination(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
super(graph, symbolTable);
|
||||
|
@ -1,11 +1,9 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
import java.util.HashMap;
|
||||
import java.util.Iterator;
|
||||
import java.util.Map;
|
||||
|
||||
/** Compiler Pass eliminating phi self assignments */
|
||||
public class Pass2SelfPhiElimination extends Pass2Optimization{
|
||||
public class Pass2SelfPhiElimination extends Pass2SsaOptimization {
|
||||
|
||||
public Pass2SelfPhiElimination(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
super(graph, symbolTable);
|
||||
|
@ -3,15 +3,15 @@ package dk.camelot64.kickc.icl;
|
||||
import java.util.*;
|
||||
|
||||
/**
|
||||
* Optimization performed during Compiler Pass 2.
|
||||
* Optimization on Single Static Assignment form (Control Flow Graph) performed during Compiler Pass 2.
|
||||
* Optimizations are performed repeatedly until none of them yield any result
|
||||
* */
|
||||
public abstract class Pass2Optimization {
|
||||
*/
|
||||
public abstract class Pass2SsaOptimization {
|
||||
|
||||
private ControlFlowGraph graph;
|
||||
private SymbolTable symbolTable;
|
||||
|
||||
public Pass2Optimization(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
public Pass2SsaOptimization(ControlFlowGraph graph, SymbolTable symbolTable) {
|
||||
this.graph = graph;
|
||||
this.symbolTable = symbolTable;
|
||||
}
|
||||
@ -31,7 +31,9 @@ public abstract class Pass2Optimization {
|
||||
*/
|
||||
public abstract boolean optimize();
|
||||
|
||||
/** Singleton signalling that an RValue is never assigned and can safely be discarded as rvalue in phi-functions.*/
|
||||
/**
|
||||
* Singleton signalling that an RValue is never assigned and can safely be discarded as rvalue in phi-functions.
|
||||
*/
|
||||
public static RValue VOID = new RValue() {
|
||||
@Override
|
||||
public String toString() {
|
||||
@ -41,33 +43,52 @@ public abstract class Pass2Optimization {
|
||||
|
||||
/**
|
||||
* Replace all usages of variables in statements with aliases.
|
||||
*
|
||||
* @param aliases Variables that have alias values.
|
||||
*/
|
||||
public void replaceVariables(final Map<Variable, ? extends RValue> aliases) {
|
||||
ControlFlowGraphBaseVisitor<Void> visitor = new ControlFlowGraphBaseVisitor<Void>() {
|
||||
@Override
|
||||
public Void visitAssignment(StatementAssignment assignment) {
|
||||
if(getAlias(aliases, assignment.getLValue()) !=null) {
|
||||
RValue alias = getAlias(aliases, assignment.getLValue());
|
||||
if(alias instanceof LValue) {
|
||||
LValue lValue = assignment.getLValue();
|
||||
if (getAlias(aliases, lValue) != null) {
|
||||
RValue alias = getAlias(aliases, lValue);
|
||||
if (alias instanceof LValue) {
|
||||
assignment.setLValue((LValue) alias);
|
||||
} else {
|
||||
throw new RuntimeException("Error replacing LValue variable " + lValue + " with " + alias);
|
||||
}
|
||||
}
|
||||
if(getAlias(aliases, assignment.getRValue1())!=null) {
|
||||
if (getAlias(aliases, assignment.getRValue1()) != null) {
|
||||
assignment.setRValue1(getAlias(aliases, assignment.getRValue1()));
|
||||
}
|
||||
if(getAlias(aliases, assignment.getRValue2())!=null) {
|
||||
if (getAlias(aliases, assignment.getRValue2()) != null) {
|
||||
assignment.setRValue2(getAlias(aliases, assignment.getRValue2()));
|
||||
}
|
||||
// Handle pointer dereference in LValue
|
||||
if (lValue instanceof PointerDereferenceVariable) {
|
||||
PointerDereferenceVariable deref = (PointerDereferenceVariable) lValue;
|
||||
Variable pointer = deref.getPointer();
|
||||
if (getAlias(aliases, pointer) != null) {
|
||||
RValue alias = getAlias(aliases, pointer);
|
||||
if (alias instanceof Variable) {
|
||||
deref.setPointerVariable((Variable) alias);
|
||||
} else if (alias instanceof Constant) {
|
||||
assignment.setLValue(new PointerDereferenceConstant((Constant) alias));
|
||||
} else {
|
||||
throw new RuntimeException("Error replacing LValue variable " + lValue + " with " + alias);
|
||||
}
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Void visitConditionalJump(StatementConditionalJump conditionalJump) {
|
||||
if(getAlias(aliases, conditionalJump.getRValue1())!=null) {
|
||||
if (getAlias(aliases, conditionalJump.getRValue1()) != null) {
|
||||
conditionalJump.setRValue1(getAlias(aliases, conditionalJump.getRValue1()));
|
||||
}
|
||||
if(getAlias(aliases, conditionalJump.getRValue2())!=null) {
|
||||
if (getAlias(aliases, conditionalJump.getRValue2()) != null) {
|
||||
conditionalJump.setRValue2(getAlias(aliases, conditionalJump.getRValue2()));
|
||||
}
|
||||
return null;
|
||||
@ -75,9 +96,9 @@ public abstract class Pass2Optimization {
|
||||
|
||||
@Override
|
||||
public Void visitPhi(StatementPhi phi) {
|
||||
if(getAlias(aliases, phi.getLValue())!=null) {
|
||||
if (getAlias(aliases, phi.getLValue()) != null) {
|
||||
RValue alias = getAlias(aliases, phi.getLValue());
|
||||
if(alias instanceof LValue) {
|
||||
if (alias instanceof LValue) {
|
||||
phi.setLValue((Variable) alias);
|
||||
}
|
||||
}
|
||||
@ -98,15 +119,16 @@ public abstract class Pass2Optimization {
|
||||
visitor.visitGraph(graph);
|
||||
}
|
||||
|
||||
/** Get the alias to use for an RValue.
|
||||
/**
|
||||
* Get the alias to use for an RValue.
|
||||
*
|
||||
* @param aliases The alias map
|
||||
* @param rValue The RValue to find an alias for
|
||||
* @param rValue The RValue to find an alias for
|
||||
* @return The alias to use. Null if no alias exists.
|
||||
*/
|
||||
private static RValue getAlias(Map<Variable, ? extends RValue> aliases,RValue rValue) {
|
||||
private static RValue getAlias(Map<Variable, ? extends RValue> aliases, RValue rValue) {
|
||||
RValue alias = aliases.get(rValue);
|
||||
while (aliases.get(alias)!=null) {
|
||||
while (aliases.get(alias) != null) {
|
||||
alias = aliases.get(alias);
|
||||
}
|
||||
return alias;
|
||||
@ -114,14 +136,15 @@ public abstract class Pass2Optimization {
|
||||
|
||||
/**
|
||||
* Replace all usages of a label in statements with another label.
|
||||
*
|
||||
* @param replacements Variables that have alias values.
|
||||
*/
|
||||
public void replaceLabels(final Map<Label, Label> replacements) {
|
||||
ControlFlowGraphBaseVisitor<Void> visitor = new ControlFlowGraphBaseVisitor<Void>() {
|
||||
|
||||
@Override
|
||||
public Void visitConditionalJump(StatementConditionalJump conditionalJump) {
|
||||
if(getReplacement(replacements, conditionalJump.getDestination())!=null) {
|
||||
public Void visitConditionalJump(StatementConditionalJump conditionalJump) {
|
||||
if (getReplacement(replacements, conditionalJump.getDestination()) != null) {
|
||||
conditionalJump.setDestination(getReplacement(replacements, conditionalJump.getDestination()));
|
||||
}
|
||||
return null;
|
||||
@ -129,7 +152,7 @@ public abstract class Pass2Optimization {
|
||||
|
||||
@Override
|
||||
public Void visitJump(StatementJump jump) {
|
||||
if(getReplacement(replacements, jump.getDestination())!=null) {
|
||||
if (getReplacement(replacements, jump.getDestination()) != null) {
|
||||
jump.setDestination(getReplacement(replacements, jump.getDestination()));
|
||||
}
|
||||
return null;
|
||||
@ -139,7 +162,7 @@ public abstract class Pass2Optimization {
|
||||
public Void visitPhi(StatementPhi phi) {
|
||||
for (StatementPhi.PreviousSymbol previousSymbol : phi.getPreviousVersions()) {
|
||||
Label replacement = getReplacement(replacements, previousSymbol.getBlock().getLabel());
|
||||
if(replacement !=null) {
|
||||
if (replacement != null) {
|
||||
previousSymbol.setBlock(graph.getBlock(replacement));
|
||||
}
|
||||
}
|
||||
@ -149,15 +172,16 @@ public abstract class Pass2Optimization {
|
||||
visitor.visitGraph(graph);
|
||||
}
|
||||
|
||||
/** Get the label to use as replacement for another label.
|
||||
/**
|
||||
* Get the label to use as replacement for another label.
|
||||
*
|
||||
* @param replacements The label replacement map
|
||||
* @param label The label to find a replacement for
|
||||
* @param label The label to find a replacement for
|
||||
* @return The alias to use. Null if no replacement exists.
|
||||
*/
|
||||
private static Label getReplacement(Map<Label, Label> replacements,Label label) {
|
||||
private static Label getReplacement(Map<Label, Label> replacements, Label label) {
|
||||
Label replacement = replacements.get(label);
|
||||
while (replacements.get(replacement)!=null) {
|
||||
while (replacements.get(replacement) != null) {
|
||||
replacement = replacements.get(replacement);
|
||||
}
|
||||
return replacement;
|
||||
@ -166,6 +190,7 @@ public abstract class Pass2Optimization {
|
||||
|
||||
/**
|
||||
* Remove all assignments to specific LValues from the control flow graph (as they are no longer needed).
|
||||
*
|
||||
* @param variables The variables to eliminate
|
||||
*/
|
||||
public void removeAssignments(Collection<? extends LValue> variables) {
|
||||
@ -177,7 +202,7 @@ public abstract class Pass2Optimization {
|
||||
if (variables.contains(assignment.getLValue())) {
|
||||
iterator.remove();
|
||||
}
|
||||
} else if(statement instanceof StatementPhi) {
|
||||
} else if (statement instanceof StatementPhi) {
|
||||
StatementPhi phi = (StatementPhi) statement;
|
||||
if (variables.contains(phi.getLValue())) {
|
||||
iterator.remove();
|
||||
@ -189,6 +214,7 @@ public abstract class Pass2Optimization {
|
||||
|
||||
/**
|
||||
* Remove variables from the symbol table
|
||||
*
|
||||
* @param variables The variables to remove
|
||||
*/
|
||||
public void deleteSymbols(Collection<? extends LValue> variables) {
|
33
src/dk/camelot64/kickc/icl/Pass5AsmOptimization.java
Normal file
33
src/dk/camelot64/kickc/icl/Pass5AsmOptimization.java
Normal file
@ -0,0 +1,33 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
import dk.camelot64.kickc.asm.AsmLine;
|
||||
import dk.camelot64.kickc.asm.AsmProgram;
|
||||
|
||||
import java.util.Iterator;
|
||||
import java.util.List;
|
||||
|
||||
/** Optimization performed on Assembler Code (Asm Code).
|
||||
* Optimizations are performed repeatedly until none of them yield any result
|
||||
**/
|
||||
public class Pass5AsmOptimization {
|
||||
|
||||
private AsmProgram program;
|
||||
|
||||
public Pass5AsmOptimization(AsmProgram program) {
|
||||
this.program = program;
|
||||
}
|
||||
|
||||
public AsmProgram getProgram() {
|
||||
return program;
|
||||
}
|
||||
|
||||
public void remove(List<AsmLine> remove) {
|
||||
for (Iterator<AsmLine> iterator = program.getLines().iterator(); iterator.hasNext(); ) {
|
||||
AsmLine line = iterator.next();
|
||||
if (remove.contains(line)) {
|
||||
System.out.println("Removing instruction "+line.getAsm());
|
||||
iterator.remove();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -10,23 +10,20 @@ import java.util.Iterator;
|
||||
import java.util.List;
|
||||
|
||||
/** Optimize assembler code by removing jumps to labels immediately following the jump */
|
||||
public class Pass5NextJumpElimination {
|
||||
|
||||
private AsmProgram program;
|
||||
public class Pass5NextJumpElimination extends Pass5AsmOptimization {
|
||||
|
||||
public Pass5NextJumpElimination(AsmProgram program) {
|
||||
this.program = program;
|
||||
super(program);
|
||||
}
|
||||
|
||||
public boolean optimize() {
|
||||
List<AsmLine> remove = new ArrayList<>();
|
||||
List<AsmLine> removeLines = new ArrayList<>();
|
||||
AsmInstruction candidate = null;
|
||||
for (AsmLine line : program.getLines()) {
|
||||
for (AsmLine line : getProgram().getLines()) {
|
||||
if(candidate!=null) {
|
||||
if(line instanceof AsmLabel) {
|
||||
if(((AsmLabel) line).getLabel().equals(candidate.getParameter())) {
|
||||
remove.add(candidate);
|
||||
candidate = null;
|
||||
removeLines.add(candidate);
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -38,13 +35,7 @@ public class Pass5NextJumpElimination {
|
||||
}
|
||||
}
|
||||
}
|
||||
for (Iterator<AsmLine> iterator = program.getLines().iterator(); iterator.hasNext(); ) {
|
||||
AsmLine line = iterator.next();
|
||||
if (remove.contains(line)) {
|
||||
System.out.println("Removing jump to next instruction "+line.getAsm());
|
||||
iterator.remove();
|
||||
}
|
||||
}
|
||||
return remove.size()>0;
|
||||
remove(removeLines);
|
||||
return removeLines.size()>0;
|
||||
}
|
||||
}
|
||||
|
@ -18,7 +18,8 @@ public class PassTypeInference {
|
||||
if (operator == null || assignment.getRValue1() == null) {
|
||||
// Copy operation or Unary operation
|
||||
RValue rValue = assignment.getRValue2();
|
||||
SymbolType type = inferType(rValue);
|
||||
SymbolType subType = inferType(rValue);
|
||||
SymbolType type = inferType(operator, subType);
|
||||
symbol.setInferredType(type);
|
||||
} else {
|
||||
// Binary operation
|
||||
@ -33,6 +34,23 @@ public class PassTypeInference {
|
||||
}
|
||||
}
|
||||
|
||||
private SymbolType inferType(Operator operator, SymbolType subType) {
|
||||
if(operator==null) {
|
||||
return subType;
|
||||
}
|
||||
String op = operator.getOperator();
|
||||
switch (op) {
|
||||
case "*":
|
||||
if(subType instanceof SymbolTypePointer) {
|
||||
return ((SymbolTypePointer) subType).getElementType();
|
||||
} else {
|
||||
throw new RuntimeException("Type error: Dereferencing a non-pointer "+subType);
|
||||
}
|
||||
default:
|
||||
return subType;
|
||||
}
|
||||
}
|
||||
|
||||
private SymbolType inferType(SymbolType type1, Operator operator, SymbolType type2) {
|
||||
String op = operator.getOperator();
|
||||
switch (op) {
|
||||
@ -55,7 +73,15 @@ public class PassTypeInference {
|
||||
if (type1 instanceof SymbolTypePointer && (type2.equals(SymbolTypeBasic.BYTE) || type2.equals(SymbolTypeBasic.WORD))) {
|
||||
return new SymbolTypePointer(((SymbolTypePointer) type1).getElementType());
|
||||
}
|
||||
if (type1 instanceof SymbolTypePointer && type2 instanceof SymbolTypePointer) {
|
||||
SymbolType elmType1 = ((SymbolTypePointer) type1).getElementType();
|
||||
SymbolType elmType2 = ((SymbolTypePointer) type2).getElementType();
|
||||
return inferType(elmType1, operator, elmType2);
|
||||
}
|
||||
case "*":
|
||||
if(type1==null && type2 instanceof SymbolTypePointer) {
|
||||
return ((SymbolTypePointer) type2).getElementType();
|
||||
}
|
||||
case "/":
|
||||
if (SymbolTypeBasic.WORD.equals(type1) || SymbolTypeBasic.WORD.equals(type2)) {
|
||||
return SymbolTypeBasic.WORD;
|
||||
|
@ -1,20 +1,8 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
/** A dereferenced pointer */
|
||||
public class PointerDereference implements LValue {
|
||||
public interface PointerDereference extends LValue {
|
||||
|
||||
private Variable pointer;
|
||||
Value getPointer();
|
||||
|
||||
public PointerDereference(Variable pointer) {
|
||||
this.pointer = pointer;
|
||||
}
|
||||
|
||||
public Variable getPointer() {
|
||||
return pointer;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "*(" + pointer + ')';
|
||||
}
|
||||
}
|
||||
|
24
src/dk/camelot64/kickc/icl/PointerDereferenceConstant.java
Normal file
24
src/dk/camelot64/kickc/icl/PointerDereferenceConstant.java
Normal file
@ -0,0 +1,24 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
/** A dereferenced constant pointer */
|
||||
public class PointerDereferenceConstant implements LValue {
|
||||
|
||||
private Constant pointer;
|
||||
|
||||
public PointerDereferenceConstant(Constant pointer) {
|
||||
this.pointer = pointer;
|
||||
}
|
||||
|
||||
public Constant getPointer() {
|
||||
return pointer;
|
||||
}
|
||||
|
||||
public Constant getPointerConstant() {
|
||||
return pointer;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "*(" + pointer + ')';
|
||||
}
|
||||
}
|
28
src/dk/camelot64/kickc/icl/PointerDereferenceVariable.java
Normal file
28
src/dk/camelot64/kickc/icl/PointerDereferenceVariable.java
Normal file
@ -0,0 +1,28 @@
|
||||
package dk.camelot64.kickc.icl;
|
||||
|
||||
/** A dereferenced variable pointer */
|
||||
public class PointerDereferenceVariable implements PointerDereference {
|
||||
|
||||
private Variable pointer;
|
||||
|
||||
public PointerDereferenceVariable(Variable pointer) {
|
||||
this.pointer = pointer;
|
||||
}
|
||||
|
||||
public Variable getPointer() {
|
||||
return pointer;
|
||||
}
|
||||
|
||||
public Variable getPointerVariable() {
|
||||
return pointer;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "*(" + pointer + ')';
|
||||
}
|
||||
|
||||
public void setPointerVariable(Variable pointer) {
|
||||
this.pointer = pointer;
|
||||
}
|
||||
}
|
@ -13,7 +13,7 @@ import java.util.List;
|
||||
/** Test my KickC Grammar */
|
||||
public class Main {
|
||||
public static void main(String[] args) throws IOException {
|
||||
final String fileName = "src/dk/camelot64/kickc/test/mem.kc";
|
||||
final String fileName = "src/dk/camelot64/kickc/test/fibmem.kc";
|
||||
final CharStream input = CharStreams.fromFileName(fileName);
|
||||
System.out.println(input.toString());
|
||||
KickCLexer lexer = new KickCLexer(input);
|
||||
@ -42,7 +42,7 @@ public class Main {
|
||||
new Pass1GenerateSingleStaticAssignmentForm(symbolTable, controlFlowGraph);
|
||||
pass1GenerateSingleStaticAssignmentForm.generate();
|
||||
|
||||
List<Pass2Optimization> optimizations = new ArrayList<>();
|
||||
List<Pass2SsaOptimization> optimizations = new ArrayList<>();
|
||||
optimizations.add(new Pass2CullEmptyBlocks(controlFlowGraph, symbolTable));
|
||||
optimizations.add(new Pass2ConstantPropagation(controlFlowGraph, symbolTable));
|
||||
optimizations.add(new Pass2AliasElimination(controlFlowGraph, symbolTable));
|
||||
@ -53,14 +53,14 @@ public class Main {
|
||||
System.out.println("INITIAL CONTROL FLOW GRAPH");
|
||||
System.out.println(controlFlowGraph.toString());
|
||||
|
||||
boolean optimized = true;
|
||||
while (optimized) {
|
||||
optimized = false;
|
||||
for (Pass2Optimization optimization : optimizations) {
|
||||
boolean ssaOptimized = true;
|
||||
while (ssaOptimized) {
|
||||
ssaOptimized = false;
|
||||
for (Pass2SsaOptimization optimization : optimizations) {
|
||||
boolean stepOptimized = optimization.optimize();
|
||||
if (stepOptimized) {
|
||||
System.out.println("Succesful optimization "+optimization);
|
||||
optimized = true;
|
||||
ssaOptimized = true;
|
||||
System.out.println("CONTROL FLOW GRAPH");
|
||||
System.out.println(controlFlowGraph.toString());
|
||||
}
|
||||
@ -73,7 +73,10 @@ public class Main {
|
||||
Pass4CodeGeneration pass4CodeGeneration = new Pass4CodeGeneration(controlFlowGraph, symbolTable);
|
||||
AsmProgram asmProgram = pass4CodeGeneration.generate();
|
||||
Pass5NextJumpElimination pass5NextJumpElimination = new Pass5NextJumpElimination(asmProgram);
|
||||
pass5NextJumpElimination.optimize();
|
||||
boolean asmOptimized = true;
|
||||
while(asmOptimized) {
|
||||
asmOptimized = pass5NextJumpElimination.optimize();
|
||||
}
|
||||
|
||||
|
||||
System.out.println("SYMBOLS");
|
||||
|
8
src/dk/camelot64/kickc/test/fibmem.kc
Normal file
8
src/dk/camelot64/kickc/test/fibmem.kc
Normal file
@ -0,0 +1,8 @@
|
||||
byte[15] fibs = $1100;
|
||||
fibs[0] = 0;
|
||||
fibs[1] = 1;
|
||||
byte i = 2;
|
||||
while(i<15) {
|
||||
fibs[i] = fibs[i-2]+fibs[i-1];
|
||||
i = i + 1;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user