//===- MemoryArea.cpp -----------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include <mcld/Support/MemoryArea.h> #include <mcld/Support/Space.h> #include <mcld/Support/MemoryRegion.h> #include <mcld/Support/FileHandle.h> #include <mcld/Support/MsgHandling.h> using namespace mcld; //===--------------------------------------------------------------------===// // MemoryArea //===--------------------------------------------------------------------===// // MemoryArea - special constructor // This constructor is used for *SPECIAL* situation. I'm sorry I can not // reveal what is the special situation. MemoryArea::MemoryArea(Space& pUniverse) : m_pFileHandle(NULL) { m_SpaceMap.insert(std::make_pair(Key(pUniverse.start(), pUniverse.size()), &pUniverse)); } MemoryArea::MemoryArea(FileHandle& pFileHandle) : m_pFileHandle(&pFileHandle) { } MemoryArea::~MemoryArea() { } // The layout of MemorySpace in the virtual memory space // // | : page boundary // [,]: MemoryRegion // - : fillment // = : data // // |---[=|====|====|==]--| // ^ ^ ^ ^ // | | | | // | r_start +r_len | // space.data +space.size // // space.file_offset is the offset of the mapped file segment from the start of // the file. if the MemorySpace's type is ALLOCATED_ARRAY, the distances of // (space.data, r_start) and (r_len, space.size) are zero. // MemoryRegion* MemoryArea::request(size_t pOffset, size_t pLength) { Space* space = find(pOffset, pLength); if (NULL == space) { // not found if (NULL == m_pFileHandle) { // if m_pFileHandle is NULL, clients delegate us an universal Space and // we never remove it. In that way, space can not be NULL. unreachable(diag::err_out_of_range_region) << pOffset << pLength; } space = Space::Create(*m_pFileHandle, pOffset, pLength); m_SpaceMap.insert(std::make_pair(Key(space->start(), space->size()), space)); } // adjust r_start off_t distance = pOffset - space->start(); void* r_start = space->memory() + distance; // now, we have a legal space to hold the new MemoryRegion return MemoryRegion::Create(r_start, pLength, *space); } // release - release a MemoryRegion void MemoryArea::release(MemoryRegion* pRegion) { if (NULL == pRegion) return; Space *space = pRegion->parent(); MemoryRegion::Destroy(pRegion); if (0 == space->numOfRegions()) { if (NULL != m_pFileHandle) { // if m_pFileHandle is NULL, clients delegate us an universal Space and // we never remove it. Otherwise, we have to synchronize and release // Space. if (m_pFileHandle->isWritable()) { // synchronize writable space before we release it. Space::Sync(space, *m_pFileHandle); } std::pair<SpaceMapType::iterator, SpaceMapType::iterator> range = m_SpaceMap.equal_range(Key(space->start(), space->size())); SpaceMapType::iterator it; for (it = range.first; it != range.second; ++it) { if (space == it->second) break; } m_SpaceMap.erase(it); Space::Release(space, *m_pFileHandle); assert(NULL != space); Space::Destroy(space); } } } // clear - release all MemoryRegions void MemoryArea::clear() { if (NULL == m_pFileHandle) return; SpaceMapType::iterator space, sEnd = m_SpaceMap.end(); if (m_pFileHandle->isWritable()) { for (space = m_SpaceMap.begin(); space != sEnd; ++space) { Space::Sync(space->second, *m_pFileHandle); Space::Release(space->second, *m_pFileHandle); assert(NULL != space->second); Space::Destroy(space->second); } } else { for (space = m_SpaceMap.begin(); space != sEnd; ++space) { Space::Release(space->second, *m_pFileHandle); assert(NULL != space->second); Space::Destroy(space->second); } } m_SpaceMap.clear(); } //===--------------------------------------------------------------------===// // SpaceList methods //===--------------------------------------------------------------------===// Space* MemoryArea::find(size_t pOffset, size_t pLength) { std::pair<SpaceMapType::iterator, SpaceMapType::iterator> range = m_SpaceMap.equal_range(Key(pOffset, pLength)); SpaceMapType::iterator it; for (it = range.first; it != range.second; ++it) { if ((it->second->start() <= pOffset) && ((pOffset + pLength) <= (it->second->start() + it->second->size()))) return it->second; } return NULL; } const Space* MemoryArea::find(size_t pOffset, size_t pLength) const { std::pair<SpaceMapType::const_iterator, SpaceMapType::const_iterator> range = m_SpaceMap.equal_range(Key(pOffset, pLength)); SpaceMapType::const_iterator it; for (it = range.first; it != range.second; ++it) { if ((it->second->start() <= pOffset) && ((pOffset + pLength) <= (it->second->start() + it->second->size()))) return it->second; } return NULL; }