From 6e39a7d36e4ccf1c6e770d80f9549f89058a25b2 Mon Sep 17 00:00:00 2001
From: quh4gko8 <88831734+quh4gko8@users.noreply.github.com>
Date: Mon, 30 Sep 2024 07:58:57 +0000
Subject: [PATCH] Support deserializing and displaying global security state
extensions
---
.../auditor/AttestationProtocol.java | 84 ++++++++++++++++++-
app/src/main/res/values/strings.xml | 18 ++++
2 files changed, 100 insertions(+), 2 deletions(-)
diff --git a/app/src/main/java/app/attestation/auditor/AttestationProtocol.java b/app/src/main/java/app/attestation/auditor/AttestationProtocol.java
index 40e18267..fded3bc8 100644
--- a/app/src/main/java/app/attestation/auditor/AttestationProtocol.java
+++ b/app/src/main/java/app/attestation/auditor/AttestationProtocol.java
@@ -958,7 +958,8 @@ private static VerificationResult verify(final Context context, final byte[] fin
final boolean accessibility, final boolean deviceAdmin,
final boolean deviceAdminNonSystem, final boolean adbEnabled,
final boolean addUsersWhenLocked, final boolean enrolledBiometrics,
- final boolean oemUnlockAllowed, final boolean systemUser)
+ final boolean oemUnlockAllowed, final boolean systemUser,
+ SecurityStateExt securityStateExt)
throws GeneralSecurityException {
final String fingerprintHex = BaseEncoding.base16().encode(fingerprint);
final byte[] currentFingerprint = getFingerprint(attestationCertificates[0]);
@@ -1134,6 +1135,80 @@ private static VerificationResult verify(final Context context, final byte[] fin
osEnforced.append(context.getString(R.string.system_user,
toYesNoString(context, systemUser)));
+ final boolean appliesToPogoPins = verified.device() == R.string.device_pixel_tablet;
+ final int usbcPortSecurityModePrefix;
+ if (appliesToPogoPins) {
+ usbcPortSecurityModePrefix = R.string.usbc_port_and_pogo_pins;
+ } else {
+ usbcPortSecurityModePrefix = R.string.usbc_port_security_mode;
+ }
+
+ final int usbcPortSecurityModeOffRes;
+ if (appliesToPogoPins) {
+ usbcPortSecurityModeOffRes = R.string.usbc_port_and_pogo_pins_off_title;
+ } else {
+ usbcPortSecurityModeOffRes = R.string.usbc_port_security_mode_off;
+ }
+
+ final byte usbcPortSecurityMode = securityStateExt.portSecurityMode();
+ final int usbcPortSecurityModeValueRes;
+ if (usbcPortSecurityMode == SecurityStateExt.UNKNOWN_VALUE) {
+ usbcPortSecurityModeValueRes = R.string.unknown_value;
+ } else if (usbcPortSecurityMode == SecurityStateExt.INVALID_VALUE) {
+ usbcPortSecurityModeValueRes = R.string.invalid_value;
+ } else {
+ usbcPortSecurityModeValueRes = switch (usbcPortSecurityMode) {
+ case 0 -> usbcPortSecurityModeOffRes;
+ case 1 -> R.string.usbc_port_security_mode_charging_only;
+ case 2 -> R.string.usbc_port_security_mode_charging_only_when_locked;
+ case 3 -> R.string.usbc_port_security_mode_charging_only_when_locked_afu;
+ case 4 -> R.string.usbc_port_security_mode_on;
+ default -> throw new IllegalArgumentException();
+ };
+ }
+ osEnforced.append(context.getString(usbcPortSecurityModePrefix,
+ context.getString(usbcPortSecurityModeValueRes)));
+
+ final short autoRebootMinutes = securityStateExt.autoRebootMinutes();
+ final String autoRebootValueString;
+ if (autoRebootMinutes == SecurityStateExt.UNKNOWN_VALUE) {
+ autoRebootValueString = context.getString(R.string.unknown_value);
+ } else if (autoRebootMinutes == SecurityStateExt.INVALID_VALUE) {
+ autoRebootValueString = context.getString(R.string.invalid_value);
+ } else {
+ final StringBuilder autoRebootValueStrBuilder = new StringBuilder();
+ long autoRebootTimeoutMinutes = TimeUnit.MINUTES.toMinutes(autoRebootMinutes);
+ if (autoRebootMinutes > TimeUnit.HOURS.toMinutes(1)) {
+ final long hours = TimeUnit.MINUTES.toHours(autoRebootTimeoutMinutes);
+ autoRebootValueStrBuilder.append(
+ context.getString(R.string.auto_reboot_hours_value, hours));
+ autoRebootTimeoutMinutes -= TimeUnit.HOURS.toMinutes(hours);
+ }
+
+ if (autoRebootTimeoutMinutes > TimeUnit.MINUTES.toMinutes(1)) {
+ final long minutes = TimeUnit.MINUTES.toMinutes(autoRebootTimeoutMinutes);
+ if (autoRebootValueStrBuilder.length() > 0) {
+ autoRebootValueStrBuilder.append(" ");
+ }
+ autoRebootValueStrBuilder.append(
+ context.getString(R.string.auto_reboot_minutes_value, minutes));
+ }
+
+ autoRebootValueString = autoRebootValueStrBuilder.toString();
+ }
+ osEnforced.append(context.getString(R.string.auto_reboot_timeout, autoRebootValueString));
+
+ final byte userCount = securityStateExt.userCount();
+ final String userCountValueString;
+ if (userCount == SecurityStateExt.UNKNOWN_VALUE) {
+ userCountValueString = context.getString(R.string.unknown_value);
+ } else if (userCount == SecurityStateExt.INVALID_VALUE) {
+ userCountValueString = context.getString(R.string.invalid_value);
+ } else {
+ userCountValueString = String.valueOf(securityStateExt.userCount());
+ }
+ osEnforced.append(context.getString(R.string.user_count, userCountValueString));
+
return new VerificationResult(hasPersistentKey, teeEnforced.toString(), osEnforced.toString(), history.toString());
}
@@ -1231,10 +1306,14 @@ static VerificationResult verifySerialized(final Context context, final byte[] a
throw new GeneralSecurityException("invalid device administrator state");
}
+ final SecurityStateExt securityStateExt;
if (version >= 6) {
final short autoRebootMinutes = deserializer.getShort();
final byte portSecurityMode = deserializer.get();
final byte userCount = deserializer.get();
+ securityStateExt = new SecurityStateExt(autoRebootMinutes, portSecurityMode, userCount);
+ } else {
+ securityStateExt = SecurityStateExt.UNKNOWN;
}
final int signatureLength = deserializer.remaining();
@@ -1247,7 +1326,8 @@ static VerificationResult verifySerialized(final Context context, final byte[] a
final byte[] challenge = Arrays.copyOfRange(challengeMessage, 1 + RANDOM_TOKEN_LENGTH, 1 + RANDOM_TOKEN_LENGTH * 2);
return verify(context, fingerprint, challenge, deserializer.asReadOnlyBuffer(), signature,
certificates, userProfileSecure, accessibility, deviceAdmin, deviceAdminNonSystem,
- adbEnabled, addUsersWhenLocked, enrolledBiometrics, oemUnlockAllowed, systemUser);
+ adbEnabled, addUsersWhenLocked, enrolledBiometrics, oemUnlockAllowed, systemUser,
+ securityStateExt);
}
record AttestationResult(boolean pairing, byte[] serialized) {}
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index c29d7031..29a62976 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -86,6 +86,24 @@
Add users from lock screen: %s\n
OEM unlocking allowed: %s\n
Main user account: %s\n
+ Auto reboot timeout: %s\n
+ %d minutes
+ %d hours
+
+ USB-C port: %s\n
+ USB-C port and pogo pins: %s\n
+
+ Off
+ USB-C port off, pogo pins used only for charging
+ Charging-only
+ Charging-only when locked
+ Charging-only when locked, except before first unlock
+ On
+
+ User count: %s\n
+
+ Unknown
+ Invalid
\nAttestation history:\n\n
First verified: %s\n