/*
 * Copyright (C) 2012 The Android Open Source Project
 *
 * 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.
 */

#include "inferred_reg_category_map.h"

#include "stl_util.h"

#include <stdint.h>
#include <vector>

namespace art {
namespace greenland {


InferredRegCategoryMap::InferredRegCategoryMap(uint32_t insns_size,
                                               uint16_t regs_size)
: registers_size_(regs_size) {
}

InferredRegCategoryMap::~InferredRegCategoryMap() {
  STLDeleteValues(&lines_);
}

RegCategory InferredRegCategoryMap::GetRegCategory(uint32_t dex_pc,
                                                   uint16_t reg_idx) const {
  if (lines_.count(dex_pc) == 0) {
    return kRegUnknown;
  }
  return lines_.Get(dex_pc)->GetRegCategory(reg_idx);
}

void InferredRegCategoryMap::SetRegCategory(uint32_t dex_pc,
                                            uint16_t reg_idx,
                                            RegCategory cat) {
  if (cat != kRegUnknown) {
    if (lines_.count(dex_pc) == 0) {
      lines_.Put(dex_pc, new RegCategoryLine(registers_size_));
    }

    lines_.Get(dex_pc)->SetRegCategory(reg_idx, cat);
  }
}

bool InferredRegCategoryMap::
operator==(InferredRegCategoryMap const& rhs) const {

  if (registers_size_ != rhs.registers_size_) {
    return false;
  }

  if (lines_.size() != rhs.lines_.size()) {
    return false;
  }

  for (size_t i = 0; i < lines_.size(); ++i) {
    if (lines_.count(i) == 0 && rhs.lines_.count(i) == 0) {
      continue;
    }

    if ((lines_.count(i) == 0 && rhs.lines_.count(i) != 0) ||
        (lines_.count(i) != 0 && rhs.lines_.count(i) == 0)) {
      return false;
    }

    if (*lines_.Get(i) != *rhs.lines_.Get(i)) {
      return false;
    }
  }

  return true;
}

bool InferredRegCategoryMap::
operator!=(InferredRegCategoryMap const& rhs) const {

  return !(*this == rhs);
}


} // namespace greenland
} // namespace art
