LineParser.java 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. package eu.tankernn.assembly.compiler;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. public class LineParser {
  5. public static class OPCodes {
  6. public static final byte NOP = 0x0, LDA = 0x1, ADD = 0x2, SUB = 0x3, AND = 0x4, OR = 0x5, NOT_B = 0x6,
  7. STO = 0x7, OUT = 0x8, JMP = 0x9, HLT = 0xA;
  8. }
  9. String line;
  10. int lineIndex;
  11. ArrayList<Byte> list = new ArrayList<Byte>();
  12. String[] instruction;
  13. int pos = 0;
  14. public LineParser(String line, int lineIndex) {
  15. this.line = line;
  16. this.lineIndex = lineIndex;
  17. instruction = line.split("\\s+");
  18. }
  19. public Byte[] parse() throws SyntaxErrorException {
  20. // Comment
  21. if (line.startsWith(";")) {
  22. return null;
  23. }
  24. // Add a label
  25. if (instruction[0].endsWith(":")) {
  26. Assemble.addLabel(instruction[0].substring(0, instruction[0].length() - 2));
  27. pos++;
  28. }
  29. // Add a literal byte
  30. if (instruction[pos].startsWith("$"))
  31. try {
  32. byte b = parseToByte(instruction[pos]);
  33. list.add(b);
  34. return listToByteArray(list);
  35. } catch (NumberFormatException ex) {
  36. }
  37. String command = instruction[pos].toUpperCase();
  38. switch (command) {
  39. case "NOP":
  40. addWithoutAddress(OPCodes.NOP);
  41. break;
  42. case "LDA":
  43. addWithAddress(OPCodes.LDA);
  44. break;
  45. case "ADD":
  46. addWithAddress(OPCodes.ADD);
  47. break;
  48. case "SUB":
  49. addWithAddress(OPCodes.SUB);
  50. break;
  51. case "AND":
  52. addWithAddress(OPCodes.AND);
  53. break;
  54. case "OR":
  55. addWithAddress(OPCodes.OR);
  56. break;
  57. case "NOT":
  58. case "NOTB":
  59. addWithAddress(OPCodes.NOT_B);
  60. break;
  61. case "STO":
  62. addWithoutAddress(OPCodes.STO);
  63. break;
  64. case "OUT":
  65. addWithoutAddress(OPCodes.OUT);
  66. break;
  67. case "JMP":
  68. pos++;
  69. switch (instruction[pos]) {
  70. case "":
  71. }
  72. addWithoutAddress(OPCodes.JMP);
  73. break;
  74. case "HLT":
  75. addWithoutAddress(OPCodes.HLT);
  76. break;
  77. default:
  78. throw new SyntaxErrorException("Commmand not found: " + command + " At line: " + lineIndex);
  79. }
  80. return listToByteArray(list);
  81. }
  82. private static Byte[] listToByteArray(List<Byte> list) {
  83. return list.toArray(new Byte[list.size()]);
  84. }
  85. /**
  86. * Converts a string representation of a byte value into a byte.
  87. *
  88. * @param s
  89. * A string with the format <code>$base:value</code>
  90. * @return The byte value represented by the string
  91. */
  92. private Byte parseToByte(String s) {
  93. String[] sArr = instruction[pos].replace("$", "").split(":");
  94. byte b = (byte) Integer.parseInt(sArr[1], Integer.parseInt(sArr[0]));
  95. return Assemble.bigEndianData ? reverseByte(b) : b;
  96. }
  97. private byte reverseByte(byte x) {
  98. int intSize = 8;
  99. byte y = 0;
  100. for (int position = intSize - 1; position > 0; position--) {
  101. y += ((x & 1) << position);
  102. x >>= 1;
  103. }
  104. return y;
  105. }
  106. private void addWithAddress(byte OPCode) throws SyntaxErrorException {
  107. pos++;
  108. byte address;
  109. try {
  110. address = parseToByte(instruction[pos]);
  111. } catch (IndexOutOfBoundsException ex) {
  112. throw new SyntaxErrorException("Missing address." + " At line: " + lineIndex);
  113. }
  114. list.add((byte) ((OPCode << 0x4) | address));
  115. }
  116. private void addWithoutAddress(byte OPCode) {
  117. list.add((byte) ((OPCode << 0x4)));
  118. }
  119. }