-
Notifications
You must be signed in to change notification settings - Fork 208
/
_info-bin.sh
executable file
·126 lines (119 loc) · 5.8 KB
/
_info-bin.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#!/bin/sh
# Copyright (C) Viktor Szakats. See LICENSE.md
# SPDX-License-Identifier: MIT
# shellcheck disable=SC3040,SC2039
set -o xtrace -o errexit -o nounset; [ -n "${BASH:-}${ZSH_NAME:-}" ] && set -o pipefail
filetype=''
is_curl='0'
while [ "${1#--*}" != "${1:-}" ]; do
if [ "$1" = '--filetype' ]; then
shift; filetype="$1"; shift
elif [ "$1" = '--is-curl' ]; then
is_curl='1'; shift
fi
done
while [ -n "${1:-}" ]; do
f="$1"; shift
if [ "${_OS}" = 'win' ]; then
# = llvm-readobj --coff-imports | grep -a -F 'Name: ' | sort -r -f
TZ=UTC "${_OBJDUMP}" --all-headers "${f}" | grep -a -E -i "(file format|DLL Name|Time/Date)" | sort -r -f
if [ "${is_curl}" = '1' ]; then
# Verify exported curl symbols
# = llvm-readobj --coff-exports | grep -a -E 'Name: .+' | grep -a -F ' curl_'
if [ "${filetype}" = 'exe' ]; then
"${_OBJDUMP}" --all-headers "${f}" | grep -a -F ' curl_' && false # should not have any hits for statically linked curl
else
"${_OBJDUMP}" --all-headers "${f}" | grep -a -F ' curl_' || false # show public libcurl APIs (in a well-defined order)
if [ "${_CC}" = 'llvm' ]; then
"${_READELF}" --coff-exports "${f}" | grep -a -E 'Name: .+' | grep -a -F -v ' curl_' && false # should not export anything else except the libcurl API
fi
fi
fi
if [ "${_CC}" = 'llvm' ]; then
# Dump 'Characteristics' flags, e.g.
# IMAGE_DLL_CHARACTERISTICS_DYNAMIC_BASE, IMAGE_DLL_CHARACTERISTICS_HIGH_ENTROPY_VA
# IMAGE_DLL_CHARACTERISTICS_NX_COMPAT, IMAGE_DLL_CHARACTERISTICS_TERMINAL_SERVER_AWARE
# Dump 'ExtendedCharacteristics' flags, e.g.
# IMAGE_DLL_CHARACTERISTICS_EX_CET_COMPAT
"${_READELF}" --file-headers --coff-debug-directory "${f}" | grep -a -F 'IMAGE_DLL_CHARACTERISTICS_' | sort || true
# Dump cfguard load configuration flags
# CF_FUNCTION_TABLE_PRESENT, CF_INSTRUMENTED, CF_LONGJUMP_TABLE_PRESENT (optional)
# binutils readelf (as of v2.40) does not recognize this option
"${_READELF}" --coff-load-config "${f}" | grep -a -E 'CF_[A-Z_]' | sort || true
else
# Dump 'DllCharacteristics' flags, e.g.
# HIGH_ENTROPY_VA, DYNAMIC_BASE, NX_COMPAT, GUARD_CF, TERMINAL_SERVICE_AWARE
"${_OBJDUMP}" --all-headers "${f}" | grep -a -E -o '^\s+[A-Z_]{4,}$' | sort
fi
if [ "${filetype}" = 'exe' ] && [ "${is_curl}" = '1' ]; then # should be the same output for the DLL
# regexp compatible with llvm and binutils output
# = llvm-readobj --coff-imports | grep -a -F 'Symbol: ' | sort
"${_OBJDUMP}" --all-headers "${f}" \
| grep -a -E ' [0-9]{1,5} +[a-zA-Z_][a-zA-Z0-9_]*$' \
| grep -a -E -o '\S+$' | sort || true
fi
elif [ "${_OS}" = 'mac' ]; then
if [ "${_CC}" = 'gcc' ] || [ "${_TOOLCHAIN}" = 'llvm-apple' ]; then
_prefix=''
else
_prefix='llvm-' # standard llvm
fi
TZ=UTC "${_prefix}objdump" --arch=all --private-headers "${f}" | grep -a -i -F 'magic'
# -dyld_info ignored by llvm-otool as of v16.0.6
TZ=UTC "${_prefix}otool" -arch all -f -v -L -dyld_info "${f}"
# Display `LC_BUILD_VERSION` / `LC_VERSION_MIN_MACOSX` info
TZ=UTC "${_prefix}otool" -arch all -f -l "${f}" | grep -a -w -E '(\(architecture|^ *(minos|version|sdk))'
if [ "${is_curl}" = '1' ] && [ "${filetype}" != 'exe' ]; then # Verify exported curl symbols
"${NM}" --extern-only --defined-only "${f}" # -g -U
"${NM}" --extern-only --defined-only "${f}" | grep -a -F ' _curl_' | sort || false # -g -U
# should not export anything else except the libcurl API
if "${NM}" --extern-only --defined-only "${f}" | grep -a -F -v ' T _curl_'; then
echo "! Error: '${f}' exports non-curl symbols."
exit 1
fi
fi
elif [ "${_OS}" = 'linux' ]; then
# NOTE: objdump -syms (-t) to show symbol visibility flags
"${_READELF}" --file-header --dynamic "${f}"
"${_READELF}" --program-headers "${f}"
if command -v checksec >/dev/null 2>&1; then
if [ "${_DISTRO}" = 'alpine' ]; then
checksec --json --file "${f}" | jq # checksec-rs
else
# FIXME: `checksec` uses `lld` which is unsafe.
# May fail e.g. in cross-builds when `ldd` is not present
checksec --format=json --file="${f}" | jq || true
# We have seen this fail in some cases, so ignore exit code
checksec --format=xml --fortify-file="${f}" || true # duplicate keys in json, cannot apply jq
fi
fi
if [ "${_CC}" = 'llvm' ]; then
"${_READELF}" --headers "${f}" | grep -a -F '.note.gnu.property' || true
fi
if [ "${is_curl}" = '1' ]; then
# Show linked GLIBC versions
# https://en.wikipedia.org/wiki/Glibc#Version_history
if [ "${_CPU}" = 'a64' ] || \
[ "${_CPU}" = 'r64' ]; then
filter='@GLIBC_2\.(17|2[0-9])$' # Exclude: 2.17 (2012-12) and 2.2x (2019-02)
else
filter='@GLIBC_([0-9]+\.[0-9]+\.[0-9]+|2\.([0-9]|1[0-9]))$' # Exclude: x.y.z, 2.x, 2.1x (-2014-02)
fi
# nm -D -u =
"${NM}" --dynamic --undefined-only "${f}" \
| grep -E -v "${filter}" \
| grep -E -o '@GLIBC_[0-9]+\.[0-9]+$' | sed 's/@GLIBC_//g' | sort -u -V || true
"${NM}" --dynamic --undefined-only "${f}" \
| grep -F '@GLIBC_' | grep -E -v "${filter}" || true
if [ "${filetype}" != 'exe' ]; then # Verify exported curl symbols
"${NM}" --dynamic --defined-only "${f}" # -D -U
"${NM}" --dynamic --defined-only "${f}" | grep -a -F ' curl_' | sort || false # -D -U
# Allowlist:
# - MUSL: _fini, _init
# - ?: _start, _start_c
"${NM}" --dynamic --defined-only "${f}" | grep -a -E -v ' T (curl_|(_fini|_init|_start|_start_c)$)' || true # FIXME: should not export anything else except the libcurl API
fi
# nm -D -g = --dynamic --extern-only
fi
fi
done