forked from Qortal/Brooklyn
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
91 lines
3.1 KiB
91 lines
3.1 KiB
#!/bin/bash |
|
# SPDX-License-Identifier: GPL-2.0 |
|
# |
|
# Copyright © 2020, Microsoft Corporation. All rights reserved. |
|
# |
|
# Author: Mickaël Salaün <[email protected]> |
|
# |
|
# Compute and print the To Be Signed (TBS) hash of a certificate. This is used |
|
# as description of keys in the blacklist keyring to identify certificates. |
|
# This output should be redirected, without newline, in a file (hash0.txt) and |
|
# signed to create a PKCS#7 file (hash0.p7s). Both of these files can then be |
|
# loaded in the kernel with. |
|
# |
|
# Exemple on a workstation: |
|
# ./print-cert-tbs-hash.sh certificate-to-invalidate.pem > hash0.txt |
|
# openssl smime -sign -in hash0.txt -inkey builtin-private-key.pem \ |
|
# -signer builtin-certificate.pem -certfile certificate-chain.pem \ |
|
# -noattr -binary -outform DER -out hash0.p7s |
|
# |
|
# Exemple on a managed system: |
|
# keyctl padd blacklist "$(< hash0.txt)" %:.blacklist < hash0.p7s |
|
|
|
set -u -e -o pipefail |
|
|
|
CERT="${1:-}" |
|
BASENAME="$(basename -- "${BASH_SOURCE[0]}")" |
|
|
|
if [ $# -ne 1 ] || [ ! -f "${CERT}" ]; then |
|
echo "usage: ${BASENAME} <certificate>" >&2 |
|
exit 1 |
|
fi |
|
|
|
# Checks that it is indeed a certificate (PEM or DER encoded) and exclude the |
|
# optional PEM text header. |
|
if ! PEM="$(openssl x509 -inform DER -in "${CERT}" 2>/dev/null || openssl x509 -in "${CERT}")"; then |
|
echo "ERROR: Failed to parse certificate" >&2 |
|
exit 1 |
|
fi |
|
|
|
# TBSCertificate starts at the second entry. |
|
# Cf. https://tools.ietf.org/html/rfc3280#section-4.1 |
|
# |
|
# Exemple of first lines printed by openssl asn1parse: |
|
# 0:d=0 hl=4 l= 763 cons: SEQUENCE |
|
# 4:d=1 hl=4 l= 483 cons: SEQUENCE |
|
# 8:d=2 hl=2 l= 3 cons: cont [ 0 ] |
|
# 10:d=3 hl=2 l= 1 prim: INTEGER :02 |
|
# 13:d=2 hl=2 l= 20 prim: INTEGER :3CEB2CB8818D968AC00EEFE195F0DF9665328B7B |
|
# 35:d=2 hl=2 l= 13 cons: SEQUENCE |
|
# 37:d=3 hl=2 l= 9 prim: OBJECT :sha256WithRSAEncryption |
|
RANGE_AND_DIGEST_RE=' |
|
2s/^\s*\([0-9]\+\):d=\s*[0-9]\+\s\+hl=\s*[0-9]\+\s\+l=\s*\([0-9]\+\)\s\+cons:\s*SEQUENCE\s*$/\1 \2/p; |
|
7s/^\s*[0-9]\+:d=\s*[0-9]\+\s\+hl=\s*[0-9]\+\s\+l=\s*[0-9]\+\s\+prim:\s*OBJECT\s*:\(.*\)$/\1/p; |
|
' |
|
|
|
RANGE_AND_DIGEST=($(echo "${PEM}" | \ |
|
openssl asn1parse -in - | \ |
|
sed -n -e "${RANGE_AND_DIGEST_RE}")) |
|
|
|
if [ "${#RANGE_AND_DIGEST[@]}" != 3 ]; then |
|
echo "ERROR: Failed to parse TBSCertificate." >&2 |
|
exit 1 |
|
fi |
|
|
|
OFFSET="${RANGE_AND_DIGEST[0]}" |
|
END="$(( OFFSET + RANGE_AND_DIGEST[1] ))" |
|
DIGEST="${RANGE_AND_DIGEST[2]}" |
|
|
|
# The signature hash algorithm is used by Linux to blacklist certificates. |
|
# Cf. crypto/asymmetric_keys/x509_cert_parser.c:x509_note_pkey_algo() |
|
DIGEST_MATCH="" |
|
while read -r DIGEST_ITEM; do |
|
if [ -z "${DIGEST_ITEM}" ]; then |
|
break |
|
fi |
|
if echo "${DIGEST}" | grep -qiF "${DIGEST_ITEM}"; then |
|
DIGEST_MATCH="${DIGEST_ITEM}" |
|
break |
|
fi |
|
done < <(openssl list -digest-commands | tr ' ' '\n' | sort -ur) |
|
|
|
if [ -z "${DIGEST_MATCH}" ]; then |
|
echo "ERROR: Unknown digest algorithm: ${DIGEST}" >&2 |
|
exit 1 |
|
fi |
|
|
|
echo "${PEM}" | \ |
|
openssl x509 -in - -outform DER | \ |
|
dd "bs=1" "skip=${OFFSET}" "count=${END}" "status=none" | \ |
|
openssl dgst "-${DIGEST_MATCH}" - | \ |
|
awk '{printf "tbs:" $2}'
|
|
|