//===-EDInst.cpp - LLVM Enhanced Disassembler -----------------------------===//
//
//                     The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
// 
//===----------------------------------------------------------------------===//
//
// This file implements the Enhanced Disassembly library's instruction class.
// The instruction is responsible for vending the string representation, 
// individual tokens, and operands for a single instruction.
//
//===----------------------------------------------------------------------===//

#include "EDInst.h"
#include "EDDisassembler.h"
#include "EDOperand.h"
#include "EDToken.h"

#include "llvm/MC/EDInstInfo.h"
#include "llvm/MC/MCInst.h"

using namespace llvm;

EDInst::EDInst(llvm::MCInst *inst,
               uint64_t byteSize, 
               EDDisassembler &disassembler,
               const llvm::EDInstInfo *info) :
  Disassembler(disassembler),
  Inst(inst),
  ThisInstInfo(info),
  ByteSize(byteSize),
  BranchTarget(-1),
  MoveSource(-1),
  MoveTarget(-1) {
  OperandOrder = ThisInstInfo->operandOrders[Disassembler.llvmSyntaxVariant()];
}

EDInst::~EDInst() {
  unsigned int index;
  unsigned int numOperands = Operands.size();
  
  for (index = 0; index < numOperands; ++index)
    delete Operands[index];
  
  unsigned int numTokens = Tokens.size();
  
  for (index = 0; index < numTokens; ++index)
    delete Tokens[index];
  
  delete Inst;
}

uint64_t EDInst::byteSize() {
  return ByteSize;
}

int EDInst::stringify() {
  if (StringifyResult.valid())
    return StringifyResult.result();
  
  if (Disassembler.printInst(String, *Inst))
    return StringifyResult.setResult(-1);

  String.push_back('\n');
  
  return StringifyResult.setResult(0);
}

int EDInst::getString(const char*& str) {
  if (stringify())
    return -1;
  
  str = String.c_str();
  
  return 0;
}

unsigned EDInst::instID() {
  return Inst->getOpcode();
}

bool EDInst::isBranch() {
  if (ThisInstInfo)
    return 
      ThisInstInfo->instructionType == kInstructionTypeBranch ||
      ThisInstInfo->instructionType == kInstructionTypeCall;
  else
    return false;
}

bool EDInst::isMove() {
  if (ThisInstInfo)
    return ThisInstInfo->instructionType == kInstructionTypeMove;
  else
    return false;
}

int EDInst::parseOperands() {
  if (ParseResult.valid())
    return ParseResult.result();
  
  if (!ThisInstInfo)
    return ParseResult.setResult(-1);
  
  unsigned int opIndex;
  unsigned int mcOpIndex = 0;
  
  for (opIndex = 0; opIndex < ThisInstInfo->numOperands; ++opIndex) {
    if (isBranch() &&
        (ThisInstInfo->operandFlags[opIndex] & kOperandFlagTarget)) {
      BranchTarget = opIndex;
    }
    else if (isMove()) {
      if (ThisInstInfo->operandFlags[opIndex] & kOperandFlagSource)
        MoveSource = opIndex;
      else if (ThisInstInfo->operandFlags[opIndex] & kOperandFlagTarget)
        MoveTarget = opIndex;
    }
    
    EDOperand *operand = new EDOperand(Disassembler, *this, opIndex, mcOpIndex);
    
    Operands.push_back(operand);
  }
  
  return ParseResult.setResult(0);
}

int EDInst::branchTargetID() {
  if (parseOperands())
    return -1;
  return BranchTarget;
}

int EDInst::moveSourceID() {
  if (parseOperands())
    return -1;
  return MoveSource;
}

int EDInst::moveTargetID() {
  if (parseOperands())
    return -1;
  return MoveTarget;
}

int EDInst::numOperands() {
  if (parseOperands())
    return -1;
  return Operands.size();
}

int EDInst::getOperand(EDOperand *&operand, unsigned int index) {
  if (parseOperands())
    return -1;
  
  if (index >= Operands.size())
    return -1;
  
  operand = Operands[index];
  return 0;
}

int EDInst::tokenize() {
  if (TokenizeResult.valid())
    return TokenizeResult.result();
    
  if (ThisInstInfo == NULL)
    return TokenizeResult.setResult(-1);
  
  if (stringify())
    return TokenizeResult.setResult(-1);
    
  return TokenizeResult.setResult(EDToken::tokenize(Tokens,
                                                    String,
                                                    OperandOrder,
                                                    Disassembler));
    
}

int EDInst::numTokens() {
  if (tokenize())
    return -1;
  return Tokens.size();
}

int EDInst::getToken(EDToken *&token, unsigned int index) {
  if (tokenize())
    return -1;
  token = Tokens[index];
  return 0;
}

#ifdef __BLOCKS__
int EDInst::visitTokens(EDTokenVisitor_t visitor) {
  if (tokenize())
    return -1;
  
  tokvec_t::iterator iter;
  
  for (iter = Tokens.begin(); iter != Tokens.end(); ++iter) {
    int ret = visitor(*iter);
    if (ret == 1)
      return 0;
    if (ret != 0)
      return -1;
  }
  
  return 0;
}
#endif