//===-- Implementation file of gnu_property_section -----------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "startup/linux/gnu_property_section.h" #include "hdr/elf_proxy.h" #include "hdr/link_macros.h" #include "src/__support/CPP/string_view.h" #include "src/__support/macros/config.h" #include "src/string/memory_utils/utils.h" namespace LIBC_NAMESPACE_DECL { // The program property note is basically a note (Elf64_Nhdr) prepended with: // * n_name[4]: should always be "GNU\0" // * n_desc: an array of n_descsz bytes with program property entries // Since we are casting a memory address into this struct, the layout needs to // *exactly* match. struct Elf64_ProgramPropertyNote { Elf64_Nhdr nhdr; unsigned char n_name[4]; unsigned char n_desc[0]; // the size of 'n_desc' depends on n_descsz and is // not known statically. }; // 32-bit variant for ProgramPropertyNote. struct Elf32_ProgramPropertyNote { Elf32_Nhdr nhdr; unsigned char n_name[4]; unsigned char n_desc[0]; // the size of 'n_desc' depends on n_descsz and is // not known statically. }; // A program property consists of a type, the data size, followed by the actual // data and potential padding (aligning it to 64bit). // Since we are casting a memory address into this struct, the layout needs to // *exactly* match (The padding is ommited since it doesn't have actual // content). // pr_data needs to be ElfW_Word aligned, therefore the whole struct needs to be // aligned. struct Elf64_ProgramProperty { Elf64_Word pr_type; Elf64_Word pr_datasz; unsigned char pr_data[0]; }; // 32-bit variant for ProgramProperty. struct Elf32_ProgramProperty { Elf32_Word pr_type; Elf32_Word pr_datasz; unsigned char pr_data[0]; }; bool GnuPropertySection::parse(const ElfW(Phdr) * gnu_property_phdr, const ElfW(Addr) base) { if (!gnu_property_phdr) return false; const auto note_nhdr_size = gnu_property_phdr->p_memsz; // Sanity check we are using the correct phdr and the memory size is large // enough to fit the program property note. if (gnu_property_phdr->p_type != PT_GNU_PROPERTY || note_nhdr_size < sizeof(ElfW(ProgramPropertyNote))) return false; const ElfW(ProgramPropertyNote) *note_nhdr = reinterpret_cast(base + gnu_property_phdr->p_vaddr); if (!note_nhdr) return false; const ElfW(Word) nhdr_desc_size = note_nhdr->nhdr.n_descsz; // sizeof(*note_nhdr) does not include the size of n_desc, // since it is not known at compile time. // The size of it combined with n_descsz cannot exceed the total size of the // program property note. if ((sizeof(*note_nhdr) + nhdr_desc_size) > note_nhdr_size) return false; if (note_nhdr->nhdr.n_namesz != 4 || note_nhdr->nhdr.n_type != NT_GNU_PROPERTY_TYPE_0 || cpp::string_view(reinterpret_cast(note_nhdr->n_name), 4) != cpp::string_view("GNU", 4)) return false; // program property note is valid, we can parse the program property array. ElfW(Word) offset = 0; // Process properties until we can no longer even fit the statically known // size of ProgramProperty. while (offset + sizeof(ElfW(ProgramProperty)) <= nhdr_desc_size) { const ElfW(ProgramProperty) *property = reinterpret_cast( ¬e_nhdr->n_desc[offset]); // Sanity check that property is correctly aligned. if (distance_to_align_up(property) > 0) return false; const ElfW(Xword) property_size = sizeof(*property) + property->pr_datasz; // Also check that pr_data does not reach out of bounds. if ((offset + property_size) > nhdr_desc_size) return false; switch (property->pr_type) { #ifdef LIBC_TARGET_ARCH_IS_X86_64 case GNU_PROPERTY_X86_FEATURE_1_AND: { // PR_DATASZ should always be 4 bytes, for both 32bit and 64bit. if (property->pr_datasz != 4) return false; const uint32_t feature_bitmap = *reinterpret_cast(&property->pr_data[0]); features_.shstk_supported = (feature_bitmap & GNU_PROPERTY_X86_FEATURE_1_SHSTK) != 0; break; } #endif default: break; } offset += static_cast(property_size); } return true; } } // namespace LIBC_NAMESPACE_DECL