//===- SectionMap.cpp -----------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include <mcld/Object/SectionMap.h>
#include <mcld/ADT/StringHash.h>
#include <cassert>
#include <cstring>
using namespace mcld;
SectionMap::NamePair SectionMap::NullName;
//===----------------------------------------------------------------------===//
// SectionMap::NamePair
//===----------------------------------------------------------------------===//
SectionMap::NamePair::NamePair()
: hash(-1) {
}
SectionMap::NamePair::NamePair(const std::string& pFrom, const std::string& pTo)
: from(pFrom), to(pTo) {
hash = SectionMap::hash(pFrom);
}
bool SectionMap::NamePair::isNull() const
{
return (&NullName == this);
}
//===----------------------------------------------------------------------===//
// SectionMap
//===----------------------------------------------------------------------===//
const SectionMap::NamePair& SectionMap::find(const std::string& pFrom) const
{
unsigned int hash = SectionMap::hash(pFrom);
return find(pFrom, hash);
}
SectionMap::NamePair& SectionMap::find(const std::string& pFrom)
{
unsigned int hash = SectionMap::hash(pFrom);
return find(pFrom, hash);
}
const SectionMap::NamePair&
SectionMap::find(const std::string& pFrom, unsigned int pHash) const
{
NamePairList::const_iterator name_hash, nEnd = m_NamePairList.end();
for (name_hash = m_NamePairList.begin(); name_hash != nEnd; ++name_hash) {
if (matched(*name_hash, pFrom, pHash)) {
return *name_hash;
}
}
return NullName;
}
SectionMap::NamePair&
SectionMap::find(const std::string& pFrom, unsigned int pHash)
{
NamePairList::iterator name_hash, nEnd = m_NamePairList.end();
for (name_hash = m_NamePairList.begin(); name_hash != nEnd; ++name_hash) {
if (matched(*name_hash, pFrom, pHash)) {
return *name_hash;
}
}
return NullName;
}
SectionMap::NamePair& SectionMap::append(const std::string &pFrom,
const std::string &pTo,
bool &pExist)
{
NamePair& result = find(pFrom);
if (!result.isNull()) {
pExist = true;
return result;
}
pExist = false;
NamePair entry(pFrom, pTo);
m_NamePairList.push_back(entry);
return m_NamePairList.back();
}
bool SectionMap::matched(const NamePair& pNamePair,
const std::string& pInput,
unsigned int pHashValue) const
{
if ('*' == pNamePair.from[0])
return true;
if (pNamePair.from.size() > pInput.size())
return false;
if (!hash::StringHash<hash::ES>::may_include(pNamePair.hash, pHashValue))
return false;
if (0 == strncmp(pInput.c_str(),
pNamePair.from.c_str(),
pNamePair.from.size())) {
return true;
}
return false;
}
unsigned int SectionMap::hash(const std::string& pString)
{
static hash::StringHash<hash::ES> hash_func;
return hash_func(pString);
}