blob: 4ba04403002d12acda19525f28e04b2b136d9f6e [file] [log] [blame]
/* Functions for x86 GNU property.
Copyright (C) 2017-2021 Free Software Foundation, Inc.
This file is part of GCC.
GCC is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
GCC is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with GCC; see the file COPYING3. If not see
<http://www.gnu.org/licenses/>. */
#include "config.h"
#include "system.h"
#include "coretypes.h"
#include "tm.h"
#include "output.h"
#include "linux-common.h"
static void
emit_gnu_property (unsigned int type, unsigned int data)
{
int p2align = ptr_mode == SImode ? 2 : 3;
switch_to_section (get_section (".note.gnu.property",
SECTION_NOTYPE, NULL));
ASM_OUTPUT_ALIGN (asm_out_file, p2align);
/* name length. */
fprintf (asm_out_file, ASM_LONG "1f - 0f\n");
/* data length. */
fprintf (asm_out_file, ASM_LONG "4f - 1f\n");
/* note type: NT_GNU_PROPERTY_TYPE_0. */
fprintf (asm_out_file, ASM_LONG "5\n");
fprintf (asm_out_file, "0:\n");
/* vendor name: "GNU". */
fprintf (asm_out_file, STRING_ASM_OP "\"GNU\"\n");
fprintf (asm_out_file, "1:\n");
ASM_OUTPUT_ALIGN (asm_out_file, p2align);
/* pr_type. */
fprintf (asm_out_file, ASM_LONG "0x%x\n", type);
/* pr_datasz. */
fprintf (asm_out_file, ASM_LONG "3f - 2f\n");
fprintf (asm_out_file, "2:\n");
fprintf (asm_out_file, ASM_LONG "0x%x\n", data);
fprintf (asm_out_file, "3:\n");
ASM_OUTPUT_ALIGN (asm_out_file, p2align);
fprintf (asm_out_file, "4:\n");
}
void
file_end_indicate_exec_stack_and_gnu_property (void)
{
file_end_indicate_exec_stack ();
if (flag_cf_protection == CF_NONE && !ix86_needed)
return;
unsigned int feature_1 = 0;
if (flag_cf_protection & CF_BRANCH)
/* GNU_PROPERTY_X86_FEATURE_1_IBT. */
feature_1 |= 0x1;
if (flag_cf_protection & CF_RETURN)
/* GNU_PROPERTY_X86_FEATURE_1_SHSTK. */
feature_1 |= 0x2;
/* Generate GNU_PROPERTY_X86_FEATURE_1_AND. */
if (feature_1)
emit_gnu_property (0xc0000002, feature_1);
unsigned int isa_1 = 0;
if (ix86_needed)
{
/* GNU_PROPERTY_X86_ISA_1_BASELINE. */
if (TARGET_64BIT
|| TARGET_FXSR
|| TARGET_80387
|| TARGET_MMX
|| TARGET_SSE
|| TARGET_SSE2)
isa_1 |= 1 << 0;
/* GNU_PROPERTY_X86_ISA_1_V2. */
if (TARGET_CMPXCHG16B
|| (TARGET_64BIT && TARGET_SAHF)
|| TARGET_POPCNT
|| TARGET_SSE3
|| TARGET_SSSE3
|| TARGET_SSE4_1
|| TARGET_SSE4_2)
isa_1 |= 1 << 1;
/* GNU_PROPERTY_X86_ISA_1_V3. */
if (TARGET_AVX
|| TARGET_AVX2
|| TARGET_F16C
|| TARGET_FMA
|| TARGET_LZCNT
|| TARGET_MOVBE
|| TARGET_XSAVE)
isa_1 |= 1 << 2;
/* GNU_PROPERTY_X86_ISA_1_V4. */
if (TARGET_AVX512F
|| TARGET_AVX512BW
|| TARGET_AVX512CD
|| TARGET_AVX512DQ
|| TARGET_AVX512VL)
isa_1 |= 1 << 3;
}
/* Generate GNU_PROPERTY_X86_ISA_1_NEEDED. */
if (isa_1)
emit_gnu_property (0xc0008002, isa_1);
}