C++程序  |  197行  |  4.71 KB

//===- OutputSectDesc.cpp -------------------------------------------------===//
//
//                     The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include <mcld/Script/OutputSectDesc.h>
#include <mcld/Script/RpnExpr.h>
#include <mcld/Script/StringList.h>
#include <mcld/Script/StrToken.h>
#include <mcld/Script/InputSectDesc.h>
#include <mcld/Support/raw_ostream.h>
#include <mcld/LinkerScript.h>
#include <mcld/Module.h>
#include <llvm/Support/Casting.h>
#include <cassert>

using namespace mcld;

//===----------------------------------------------------------------------===//
// OutputSectDesc
//===----------------------------------------------------------------------===//
OutputSectDesc::OutputSectDesc(const std::string& pName,
                               const Prolog& pProlog)
  : ScriptCommand(ScriptCommand::OUTPUT_SECT_DESC),
    m_Name(pName),
    m_Prolog(pProlog)
{
}

OutputSectDesc::~OutputSectDesc()
{
  for (iterator it = begin(), ie = end(); it != ie; ++it) {
    if (*it != NULL)
      delete *it;
  }
}

void OutputSectDesc::dump() const
{
  mcld::outs() << m_Name << "\t";

  if (m_Prolog.hasVMA()) {
    m_Prolog.vma().dump();
    mcld::outs() << "\t";
  }

  switch (m_Prolog.type()) {
  case NOLOAD:
    mcld::outs() << "(NOLOAD)";
    break;
  case DSECT:
    mcld::outs() << "(DSECT)";
    break;
  case COPY:
    mcld::outs() << "(COPY)";
    break;
  case INFO:
    mcld::outs() << "(INFO)";
    break;
  case OVERLAY:
    mcld::outs() << "(OVERLAY)";
    break;
  default:
    break;
  }
  mcld::outs() << ":\n";

  if (m_Prolog.hasLMA()) {
    mcld::outs() << "\tAT ( ";
    m_Prolog.lma().dump();
    mcld::outs() << " )\n";
  }

  if (m_Prolog.hasAlign()) {
    mcld::outs() << "\tALIGN ( ";
    m_Prolog.align().dump();
    mcld::outs() << " )\n";
  }

  if (m_Prolog.hasSubAlign()) {
    mcld::outs() << "\tSUBALIGN ( ";
    m_Prolog.subAlign().dump();
    mcld::outs() << " )\n";
  }

  switch (m_Prolog.constraint()) {
  case ONLY_IF_RO:
    mcld::outs() << "\tONLY_IF_RO\n";
    break;
  case ONLY_IF_RW:
    mcld::outs() << "\tONLY_IF_RW\n";
    break;
  default:
    break;
  }

  mcld::outs() << "\t{\n";
  for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
    switch ((*it)->getKind()) {
    case ScriptCommand::ASSIGNMENT:
    case ScriptCommand::INPUT_SECT_DESC:
      mcld::outs() << "\t\t";
      (*it)->dump();
      break;
    default:
      assert(0);
      break;
    }
  }
  mcld::outs() << "\t}";

  if (m_Epilog.hasRegion())
    mcld::outs() << "\t>" << m_Epilog.region();
  if (m_Epilog.hasLMARegion())
    mcld::outs() << "\tAT>" << m_Epilog.lmaRegion();

  if (m_Epilog.hasPhdrs()) {
    for (StringList::const_iterator it = m_Epilog.phdrs().begin(),
      ie = m_Epilog.phdrs().end(); it != ie; ++it) {
      assert((*it)->kind() == StrToken::String);
      mcld::outs() << ":" << (*it)->name() << " ";
    }
  }

  if (m_Epilog.hasFillExp()) {
    mcld::outs() << "= ";
    m_Epilog.fillExp().dump();
  }
  mcld::outs() << "\n";
}

void OutputSectDesc::push_back(ScriptCommand* pCommand)
{
  switch (pCommand->getKind()) {
  case ScriptCommand::ASSIGNMENT:
  case ScriptCommand::INPUT_SECT_DESC:
    m_OutputSectCmds.push_back(pCommand);
    break;
  default:
    assert(0);
    break;
  }
}

void OutputSectDesc::setEpilog(const Epilog& pEpilog)
{
  m_Epilog.m_pRegion    = pEpilog.m_pRegion;
  m_Epilog.m_pLMARegion = pEpilog.m_pLMARegion;
  m_Epilog.m_pPhdrs     = pEpilog.m_pPhdrs;
  m_Epilog.m_pFillExp   = pEpilog.m_pFillExp;
}

void OutputSectDesc::activate(Module& pModule)
{
  // Assignment in an output section
  OutputSectCmds assignments;

  for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
    switch ((*it)->getKind()) {
    case ScriptCommand::ASSIGNMENT:
      assignments.push_back(*it);
      break;
    case ScriptCommand::INPUT_SECT_DESC: {
      (*it)->activate(pModule);

      for (iterator assign = assignments.begin(), assignEnd = assignments.end();
        assign != assignEnd; ++assign) {
        (*assign)->activate(pModule);
      }
      assignments.clear();
      break;
    }
    default:
      assert(0);
      break;
    }
  }

  if (!assignments.empty()) {
    InputSectDesc::Spec spec;;
    spec.m_pWildcardFile = NULL;
    spec.m_pExcludeFiles = NULL;
    spec.m_pWildcardSections = NULL;
    InputSectDesc inputDesc(InputSectDesc::Keep, spec, *this);
    pModule.getScript().sectionMap().insert(inputDesc, *this);

    for (iterator assign = assignments.begin(), assignEnd = assignments.end();
      assign != assignEnd; ++assign) {
      (*assign)->activate(pModule);
    }
    assignments.clear();
  }
}