diff --git a/src/dk/camelot64/kickc/asm/AsmFragment.java b/src/dk/camelot64/kickc/asm/AsmFragment.java index 3df80db4f..88e359eb4 100644 --- a/src/dk/camelot64/kickc/asm/AsmFragment.java +++ b/src/dk/camelot64/kickc/asm/AsmFragment.java @@ -32,7 +32,7 @@ public class AsmFragment { */ private String signature; - public AsmFragment(StatementConditionalJump conditionalJump, SymbolTable symbols, ControlFlowGraph graph, ControlFlowBlock block) { + public AsmFragment(StatementConditionalJump conditionalJump, ControlFlowBlock block, SymbolTable symbols, ControlFlowGraph graph) { this.bindings = new HashMap<>(); this.symbols = symbols; StringBuilder signature = new StringBuilder(); diff --git a/src/dk/camelot64/kickc/asm/AsmInstruction.java b/src/dk/camelot64/kickc/asm/AsmInstruction.java index 130c68907..e7f11e90b 100644 --- a/src/dk/camelot64/kickc/asm/AsmInstruction.java +++ b/src/dk/camelot64/kickc/asm/AsmInstruction.java @@ -15,6 +15,14 @@ public class AsmInstruction implements AsmLine { this.invocationCountEstimate = invocationCountEstimate; } + public String getParameter() { + return parameter; + } + + public AsmInstructionType getType() { + return type; + } + @Override public int getLineBytes() { return type.getBytes(); diff --git a/src/dk/camelot64/kickc/asm/AsmInstructionType.java b/src/dk/camelot64/kickc/asm/AsmInstructionType.java index 71ebd0d46..cf2e93461 100644 --- a/src/dk/camelot64/kickc/asm/AsmInstructionType.java +++ b/src/dk/camelot64/kickc/asm/AsmInstructionType.java @@ -11,6 +11,8 @@ public class AsmInstructionType { private double cycles; + private boolean jump; + public AsmInstructionType(int opcode, String mnemnonic, AsmAddressingMode addressingMode, double cycles) { this.opcode = opcode; this.mnemnonic = mnemnonic; @@ -38,4 +40,16 @@ public class AsmInstructionType { return addressingMode.getAsm(mnemnonic, parameter); } + /** + * Tells if the instruction is a jump or a branch (and the parameter is therefore a label or destination address) + * @return true if the instruction is a jump/branch + */ + public boolean isJump() { + return jump; + } + + void setJump(boolean jump) { + this.jump = jump; + } + } diff --git a/src/dk/camelot64/kickc/asm/AsmInstuctionSet.java b/src/dk/camelot64/kickc/asm/AsmInstuctionSet.java index 889078ad8..b1ffd34e6 100644 --- a/src/dk/camelot64/kickc/asm/AsmInstuctionSet.java +++ b/src/dk/camelot64/kickc/asm/AsmInstuctionSet.java @@ -322,7 +322,20 @@ public class AsmInstuctionSet { add(0xfd, "SBC", abx, 4.5); add(0xfe, "INC", abx, 7.0); add(0xff, "ISC", abx, 7.0); - + for (AsmInstructionType instruction : instructions) { + switch(instruction.getMnemnonic()) { + case "jmp": + case "beq": + case "bne": + case "bcc": + case "bcs": + case "bvc": + case "bvs": + case "bmi": + case "bpl": + instruction.setJump(true); + } + } } public AsmInstructionType getType(String mnemonic, AsmAddressingMode addressingMode) { diff --git a/src/dk/camelot64/kickc/asm/AsmLabel.java b/src/dk/camelot64/kickc/asm/AsmLabel.java index 6138e9182..f58911911 100644 --- a/src/dk/camelot64/kickc/asm/AsmLabel.java +++ b/src/dk/camelot64/kickc/asm/AsmLabel.java @@ -9,6 +9,10 @@ public class AsmLabel implements AsmLine { this.label = label; } + public String getLabel() { + return label; + } + @Override public int getLineBytes() { return 0; diff --git a/src/dk/camelot64/kickc/icl/Pass4CodeGeneration.java b/src/dk/camelot64/kickc/icl/Pass4CodeGeneration.java index 08499a86e..1e99ed0f1 100644 --- a/src/dk/camelot64/kickc/icl/Pass4CodeGeneration.java +++ b/src/dk/camelot64/kickc/icl/Pass4CodeGeneration.java @@ -50,7 +50,7 @@ public class Pass4CodeGeneration { asm.addComment(statement + " // " + asmFragment.getSignature()); asmFragment.generate(asm); } else if (statement instanceof StatementConditionalJump) { - AsmFragment asmFragment = new AsmFragment((StatementConditionalJump) statement, symbols, graph, block); + AsmFragment asmFragment = new AsmFragment((StatementConditionalJump) statement, block, symbols, graph); asm.addComment(statement + " // " + asmFragment.getSignature()); asmFragment.generate(asm); } else { diff --git a/src/dk/camelot64/kickc/icl/Pass5NextJumpElimination.java b/src/dk/camelot64/kickc/icl/Pass5NextJumpElimination.java new file mode 100644 index 000000000..ec5bf11a0 --- /dev/null +++ b/src/dk/camelot64/kickc/icl/Pass5NextJumpElimination.java @@ -0,0 +1,50 @@ +package dk.camelot64.kickc.icl; + +import dk.camelot64.kickc.asm.AsmInstruction; +import dk.camelot64.kickc.asm.AsmLabel; +import dk.camelot64.kickc.asm.AsmLine; +import dk.camelot64.kickc.asm.AsmProgram; + +import java.util.ArrayList; +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 Pass5NextJumpElimination(AsmProgram program) { + this.program = program; + } + + public boolean optimize() { + List remove = new ArrayList<>(); + AsmInstruction candidate = null; + for (AsmLine line : program.getLines()) { + if(candidate!=null) { + if(line instanceof AsmLabel) { + if(((AsmLabel) line).getLabel().equals(candidate.getParameter())) { + remove.add(candidate); + candidate = null; + } + } + } + if(line instanceof AsmInstruction) { + candidate = null; + AsmInstruction instruction = (AsmInstruction) line; + if(instruction.getType().isJump()) { + candidate = instruction; + } + } + } + for (Iterator 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; + } +} diff --git a/src/dk/camelot64/kickc/test/Main.java b/src/dk/camelot64/kickc/test/Main.java index 68c01db5b..624cbd121 100644 --- a/src/dk/camelot64/kickc/test/Main.java +++ b/src/dk/camelot64/kickc/test/Main.java @@ -68,8 +68,10 @@ public class Main { pass3RegisterAllocation.allocate(); Pass4CodeGeneration pass4CodeGeneration = new Pass4CodeGeneration(controlFlowGraph, symbolTable); - AsmProgram asmProgram = pass4CodeGeneration.generate(); + Pass5NextJumpElimination pass5NextJumpElimination = new Pass5NextJumpElimination(asmProgram); + pass5NextJumpElimination.optimize(); + System.out.println("SYMBOLS"); System.out.println(symbolTable.toString());