mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-11 20:32:35 -03:00
Merge #19525: build: add -Wl,-z,separate-code to hardening flags
65d0f1a533
devtools: Add security check for separate_code (Wladimir J. van der Laan)2e9e6377f1
build: add -Wl,-z,separate-code to hardening flags (fanquake) Pull request description: TLDR: We are generally explicit about the hardening related flags we use, rather than letting the distro / toolchain decide via their defaults. This PR adds `-z,separate-code` which has been enabled by default for Linux targets since binutils 2.31. Ubuntu Bionic (currently used for gitian) ships with binutils 2.30, so this will enable the option for those builds. This flag was added to binutils/ld in the 2.30 release, see commit c11c786f0b45617bb8807ab6a57220d5ff50e414: > The new "-z separate-code" option will generate separate code LOAD segment which must be in wholly disjoint pages from any other data. It was made the default for Linux/x86 targets in the 2.31 release, see commit f6aec96dce1ddbd8961a3aa8a2925db2021719bb: > This patch adds --enable-separate-code to ld configure to turn on -z separate-code by default and enables it by default for Linux/x86. This avoids mixing code pages with data to improve cache performance as well as security. > To reduce x86-64 executable and shared object sizes, the maximum page size is reduced from 2MB to 4KB when -z separate-code is turned on by default. Note: -z max-page-size= can be used to set the maximum page size. > We compared SPEC CPU 2017 performance before and after this change on Skylake server. There are no any significant performance changes. Everything is mostly below +/-1%. Support was also added to LLVMs lld: https://reviews.llvm.org/D64903, however there it remains off by default. There were concerns about an increase in binary size, however in our case, the difference would seem negligible, given we are shipping a multi-megabyte binary, which then downloads 100's of GBs of data. Also note that most recent versions of distros are shipping a new enough version of binutils that this is available and/or already on by default (assuming the distro has not turned it off, I haven't checked everywhere): CentOS 8: 2.30 Debian Buster 2.31.1 Fedora 29: 2.31.1 FreeBSD: 2.33 GNU Guix: 2.33 / 2.34 Ubuntu 18.04: 2.30 Related threads / discussion: https://bugzilla.redhat.com/show_bug.cgi?id=1623218 The ELF header when building on Debian Buster (where it's already enabled by default in binutils): ```bash Program Header: PHDR off 0x0000000000000040 vaddr 0x0000000000000040 paddr 0x0000000000000040 align 2**3 filesz 0x00000000000002a0 memsz 0x00000000000002a0 flags r-- INTERP off 0x00000000000002e0 vaddr 0x00000000000002e0 paddr 0x00000000000002e0 align 2**0 filesz 0x000000000000001c memsz 0x000000000000001c flags r-- LOAD off 0x0000000000000000 vaddr 0x0000000000000000 paddr 0x0000000000000000 align 2**12 filesz 0x0000000000038f10 memsz 0x0000000000038f10 flags r-- LOAD off 0x0000000000039000 vaddr 0x0000000000039000 paddr 0x0000000000039000 align 2**12 filesz 0x00000000006b9389 memsz 0x00000000006b9389 flags r-x LOAD off 0x00000000006f3000 vaddr 0x00000000006f3000 paddr 0x00000000006f3000 align 2**12 filesz 0x0000000000204847 memsz 0x0000000000204847 flags r-- LOAD off 0x00000000008f7920 vaddr 0x00000000008f8920 paddr 0x00000000008f8920 align 2**12 filesz 0x00000000000183e0 memsz 0x0000000000022fd0 flags rw- DYNAMIC off 0x000000000090adb0 vaddr 0x000000000090bdb0 paddr 0x000000000090bdb0 align 2**3 filesz 0x0000000000000240 memsz 0x0000000000000240 flags rw- ``` vs when opting out using `-Wl,-z,noseparate-code`: ```bash Program Header: PHDR off 0x0000000000000040 vaddr 0x0000000000000040 paddr 0x0000000000000040 align 2**3 filesz 0x0000000000000230 memsz 0x0000000000000230 flags r-- INTERP off 0x0000000000000270 vaddr 0x0000000000000270 paddr 0x0000000000000270 align 2**0 filesz 0x000000000000001c memsz 0x000000000000001c flags r-- LOAD off 0x0000000000000000 vaddr 0x0000000000000000 paddr 0x0000000000000000 align 2**12 filesz 0x00000000008f6a87 memsz 0x00000000008f6a87 flags r-x LOAD off 0x00000000008f7920 vaddr 0x00000000008f8920 paddr 0x00000000008f8920 align 2**12 filesz 0x00000000000183e0 memsz 0x0000000000022fd0 flags rw- DYNAMIC off 0x000000000090adb0 vaddr 0x000000000090bdb0 paddr 0x000000000090bdb0 align 2**3 filesz 0x0000000000000240 memsz 0x0000000000000240 flags rw- ``` ACKs for top commit: laanwj: ACK65d0f1a533
Tree-SHA512: 6e40e434efea8a8e39f6cb244dfd16aaa5a9db5a2ea762a05d1727357b20e33b7e47c1a652ee88490c9d7952a4caa2f992396fb30346239300d37ae123e36d49
This commit is contained in:
commit
400f45ec9b
3 changed files with 100 additions and 17 deletions
|
@ -785,6 +785,7 @@ if test x$use_hardening != xno; then
|
||||||
AX_CHECK_LINK_FLAG([[-Wl,--high-entropy-va]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,--high-entropy-va"],, [[$LDFLAG_WERROR]])
|
AX_CHECK_LINK_FLAG([[-Wl,--high-entropy-va]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,--high-entropy-va"],, [[$LDFLAG_WERROR]])
|
||||||
AX_CHECK_LINK_FLAG([[-Wl,-z,relro]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,-z,relro"],, [[$LDFLAG_WERROR]])
|
AX_CHECK_LINK_FLAG([[-Wl,-z,relro]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,-z,relro"],, [[$LDFLAG_WERROR]])
|
||||||
AX_CHECK_LINK_FLAG([[-Wl,-z,now]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,-z,now"],, [[$LDFLAG_WERROR]])
|
AX_CHECK_LINK_FLAG([[-Wl,-z,now]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,-z,now"],, [[$LDFLAG_WERROR]])
|
||||||
|
AX_CHECK_LINK_FLAG([[-Wl,-z,separate-code]], [HARDENED_LDFLAGS="$HARDENED_LDFLAGS -Wl,-z,separate-code"],, [[$LDFLAG_WERROR]])
|
||||||
AX_CHECK_LINK_FLAG([[-fPIE -pie]], [PIE_FLAGS="-fPIE"; HARDENED_LDFLAGS="$HARDENED_LDFLAGS -pie"],, [[$CXXFLAG_WERROR]])
|
AX_CHECK_LINK_FLAG([[-fPIE -pie]], [PIE_FLAGS="-fPIE"; HARDENED_LDFLAGS="$HARDENED_LDFLAGS -pie"],, [[$CXXFLAG_WERROR]])
|
||||||
|
|
||||||
case $host in
|
case $host in
|
||||||
|
|
|
@ -40,25 +40,48 @@ def get_ELF_program_headers(executable):
|
||||||
stdout = run_command([READELF_CMD, '-l', '-W', executable])
|
stdout = run_command([READELF_CMD, '-l', '-W', executable])
|
||||||
|
|
||||||
in_headers = False
|
in_headers = False
|
||||||
count = 0
|
|
||||||
headers = []
|
headers = []
|
||||||
for line in stdout.splitlines():
|
for line in stdout.splitlines():
|
||||||
if line.startswith('Program Headers:'):
|
if line.startswith('Program Headers:'):
|
||||||
in_headers = True
|
in_headers = True
|
||||||
|
count = 0
|
||||||
if line == '':
|
if line == '':
|
||||||
in_headers = False
|
in_headers = False
|
||||||
if in_headers:
|
if in_headers:
|
||||||
if count == 1: # header line
|
if count == 1: # header line
|
||||||
ofs_typ = line.find('Type')
|
header = [x.strip() for x in line.split()]
|
||||||
ofs_offset = line.find('Offset')
|
ofs_typ = header.index('Type')
|
||||||
ofs_flags = line.find('Flg')
|
ofs_flags = header.index('Flg')
|
||||||
ofs_align = line.find('Align')
|
# assert readelf output is what we expect
|
||||||
if ofs_typ == -1 or ofs_offset == -1 or ofs_flags == -1 or ofs_align == -1:
|
if ofs_typ == -1 or ofs_flags == -1:
|
||||||
raise ValueError('Cannot parse elfread -lW output')
|
raise ValueError('Cannot parse elfread -lW output')
|
||||||
elif count > 1:
|
elif count > 1:
|
||||||
typ = line[ofs_typ:ofs_offset].rstrip()
|
splitline = [x.strip() for x in line.split()]
|
||||||
flags = line[ofs_flags:ofs_align].rstrip()
|
typ = splitline[ofs_typ]
|
||||||
headers.append((typ, flags))
|
if not typ.startswith('[R'): # skip [Requesting ...]
|
||||||
|
splitline = [x.strip() for x in line.split()]
|
||||||
|
flags = splitline[ofs_flags]
|
||||||
|
# check for 'R', ' E'
|
||||||
|
if splitline[ofs_flags + 1] is 'E':
|
||||||
|
flags += ' E'
|
||||||
|
headers.append((typ, flags, []))
|
||||||
|
count += 1
|
||||||
|
|
||||||
|
if line.startswith(' Section to Segment mapping:'):
|
||||||
|
in_mapping = True
|
||||||
|
count = 0
|
||||||
|
if line == '':
|
||||||
|
in_mapping = False
|
||||||
|
if in_mapping:
|
||||||
|
if count == 1: # header line
|
||||||
|
ofs_segment = line.find('Segment')
|
||||||
|
ofs_sections = line.find('Sections...')
|
||||||
|
if ofs_segment == -1 or ofs_sections == -1:
|
||||||
|
raise ValueError('Cannot parse elfread -lW output')
|
||||||
|
elif count > 1:
|
||||||
|
segment = int(line[ofs_segment:ofs_sections].strip())
|
||||||
|
sections = line[ofs_sections:].strip().split()
|
||||||
|
headers[segment][2].extend(sections)
|
||||||
count += 1
|
count += 1
|
||||||
return headers
|
return headers
|
||||||
|
|
||||||
|
@ -68,7 +91,7 @@ def check_ELF_NX(executable) -> bool:
|
||||||
'''
|
'''
|
||||||
have_wx = False
|
have_wx = False
|
||||||
have_gnu_stack = False
|
have_gnu_stack = False
|
||||||
for (typ, flags) in get_ELF_program_headers(executable):
|
for (typ, flags, _) in get_ELF_program_headers(executable):
|
||||||
if typ == 'GNU_STACK':
|
if typ == 'GNU_STACK':
|
||||||
have_gnu_stack = True
|
have_gnu_stack = True
|
||||||
if 'W' in flags and 'E' in flags: # section is both writable and executable
|
if 'W' in flags and 'E' in flags: # section is both writable and executable
|
||||||
|
@ -82,7 +105,7 @@ def check_ELF_RELRO(executable) -> bool:
|
||||||
Dynamic section must have BIND_NOW flag
|
Dynamic section must have BIND_NOW flag
|
||||||
'''
|
'''
|
||||||
have_gnu_relro = False
|
have_gnu_relro = False
|
||||||
for (typ, flags) in get_ELF_program_headers(executable):
|
for (typ, flags, _) in get_ELF_program_headers(executable):
|
||||||
# Note: not checking flags == 'R': here as linkers set the permission differently
|
# Note: not checking flags == 'R': here as linkers set the permission differently
|
||||||
# This does not affect security: the permission flags of the GNU_RELRO program
|
# This does not affect security: the permission flags of the GNU_RELRO program
|
||||||
# header are ignored, the PT_LOAD header determines the effective permissions.
|
# header are ignored, the PT_LOAD header determines the effective permissions.
|
||||||
|
@ -113,6 +136,62 @@ def check_ELF_Canary(executable) -> bool:
|
||||||
ok = True
|
ok = True
|
||||||
return ok
|
return ok
|
||||||
|
|
||||||
|
def check_ELF_separate_code(executable):
|
||||||
|
'''
|
||||||
|
Check that sections are appropriately separated in virtual memory,
|
||||||
|
based on their permissions. This checks for missing -Wl,-z,separate-code
|
||||||
|
and potentially other problems.
|
||||||
|
'''
|
||||||
|
EXPECTED_FLAGS = {
|
||||||
|
# Read + execute
|
||||||
|
'.init': 'R E',
|
||||||
|
'.plt': 'R E',
|
||||||
|
'.plt.got': 'R E',
|
||||||
|
'.plt.sec': 'R E',
|
||||||
|
'.text': 'R E',
|
||||||
|
'.fini': 'R E',
|
||||||
|
# Read-only data
|
||||||
|
'.interp': 'R',
|
||||||
|
'.note.gnu.property': 'R',
|
||||||
|
'.note.gnu.build-id': 'R',
|
||||||
|
'.note.ABI-tag': 'R',
|
||||||
|
'.gnu.hash': 'R',
|
||||||
|
'.dynsym': 'R',
|
||||||
|
'.dynstr': 'R',
|
||||||
|
'.gnu.version': 'R',
|
||||||
|
'.gnu.version_r': 'R',
|
||||||
|
'.rela.dyn': 'R',
|
||||||
|
'.rela.plt': 'R',
|
||||||
|
'.rodata': 'R',
|
||||||
|
'.eh_frame_hdr': 'R',
|
||||||
|
'.eh_frame': 'R',
|
||||||
|
'.qtmetadata': 'R',
|
||||||
|
'.gcc_except_table': 'R',
|
||||||
|
'.stapsdt.base': 'R',
|
||||||
|
# Writable data
|
||||||
|
'.init_array': 'RW',
|
||||||
|
'.fini_array': 'RW',
|
||||||
|
'.dynamic': 'RW',
|
||||||
|
'.got': 'RW',
|
||||||
|
'.data': 'RW',
|
||||||
|
'.bss': 'RW',
|
||||||
|
}
|
||||||
|
# For all LOAD program headers get mapping to the list of sections,
|
||||||
|
# and for each section, remember the flags of the associated program header.
|
||||||
|
flags_per_section = {}
|
||||||
|
for (typ, flags, sections) in get_ELF_program_headers(executable):
|
||||||
|
if typ == 'LOAD':
|
||||||
|
for section in sections:
|
||||||
|
assert(section not in flags_per_section)
|
||||||
|
flags_per_section[section] = flags
|
||||||
|
# Spot-check ELF LOAD program header flags per section
|
||||||
|
# If these sections exist, check them against the expected R/W/E flags
|
||||||
|
for (section, flags) in flags_per_section.items():
|
||||||
|
if section in EXPECTED_FLAGS:
|
||||||
|
if EXPECTED_FLAGS[section] != flags:
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
def get_PE_dll_characteristics(executable) -> int:
|
def get_PE_dll_characteristics(executable) -> int:
|
||||||
'''Get PE DllCharacteristics bits'''
|
'''Get PE DllCharacteristics bits'''
|
||||||
stdout = run_command([OBJDUMP_CMD, '-x', executable])
|
stdout = run_command([OBJDUMP_CMD, '-x', executable])
|
||||||
|
@ -225,7 +304,8 @@ CHECKS = {
|
||||||
('PIE', check_ELF_PIE),
|
('PIE', check_ELF_PIE),
|
||||||
('NX', check_ELF_NX),
|
('NX', check_ELF_NX),
|
||||||
('RELRO', check_ELF_RELRO),
|
('RELRO', check_ELF_RELRO),
|
||||||
('Canary', check_ELF_Canary)
|
('Canary', check_ELF_Canary),
|
||||||
|
('separate_code', check_ELF_separate_code),
|
||||||
],
|
],
|
||||||
'PE': [
|
'PE': [
|
||||||
('DYNAMIC_BASE', check_PE_DYNAMIC_BASE),
|
('DYNAMIC_BASE', check_PE_DYNAMIC_BASE),
|
||||||
|
|
|
@ -31,15 +31,17 @@ class TestSecurityChecks(unittest.TestCase):
|
||||||
cc = 'gcc'
|
cc = 'gcc'
|
||||||
write_testcode(source)
|
write_testcode(source)
|
||||||
|
|
||||||
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-zexecstack','-fno-stack-protector','-Wl,-znorelro','-no-pie','-fno-PIE']),
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-zexecstack','-fno-stack-protector','-Wl,-znorelro','-no-pie','-fno-PIE', '-Wl,-z,separate-code']),
|
||||||
(1, executable+': failed PIE NX RELRO Canary'))
|
(1, executable+': failed PIE NX RELRO Canary'))
|
||||||
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fno-stack-protector','-Wl,-znorelro','-no-pie','-fno-PIE']),
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fno-stack-protector','-Wl,-znorelro','-no-pie','-fno-PIE', '-Wl,-z,separate-code']),
|
||||||
(1, executable+': failed PIE RELRO Canary'))
|
(1, executable+': failed PIE RELRO Canary'))
|
||||||
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-znorelro','-no-pie','-fno-PIE']),
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-znorelro','-no-pie','-fno-PIE', '-Wl,-z,separate-code']),
|
||||||
(1, executable+': failed PIE RELRO'))
|
(1, executable+': failed PIE RELRO'))
|
||||||
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-znorelro','-pie','-fPIE']),
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-znorelro','-pie','-fPIE', '-Wl,-z,separate-code']),
|
||||||
(1, executable+': failed RELRO'))
|
(1, executable+': failed RELRO'))
|
||||||
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-zrelro','-Wl,-z,now','-pie','-fPIE']),
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-zrelro','-Wl,-z,now','-pie','-fPIE', '-Wl,-z,noseparate-code']),
|
||||||
|
(1, executable+': failed separate_code'))
|
||||||
|
self.assertEqual(call_security_check(cc, source, executable, ['-Wl,-znoexecstack','-fstack-protector-all','-Wl,-zrelro','-Wl,-z,now','-pie','-fPIE', '-Wl,-z,separate-code']),
|
||||||
(0, ''))
|
(0, ''))
|
||||||
|
|
||||||
def test_PE(self):
|
def test_PE(self):
|
||||||
|
|
Loading…
Reference in a new issue