mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-04 04:02:26 +00:00
51da9dfb7f
ELFNOTE_START allows callers to specify flags for .pushsection assembler directives. All callsites but ELF_NOTE use "a" for SHF_ALLOC. For vdso's that explicitly use ELF_NOTE_START and BUILD_SALT, the same section is specified twice after preprocessing, once with "a" flag, once without. Example: .pushsection .note.Linux, "a", @note ; .pushsection .note.Linux, "", @note ; While GNU as allows this ordering, it warns for the opposite ordering, making these directives position dependent. We'd prefer not to precisely match this behavior in Clang's integrated assembler. Instead, the non __ASSEMBLY__ definition of ELF_NOTE uses __attribute__((section(".note.Linux"))) which is created with SHF_ALLOC, so let's make the __ASSEMBLY__ definition of ELF_NOTE consistent with C and just always use "a" flag. This allows Clang to assemble a working mainline (5.6) kernel via: $ make CC=clang AS=clang Signed-off-by: Nick Desaulniers <ndesaulniers@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Reviewed-by: Nathan Chancellor <natechancellor@gmail.com> Reviewed-by: Fangrui Song <maskray@google.com> Cc: Jeremy Fitzhardinge <jeremy@goop.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Vincenzo Frascino <vincenzo.frascino@arm.com> Link: https://github.com/ClangBuiltLinux/linux/issues/913 Link: http://lkml.kernel.org/r/20200325231250.99205-1-ndesaulniers@google.com Debugged-by: Ilie Halip <ilie.halip@gmail.com> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
100 lines
3.5 KiB
C
100 lines
3.5 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef _LINUX_ELFNOTE_H
|
|
#define _LINUX_ELFNOTE_H
|
|
/*
|
|
* Helper macros to generate ELF Note structures, which are put into a
|
|
* PT_NOTE segment of the final vmlinux image. These are useful for
|
|
* including name-value pairs of metadata into the kernel binary (or
|
|
* modules?) for use by external programs.
|
|
*
|
|
* Each note has three parts: a name, a type and a desc. The name is
|
|
* intended to distinguish the note's originator, so it would be a
|
|
* company, project, subsystem, etc; it must be in a suitable form for
|
|
* use in a section name. The type is an integer which is used to tag
|
|
* the data, and is considered to be within the "name" namespace (so
|
|
* "FooCo"'s type 42 is distinct from "BarProj"'s type 42). The
|
|
* "desc" field is the actual data. There are no constraints on the
|
|
* desc field's contents, though typically they're fairly small.
|
|
*
|
|
* All notes from a given NAME are put into a section named
|
|
* .note.NAME. When the kernel image is finally linked, all the notes
|
|
* are packed into a single .notes section, which is mapped into the
|
|
* PT_NOTE segment. Because notes for a given name are grouped into
|
|
* the same section, they'll all be adjacent the output file.
|
|
*
|
|
* This file defines macros for both C and assembler use. Their
|
|
* syntax is slightly different, but they're semantically similar.
|
|
*
|
|
* See the ELF specification for more detail about ELF notes.
|
|
*/
|
|
|
|
#ifdef __ASSEMBLER__
|
|
/*
|
|
* Generate a structure with the same shape as Elf{32,64}_Nhdr (which
|
|
* turn out to be the same size and shape), followed by the name and
|
|
* desc data with appropriate padding. The 'desctype' argument is the
|
|
* assembler pseudo op defining the type of the data e.g. .asciz while
|
|
* 'descdata' is the data itself e.g. "hello, world".
|
|
*
|
|
* e.g. ELFNOTE(XYZCo, 42, .asciz, "forty-two")
|
|
* ELFNOTE(XYZCo, 12, .long, 0xdeadbeef)
|
|
*/
|
|
#define ELFNOTE_START(name, type, flags) \
|
|
.pushsection .note.name, flags,@note ; \
|
|
.balign 4 ; \
|
|
.long 2f - 1f /* namesz */ ; \
|
|
.long 4484f - 3f /* descsz */ ; \
|
|
.long type ; \
|
|
1:.asciz #name ; \
|
|
2:.balign 4 ; \
|
|
3:
|
|
|
|
#define ELFNOTE_END \
|
|
4484:.balign 4 ; \
|
|
.popsection ;
|
|
|
|
#define ELFNOTE(name, type, desc) \
|
|
ELFNOTE_START(name, type, "a") \
|
|
desc ; \
|
|
ELFNOTE_END
|
|
|
|
#else /* !__ASSEMBLER__ */
|
|
#include <uapi/linux/elf.h>
|
|
/*
|
|
* Use an anonymous structure which matches the shape of
|
|
* Elf{32,64}_Nhdr, but includes the name and desc data. The size and
|
|
* type of name and desc depend on the macro arguments. "name" must
|
|
* be a literal string, and "desc" must be passed by value. You may
|
|
* only define one note per line, since __LINE__ is used to generate
|
|
* unique symbols.
|
|
*/
|
|
#define _ELFNOTE_PASTE(a,b) a##b
|
|
#define _ELFNOTE(size, name, unique, type, desc) \
|
|
static const struct { \
|
|
struct elf##size##_note _nhdr; \
|
|
unsigned char _name[sizeof(name)] \
|
|
__attribute__((aligned(sizeof(Elf##size##_Word)))); \
|
|
typeof(desc) _desc \
|
|
__attribute__((aligned(sizeof(Elf##size##_Word)))); \
|
|
} _ELFNOTE_PASTE(_note_, unique) \
|
|
__used \
|
|
__attribute__((section(".note." name), \
|
|
aligned(sizeof(Elf##size##_Word)), \
|
|
unused)) = { \
|
|
{ \
|
|
sizeof(name), \
|
|
sizeof(desc), \
|
|
type, \
|
|
}, \
|
|
name, \
|
|
desc \
|
|
}
|
|
#define ELFNOTE(size, name, type, desc) \
|
|
_ELFNOTE(size, name, __LINE__, type, desc)
|
|
|
|
#define ELFNOTE32(name, type, desc) ELFNOTE(32, name, type, desc)
|
|
#define ELFNOTE64(name, type, desc) ELFNOTE(64, name, type, desc)
|
|
#endif /* __ASSEMBLER__ */
|
|
|
|
#endif /* _LINUX_ELFNOTE_H */
|