// Copyright 2017 The Crashpad Authors
//
// 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 "snapshot/elf/elf_symbol_table_reader.h"

#include <elf.h>

#include "snapshot/elf/elf_image_reader.h"

namespace crashpad {

namespace {

uint8_t GetBinding(const Elf32_Sym& sym) {
  return ELF32_ST_BIND(sym.st_info);
}

uint8_t GetBinding(const Elf64_Sym& sym) {
  return ELF64_ST_BIND(sym.st_info);
}

uint8_t GetType(const Elf32_Sym& sym) {
  return ELF32_ST_TYPE(sym.st_info);
}

uint8_t GetType(const Elf64_Sym& sym) {
  return ELF64_ST_TYPE(sym.st_info);
}

uint8_t GetVisibility(const Elf32_Sym& sym) {
  return ELF32_ST_VISIBILITY(sym.st_other);
}

uint8_t GetVisibility(const Elf64_Sym& sym) {
  return ELF64_ST_VISIBILITY(sym.st_other);
}

}  // namespace

ElfSymbolTableReader::ElfSymbolTableReader(const ProcessMemoryRange* memory,
                                           ElfImageReader* elf_reader,
                                           VMAddress address,
                                           VMSize num_entries)
    : memory_(memory),
      elf_reader_(elf_reader),
      base_address_(address),
      num_entries_(num_entries) {}

ElfSymbolTableReader::~ElfSymbolTableReader() {}

bool ElfSymbolTableReader::GetSymbol(const std::string& name,
                                     SymbolInformation* info) {
  return memory_->Is64Bit() ? ScanSymbolTable<Elf64_Sym>(name, info)
                            : ScanSymbolTable<Elf32_Sym>(name, info);
}

template <typename SymEnt>
bool ElfSymbolTableReader::ScanSymbolTable(const std::string& name,
                                           SymbolInformation* info_out) {
  VMAddress address = base_address_;
  SymEnt entry;
  std::string string;
  size_t i = 0;
  while (i < num_entries_ && memory_->Read(address, sizeof(entry), &entry)) {
    if (elf_reader_->ReadDynamicStringTableAtOffset(entry.st_name, &string) &&
        string == name) {
      info_out->address = entry.st_value;
      info_out->size = entry.st_size;
      info_out->shndx = entry.st_shndx;
      info_out->binding = GetBinding(entry);
      info_out->type = GetType(entry);
      info_out->visibility = GetVisibility(entry);
      return true;
    }
    // TODO(scottmg): This should respect DT_SYMENT if present.
    address += sizeof(entry);
    ++i;
  }
  return false;
}

}  // namespace crashpad