/* Copyright 2011-2016 Google Inc. All Rights Reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package com.google.security.zynamics.reil.interpreter; import com.google.common.base.Preconditions; import com.google.security.zynamics.reil.OperandSize; import com.google.security.zynamics.reil.OperandType; import com.google.security.zynamics.reil.ReilHelpers; import com.google.security.zynamics.reil.ReilInstruction; import com.google.security.zynamics.reil.ReilOperand; import com.google.security.zynamics.zylib.general.Pair; import java.math.BigInteger; import java.util.ArrayList; import java.util.HashMap; import java.util.List; /** * Interpreter for REIL code */ public class ReilInterpreter { private static final String SUB_PC = "sub_PC"; /** * CPU policy used by the interpreter */ private final ICpuPolicy cpuPolicy; /** * Interpreter policy used by the interpreter */ private final IInterpreterPolicy interpreterPolicy; /** * Registers currently in use by the interpreter */ private final HashMap<String, ReilRegister> registers = new HashMap<String, ReilRegister>(); /** * Memory currently in use by the interpreter */ private final ReilMemory memory; /** * Creates a new REIL interpreter with the given options. * * @param endianness Endianness of the memory layout of the interpreter * @param cpuPolicy CPU policy used by the interpreter * @param interpreterPolicy Interpreter policy used by the interpreter */ public ReilInterpreter(final Endianness endianness, final ICpuPolicy cpuPolicy, final IInterpreterPolicy interpreterPolicy) { Preconditions.checkNotNull(endianness, "Error: Argument endianness can't be null"); this.cpuPolicy = Preconditions.checkNotNull(cpuPolicy, "Error: Argument cpuPolicy can't be null"); this.interpreterPolicy = Preconditions.checkNotNull(interpreterPolicy, "Error: Argument interpreterPolicy can't be null"); this.memory = new ReilMemory(endianness); } private static BigInteger getTruncateMask(final OperandSize targetSize) { switch (targetSize) { case BYTE: return BigInteger.valueOf(0xFFL); case DWORD: return BigInteger.valueOf(0xFFFFFFFFL); case QWORD: return BigInteger.valueOf(0xFFFFFFFFFFFFFFFFL); case OWORD: return BigInteger.valueOf(0xFFFFFFFFFFFFFFL).shiftLeft(56) .add(BigInteger.valueOf(0xFFFFFFFFFFFFFFL)).shiftLeft(16) .add(BigInteger.valueOf(0xFFFFL)); case WORD: return BigInteger.valueOf(0xFFFFL); default: throw new IllegalStateException("Error: Unknown target size for truncate mask"); } } private static void log(final String msg, final Object... args) { System.out.printf(msg, args); } public static BigInteger nativeToReil(final BigInteger address) { return address.multiply(BigInteger.valueOf(0x100)); } /** * Interprets an ADD instruction. * * @param instruction The ADD instruction to interpret */ private void interpretAdd(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final OperandSize targetSize = instruction.getThirdOperand().getSize(); final BigInteger result = firstValue.second().add(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets an AND instruction. * * @param instruction The AND instruction to interpret */ private void interpretAnd(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().and(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a BISZ instruction. * * @param instruction The BISZ instruction to interpret */ private void interpretBisz(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); if (firstValue.first()) { final BigInteger result = firstValue.second().equals(BigInteger.ZERO) ? BigInteger.ONE : BigInteger.ZERO; final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a BSH instruction. * * @param instruction The BSH instruction to interpret */ private void interpretBsh(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final boolean isMsbSet = !(secondValue.second().and( BigInteger.valueOf((long) Math.pow(2, instruction.getSecondOperand().getSize() .getBitSize() - 1))).equals(BigInteger.ZERO)); BigInteger result = BigInteger.ZERO; if (secondValue.second().compareTo(BigInteger.ZERO) < 0) { System.out.println("F" + firstValue.second()); System.out.println("S" + secondValue.second().negate()); result = firstValue.second().shiftRight(secondValue.second().negate().intValue()); } else if (isMsbSet) { result = firstValue.second().shiftRight( BigInteger.ZERO.subtract(secondValue.second()).and(BigInteger.valueOf(0xFF)) .intValue()); } else { result = firstValue.second().shiftLeft(secondValue.second().intValue()); } final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a DIV instruction. * * @param instruction The DIV instruction to interpret */ private void interpretDiv(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().divide(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a single REIL instruction. * * @param instruction The REIL instruction to interpret * @param programCounter The name of the program counter register */ private void interpretInstruction(final ReilInstruction instruction, final String programCounter) { final Integer mnemonic = instruction.getMnemonicCode(); switch (mnemonic) { case ReilHelpers._OPCODE_ADD: interpretAdd(instruction); break; case ReilHelpers._OPCODE_AND: interpretAnd(instruction); break; case ReilHelpers._OPCODE_BISZ: interpretBisz(instruction); break; case ReilHelpers._OPCODE_BSH: interpretBsh(instruction); break; case ReilHelpers._OPCODE_DIV: interpretDiv(instruction); break; case ReilHelpers._OPCODE_JCC: interpretJcc(instruction, programCounter); break; case ReilHelpers._OPCODE_LDM: interpretLdm(instruction); break; case ReilHelpers._OPCODE_MOD: interpretMod(instruction); break; case ReilHelpers._OPCODE_MUL: interpretMul(instruction); break; case ReilHelpers._OPCODE_NOP: break; case ReilHelpers._OPCODE_OR: interpretOr(instruction); break; case ReilHelpers._OPCODE_STM: interpretStm(instruction); break; case ReilHelpers._OPCODE_STR: interpretStr(instruction); break; case ReilHelpers._OPCODE_SUB: interpretSub(instruction); break; case ReilHelpers._OPCODE_UNDEF: interpretUndef(instruction); break; case ReilHelpers._OPCODE_UNKNOWN: // TODO: not implemented break; case ReilHelpers._OPCODE_XOR: interpretXor(instruction); break; default: throw new IllegalArgumentException("invalid Mnemonic in REIL interpreting loop"); } } /** * Interprets a JCC instruction. * * @param instruction The JCC instruction to interpret * @param programCounter The name of the program counter register */ private void interpretJcc(final ReilInstruction instruction, final String programCounter) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); if (!firstValue.second().equals(BigInteger.ZERO) && (instruction.getThirdOperand().getType() == OperandType.SUB_ADDRESS)) { final String[] parts = instruction.getThirdOperand().getValue().split("\\."); assert parts.length == 2; setRegister(programCounter, new BigInteger(parts[0]), OperandSize.DWORD, ReilRegisterStatus.DEFINED); setRegister(SUB_PC, new BigInteger(parts[1]), OperandSize.DWORD, ReilRegisterStatus.DEFINED); } else if (!firstValue.second().equals(BigInteger.ZERO)) { final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getThirdOperand()); setRegister(programCounter, secondValue.second(), OperandSize.DWORD, ReilRegisterStatus.DEFINED); } } /** * Interprets a LDM instruction. * * @param instruction The LDM instruction to interpret */ private void interpretLdm(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); if (firstValue.first()) { final String target = instruction.getThirdOperand().getValue(); final int targetSize = instruction.getThirdOperand().getSize().getByteSize(); final BigInteger value = new BigInteger(String.valueOf(memory.load(firstValue.second().longValue(), targetSize))); setRegister(target, value, instruction.getThirdOperand().getSize(), ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a MOD instruction. * * @param instruction The MOD instruction to interpret */ private void interpretMod(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().mod(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a MUL instruction. * * @param instruction The MUL instruction to interpret */ private void interpretMul(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().multiply(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a OR instruction. * * @param instruction The OR instruction to interpret */ private void interpretOr(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().or(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a STM instruction. * * @param instruction The STM instruction to interpret */ private void interpretStm(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> sourceValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> targetValue = loadLongValue(instruction.getThirdOperand()); if (sourceValue.first() && targetValue.first()) { final int targetSize = instruction.getFirstOperand().getSize().getByteSize(); memory.store(targetValue.second().longValue(), sourceValue.second().longValue(), targetSize); } else { log("%s", sourceValue.toString()); log("%s", targetValue.toString()); assert false; } } /** * Interprets a STR instruction. * * @param instruction The STR instruction to interpret */ private void interpretStr(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); if (firstValue.first()) { final BigInteger result = firstValue.second(); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets a SUB instruction. * * @param instruction The SUB instruction to interpret */ private void interpretSub(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final OperandSize targetSize = instruction.getThirdOperand().getSize(); final BigInteger result = firstValue.second().subtract(secondValue.second()) .and(getTruncateMask(targetSize)); final String targetRegister = instruction.getThirdOperand().getValue(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Interprets an UNDEF instruction. * * @param instruction The UNDEF instruction to interpret */ private void interpretUndef(final ReilInstruction instruction) { registers.remove(instruction.getThirdOperand().getValue()); } /** * Interprets a XOR instruction. * * @param instruction The XORinstruction to interpret */ private void interpretXor(final ReilInstruction instruction) { final Pair<Boolean, BigInteger> firstValue = loadLongValue(instruction.getFirstOperand()); final Pair<Boolean, BigInteger> secondValue = loadLongValue(instruction.getSecondOperand()); if (firstValue.first() && secondValue.first()) { final BigInteger result = firstValue.second().xor(secondValue.second()); final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, result, targetSize, ReilRegisterStatus.DEFINED); } else if (instruction.getFirstOperand().getValue() .equals(instruction.getSecondOperand().getValue())) { final String targetRegister = instruction.getThirdOperand().getValue(); final OperandSize targetSize = instruction.getThirdOperand().getSize(); setRegister(targetRegister, BigInteger.ZERO, targetSize, ReilRegisterStatus.DEFINED); } else { assert false; } } /** * Loads the value of an operand into a long value. * * @param operand The operand to load * @return A pair made of a bool and a long value. The bool indicates whether loading the value * was successful. */ private Pair<Boolean, BigInteger> loadLongValue(final ReilOperand operand) { final OperandType type = operand.getType(); String value = operand.getValue(); if (type == OperandType.INTEGER_LITERAL) { return new Pair<Boolean, BigInteger>(true, new BigInteger(value)); } else if (type == OperandType.REGISTER) { // Check if we have a negative prefix before the register name. This is // a bit of a hack, because we never explicitly stated that we would allow // a negation of a register operand in REIL. // TODO(thomasdullien) remove this code once we have introduced explicit // left-shift and right-shift instructions. value = (value.charAt(0) == '-') ? operand.getValue().substring(1) : value; return !isDefined(value) ? new Pair<Boolean, BigInteger>(false, BigInteger.ZERO) : new Pair<Boolean, BigInteger>(true, getVariableValue(value)); } else { return new Pair<Boolean, BigInteger>(false, BigInteger.ZERO); } } /** * Searches the instruction at the next valid PC offset. * * @param pc The current value of the program counter * @param instructions The instructions to interpret * @param programCounter The name of the program counter register * * @return True, if the next valid PC offset could be found. False, otherwise. */ private boolean searchNextPc(final BigInteger pc, final HashMap<BigInteger, List<ReilInstruction>> instructions, final String programCounter) { for (int i = 0; i < 10; i++) { final BigInteger current = pc.add(BigInteger.valueOf(i)); if (instructions.containsKey(nativeToReil(current))) { setRegister(programCounter, current, OperandSize.DWORD, ReilRegisterStatus.DEFINED); return true; } } return false; } /** * Returns the CPU policy of the interpreter. * * @return The CPU policy of the interpreter */ public ICpuPolicy getCpuPolicy() { return cpuPolicy; } public List<ReilRegister> getDefinedRegisters() { return new ArrayList<ReilRegister>(registers.values()); } public ReilMemory getMemory() { return memory; } public long getMemorySize() { return memory.getAllocatedMemory(); } /** * Returns the current value of a register. * * @param register The register in question * * @return The value of the register * * @throws IllegalArgumentException Thrown if the register has no assigned value */ public BigInteger getVariableValue(final String register) { Preconditions.checkNotNull(register, "Error: register argument can not be null"); Preconditions.checkArgument(registers.containsKey(register), "Error: Register has no value"); return registers.get(register).getValue(); } /** * Interprets a list of instructions in the current REIL interpreter * * @param instructions The instructions to interpret * @param entryPoint The offset of the first instruction to interpret * @throws InterpreterException */ public void interpret(final HashMap<BigInteger, List<ReilInstruction>> instructions, final BigInteger entryPoint) throws InterpreterException { cpuPolicy.start(this); interpreterPolicy.start(); final String programCounter = cpuPolicy.getProgramCounter(); Preconditions.checkNotNull(programCounter, "Error: CPU Policy returned an invalid program counter"); // Set the program counter to the entry point setRegister(programCounter, entryPoint, cpuPolicy.getRegisterSize(programCounter), ReilRegisterStatus.DEFINED); while (true) { BigInteger pc = nativeToReil(getVariableValue(programCounter)); interpreterPolicy.nextInstruction(this); log("Interpreting: %X%n", pc.longValue()); if (!instructions.containsKey(pc)) { throw new InterpreterException(String.format("Error: Instruction at offset %X not found", pc)); } final List<ReilInstruction> instructionList = instructions.get(pc); if ((instructionList == null) || (instructionList.size() == 0)) { throw new InterpreterException(String.format( "Error: Instruction at offset %X has invalid REIL code", pc)); } setRegister(SUB_PC, BigInteger.ZERO, OperandSize.DWORD, ReilRegisterStatus.DEFINED); int subPc = getVariableValue(SUB_PC).intValue(); while (subPc < instructionList.size()) { final ReilInstruction inst = instructionList.get(subPc); log("Next instruction: " + inst.toString().replaceAll("%", "")); for (final ReilRegister r : registers.values()) { log("%s: %X%n", r.getRegister(), r.getValue()); } interpretInstruction(inst, programCounter); final int newSubPc = getVariableValue(SUB_PC).intValue(); if (subPc == newSubPc) { subPc = newSubPc + 1; } else { subPc = newSubPc; } setRegister(SUB_PC, BigInteger.valueOf(subPc), OperandSize.DWORD, ReilRegisterStatus.DEFINED); } final BigInteger pcNew = getVariableValue(programCounter); if (pc.equals(nativeToReil(pcNew))) { pc = pcNew.add(BigInteger.ONE); } else { pc = pcNew; } if (pcNew.equals(BigInteger.valueOf(4294967295L))) { break; } if (!searchNextPc(pc, instructions, programCounter)) { break; } } interpreterPolicy.end(); } /** * Determines whether a register currently holds a value or not. * * @param register The register in question * * @return True, if the register has a value. False, otherwise. */ public boolean isDefined(final String register) { System.out.println(registers.keySet()); return registers.containsKey(register); } public byte readMemoryByte(final long address) { return (byte) memory.load(address, 1); } public long readMemoryDword(final long address) { return memory.load(address, 4); } public long readMemoryWord(final long address) { return memory.load(address, 2); } /** * Sets a value in the simulated memory. * * @param address The address of the value * @param value The value to write to the address * @param length The number of bytes to write to the address */ public void setMemory(final long address, final long value, final int length) { memory.store(address, value, length); } /** * Sets the value of a register. * * @param register The register in question * @param value The new value of the register * @param size The size of the register * @param status The status of the register */ public void setRegister(final String register, final BigInteger value, final OperandSize size, final ReilRegisterStatus status) { final BigInteger truncatedValue = value.and(getTruncateMask(size)); final ReilRegister r = new ReilRegister(register, size, truncatedValue); if (registers.containsKey(register)) { registers.remove(register); } if (status == ReilRegisterStatus.DEFINED) { registers.put(register, r); } } }