//===- Space.cpp ----------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include <mcld/Support/Space.h> #include <mcld/Support/FileHandle.h> #include <mcld/Support/MsgHandling.h> #include <mcld/Support/SystemUtils.h> #include <cstdlib> #include <unistd.h> using namespace mcld; //===----------------------------------------------------------------------===// // constant data //===----------------------------------------------------------------------===// static const int PageSize = mcld::sys::GetPageSize(); //===----------------------------------------------------------------------===// // Non-member functions //===----------------------------------------------------------------------===// // // low address A page high address // |--------------------|------------------| // ^ page_offset ^ pFileOffset ^ page_boundary // // Given a file offset, return the page offset. // return the first page boundary \b before pFileOffset inline static off_t page_offset(off_t pFileOffset) { return pFileOffset & ~ (PageSize - 1); } // page_boundary - Given a file size, return the size to read integral pages. // return the first page boundary \b after pFileOffset inline static off_t page_boundary(off_t pFileOffset) { return (pFileOffset + (PageSize - 1)) & ~ (PageSize - 1); } inline static Space::Type policy(off_t pOffset, size_t pLength) { #if defined(MCLD_ON_WIN32) return Space::ALLOCATED_ARRAY; #endif const size_t threshold = (PageSize*3)/4; // 3/4 page size in Linux if (pLength < threshold) return Space::ALLOCATED_ARRAY; else return Space::MMAPED; } //===----------------------------------------------------------------------===// // Space //===----------------------------------------------------------------------===// Space::Space() : m_Data(NULL), m_StartOffset(0), m_Size(0), m_RegionCount(0), m_Type(UNALLOCATED) { } Space::Space(Space::Type pType, void* pMemBuffer, size_t pSize) : m_Data(static_cast<Address>(pMemBuffer)), m_StartOffset(0), m_Size(pSize), m_RegionCount(0), m_Type(pType) { } Space::~Space() { // do nothing. m_Data is deleted by @ref releaseSpace } Space* Space::Create(void* pMemBuffer, size_t pSize) { Space* result = new Space(EXTERNAL, pMemBuffer, pSize); return result; } Space* Space::Create(FileHandle& pHandler, size_t pStart, size_t pSize) { Type type; void* memory = NULL; Space* result = NULL; size_t start = 0, size = 0, total_offset = 0; switch(type = policy(pStart, pSize)) { case ALLOCATED_ARRAY: { // adjust total_offset, start and size total_offset = pStart + pSize; start = pStart; if (total_offset > pHandler.size()) { if (pHandler.isWritable()) { size = pSize; pHandler.truncate(total_offset); } else if (pHandler.size() > start) { // not writable -> shrink the size size = pHandler.size() - start; } else { // create a space out of a read-only file. fatal(diag::err_cannot_read_small_file) << pHandler.path() << pHandler.size() << start << size; } } else { // within the space. size = pSize; } // malloc memory = (void*)malloc(size); if (!pHandler.read(memory, start, size)) error(diag::err_cannot_read_file) << pHandler.path() << start << size; break; } case MMAPED: { // adjust total_offset, start and size total_offset = page_boundary(pStart + pSize); start = page_offset(pStart); if (total_offset > pHandler.size()) { if (pHandler.isWritable()) { size = page_boundary((pStart - start) + pSize); pHandler.truncate(total_offset); } else if (pHandler.size() > start) size = pHandler.size() - start; else { // create a space out of a read-only file. fatal(diag::err_cannot_read_small_file) << pHandler.path() << pHandler.size() << start << size; } } else size = page_boundary((pStart - start) + pSize); // mmap if (!pHandler.mmap(memory, start, size)) error(diag::err_cannot_mmap_file) << pHandler.path() << start << size; break; } default: break; } // end of switch result = new Space(type, memory, size); result->setStart(start); return result; } void Space::Destroy(Space*& pSpace) { delete pSpace; pSpace = NULL; } void Space::Release(Space* pSpace, FileHandle& pHandler) { if (NULL == pSpace) return; switch(pSpace->type()) { case ALLOCATED_ARRAY: free(pSpace->memory()); break; case MMAPED: if (!pHandler.munmap(pSpace->memory(), pSpace->size())) error(diag::err_cannot_munmap_file) << pHandler.path(); break; default: // external and unallocated memory buffers break; } // end of switch } void Space::Sync(Space* pSpace, FileHandle& pHandler) { if (NULL == pSpace || !pHandler.isWritable()) return; switch(pSpace->type()) { case Space::ALLOCATED_ARRAY: { if (!pHandler.write(pSpace->memory(), pSpace->start(), pSpace->size())) { error(diag::err_cannot_write_file) << pHandler.path() << pSpace->start() << pSpace->size(); } return; } case Space::MMAPED: default: { // system will eventually write bakc the memory after // calling ::munmap return; } } // end of switch }