/* * Copyright 2011, The Android Open Source Project * * 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. */ #ifndef ELF_SECTION_PROGBITS_HXX #define ELF_SECTION_PROGBITS_HXX #include "ELFTypes.h" #include "StubLayout.h" #include <llvm/Support/Format.h> #include <llvm/Support/raw_ostream.h> #include "utils/raw_ostream.h" #include <string.h> template <unsigned Bitwidth> template <typename Archiver> ELFSectionProgBits<Bitwidth> * ELFSectionProgBits<Bitwidth>::read(Archiver &AR, ELFObjectTy *owner, ELFSectionHeaderTy const *sh) { int machine = owner->getHeader()->getMachine(); ELFSectionProgBits *secp = new ELFSectionProgBits(machine); std::unique_ptr<ELFSectionProgBits> result(secp); size_t max_num_stubs = 0; // Align section boundary to 4 bytes. size_t section_size = (sh->getSize() + 3) / 4 * 4; size_t alloc_size = section_size; StubLayout *stubs = result->getStubLayout(); if (stubs) { // Compute the maximal possible numbers of stubs max_num_stubs = 0; for (const char* prefix : {".rel", ".rela"}) { std::string reltab_name(prefix + std::string(sh->getName())); ELFSectionRelTableTy const *reltab = static_cast<ELFSectionRelTableTy *>( owner->getSectionByName(reltab_name.c_str())); if (reltab) { // If we have relocation table, then get the approximation of // maximum numbers of stubs. max_num_stubs += reltab->getMaxNumStubs(owner); } } // Compute the stub table size size_t stub_table_size = stubs->calcStubTableSize(max_num_stubs); // Allocate PROGBITS section with stubs table alloc_size += stub_table_size; } // Allocate text section if (!result->chunk.allocate(alloc_size)) { return NULL; } if (stubs) { stubs->initStubTable(result->chunk.getBuffer() + section_size, max_num_stubs); } result->sh = sh; if (!result->serialize(AR)) { // Unable to read the progbits section. return NULL; } return result.release(); } #endif // ELF_SECTION_PROGBITS_HXX