1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.keystore.cts; 18 19 import android.content.Context; 20 import android.security.keystore.KeyGenParameterSpec; 21 import android.security.keystore.KeyInfo; 22 import android.security.keystore.KeyProperties; 23 import android.security.keystore.KeyProtection; 24 import android.test.MoreAsserts; 25 import junit.framework.Assert; 26 27 import java.io.ByteArrayOutputStream; 28 import java.io.IOException; 29 import java.io.InputStream; 30 import java.math.BigInteger; 31 import java.security.Key; 32 import java.security.KeyFactory; 33 import java.security.KeyPair; 34 import java.security.KeyStore; 35 import java.security.KeyStoreException; 36 import java.security.MessageDigest; 37 import java.security.NoSuchAlgorithmException; 38 import java.security.NoSuchProviderException; 39 import java.security.PrivateKey; 40 import java.security.PublicKey; 41 import java.security.UnrecoverableEntryException; 42 import java.security.cert.Certificate; 43 import java.security.cert.CertificateFactory; 44 import java.security.cert.X509Certificate; 45 import java.security.interfaces.ECKey; 46 import java.security.interfaces.ECPrivateKey; 47 import java.security.interfaces.ECPublicKey; 48 import java.security.interfaces.RSAKey; 49 import java.security.interfaces.RSAPrivateKey; 50 import java.security.interfaces.RSAPublicKey; 51 import java.security.spec.ECParameterSpec; 52 import java.security.spec.EllipticCurve; 53 import java.security.spec.InvalidKeySpecException; 54 import java.security.spec.PKCS8EncodedKeySpec; 55 import java.util.ArrayList; 56 import java.util.Collections; 57 import java.util.HashMap; 58 import java.util.List; 59 import java.util.Locale; 60 import java.util.Map; 61 62 import javax.crypto.SecretKey; 63 import javax.crypto.SecretKeyFactory; 64 import javax.crypto.spec.SecretKeySpec; 65 66 abstract class TestUtils extends Assert { 67 68 static final String EXPECTED_CRYPTO_OP_PROVIDER_NAME = "AndroidKeyStoreBCWorkaround"; 69 static final String EXPECTED_PROVIDER_NAME = "AndroidKeyStore"; 70 71 static final long DAY_IN_MILLIS = 1000 * 60 * 60 * 24; 72 73 TestUtils()74 private TestUtils() {} 75 76 /** 77 * Asserts the the key algorithm and algorithm-specific parameters of the two keys in the 78 * provided pair match. 79 */ assertKeyPairSelfConsistent(KeyPair keyPair)80 static void assertKeyPairSelfConsistent(KeyPair keyPair) { 81 assertKeyPairSelfConsistent(keyPair.getPublic(), keyPair.getPrivate()); 82 } 83 84 /** 85 * Asserts the the key algorithm and public algorithm-specific parameters of the two provided 86 * keys match. 87 */ assertKeyPairSelfConsistent(PublicKey publicKey, PrivateKey privateKey)88 static void assertKeyPairSelfConsistent(PublicKey publicKey, PrivateKey privateKey) { 89 assertNotNull(publicKey); 90 assertNotNull(privateKey); 91 assertEquals(publicKey.getAlgorithm(), privateKey.getAlgorithm()); 92 String keyAlgorithm = publicKey.getAlgorithm(); 93 if ("EC".equalsIgnoreCase(keyAlgorithm)) { 94 assertTrue("EC public key must be instanceof ECKey: " 95 + publicKey.getClass().getName(), 96 publicKey instanceof ECKey); 97 assertTrue("EC private key must be instanceof ECKey: " 98 + privateKey.getClass().getName(), 99 privateKey instanceof ECKey); 100 assertECParameterSpecEqualsIgnoreSeedIfNotPresent( 101 "Private key must have the same EC parameters as public key", 102 ((ECKey) publicKey).getParams(), ((ECKey) privateKey).getParams()); 103 } else if ("RSA".equalsIgnoreCase(keyAlgorithm)) { 104 assertTrue("RSA public key must be instance of RSAKey: " 105 + publicKey.getClass().getName(), 106 publicKey instanceof RSAKey); 107 assertTrue("RSA private key must be instance of RSAKey: " 108 + privateKey.getClass().getName(), 109 privateKey instanceof RSAKey); 110 assertEquals("Private and public key must have the same RSA modulus", 111 ((RSAKey) publicKey).getModulus(), ((RSAKey) privateKey).getModulus()); 112 } else { 113 fail("Unsuported key algorithm: " + keyAlgorithm); 114 } 115 } 116 getKeySizeBits(Key key)117 static int getKeySizeBits(Key key) { 118 if (key instanceof ECKey) { 119 return ((ECKey) key).getParams().getCurve().getField().getFieldSize(); 120 } else if (key instanceof RSAKey) { 121 return ((RSAKey) key).getModulus().bitLength(); 122 } else { 123 throw new IllegalArgumentException("Unsupported key type: " + key.getClass()); 124 } 125 } 126 assertKeySize(int expectedSizeBits, KeyPair keyPair)127 static void assertKeySize(int expectedSizeBits, KeyPair keyPair) { 128 assertEquals(expectedSizeBits, getKeySizeBits(keyPair.getPrivate())); 129 assertEquals(expectedSizeBits, getKeySizeBits(keyPair.getPublic())); 130 } 131 132 /** 133 * Asserts that the provided key pair is an Android Keystore key pair stored under the provided 134 * alias. 135 */ assertKeyStoreKeyPair(KeyStore keyStore, String alias, KeyPair keyPair)136 static void assertKeyStoreKeyPair(KeyStore keyStore, String alias, KeyPair keyPair) { 137 assertKeyMaterialExportable(keyPair.getPublic()); 138 assertKeyMaterialNotExportable(keyPair.getPrivate()); 139 assertTransparentKey(keyPair.getPublic()); 140 assertOpaqueKey(keyPair.getPrivate()); 141 142 KeyStore.Entry entry; 143 Certificate cert; 144 try { 145 entry = keyStore.getEntry(alias, null); 146 cert = keyStore.getCertificate(alias); 147 } catch (KeyStoreException | UnrecoverableEntryException | NoSuchAlgorithmException e) { 148 throw new RuntimeException("Failed to load entry: " + alias, e); 149 } 150 assertNotNull(entry); 151 152 assertTrue(entry instanceof KeyStore.PrivateKeyEntry); 153 KeyStore.PrivateKeyEntry privEntry = (KeyStore.PrivateKeyEntry) entry; 154 assertEquals(cert, privEntry.getCertificate()); 155 assertTrue("Certificate must be an X.509 certificate: " + cert.getClass(), 156 cert instanceof X509Certificate); 157 final X509Certificate x509Cert = (X509Certificate) cert; 158 159 PrivateKey keystorePrivateKey = privEntry.getPrivateKey(); 160 PublicKey keystorePublicKey = cert.getPublicKey(); 161 assertEquals(keyPair.getPrivate(), keystorePrivateKey); 162 assertEquals(keyPair.getPublic(), keystorePublicKey); 163 164 assertEquals( 165 "Public key used to sign certificate should have the same algorithm as in KeyPair", 166 keystorePublicKey.getAlgorithm(), x509Cert.getPublicKey().getAlgorithm()); 167 168 Certificate[] chain = privEntry.getCertificateChain(); 169 if (chain.length == 0) { 170 fail("Empty certificate chain"); 171 return; 172 } 173 assertEquals(cert, chain[0]); 174 } 175 176 assertKeyMaterialExportable(Key key)177 private static void assertKeyMaterialExportable(Key key) { 178 if (key instanceof PublicKey) { 179 assertEquals("X.509", key.getFormat()); 180 } else if (key instanceof PrivateKey) { 181 assertEquals("PKCS#8", key.getFormat()); 182 } else if (key instanceof SecretKey) { 183 assertEquals("RAW", key.getFormat()); 184 } else { 185 fail("Unsupported key type: " + key.getClass().getName()); 186 } 187 byte[] encodedForm = key.getEncoded(); 188 assertNotNull(encodedForm); 189 if (encodedForm.length == 0) { 190 fail("Empty encoded form"); 191 } 192 } 193 assertKeyMaterialNotExportable(Key key)194 private static void assertKeyMaterialNotExportable(Key key) { 195 assertEquals(null, key.getFormat()); 196 assertEquals(null, key.getEncoded()); 197 } 198 assertOpaqueKey(Key key)199 private static void assertOpaqueKey(Key key) { 200 assertFalse(key.getClass().getName() + " is a transparent key", isTransparentKey(key)); 201 } 202 assertTransparentKey(Key key)203 private static void assertTransparentKey(Key key) { 204 assertTrue(key.getClass().getName() + " is not a transparent key", isTransparentKey(key)); 205 } 206 isTransparentKey(Key key)207 private static boolean isTransparentKey(Key key) { 208 if (key instanceof PrivateKey) { 209 return (key instanceof ECPrivateKey) || (key instanceof RSAPrivateKey); 210 } else if (key instanceof PublicKey) { 211 return (key instanceof ECPublicKey) || (key instanceof RSAPublicKey); 212 } else if (key instanceof SecretKey) { 213 return (key instanceof SecretKeySpec); 214 } else { 215 throw new IllegalArgumentException("Unsupported key type: " + key.getClass().getName()); 216 } 217 } 218 assertECParameterSpecEqualsIgnoreSeedIfNotPresent( ECParameterSpec expected, ECParameterSpec actual)219 static void assertECParameterSpecEqualsIgnoreSeedIfNotPresent( 220 ECParameterSpec expected, ECParameterSpec actual) { 221 assertECParameterSpecEqualsIgnoreSeedIfNotPresent(null, expected, actual); 222 } 223 assertECParameterSpecEqualsIgnoreSeedIfNotPresent(String message, ECParameterSpec expected, ECParameterSpec actual)224 static void assertECParameterSpecEqualsIgnoreSeedIfNotPresent(String message, 225 ECParameterSpec expected, ECParameterSpec actual) { 226 EllipticCurve expectedCurve = expected.getCurve(); 227 EllipticCurve actualCurve = actual.getCurve(); 228 String msgPrefix = (message != null) ? message + ": " : ""; 229 assertEquals(msgPrefix + "curve field", expectedCurve.getField(), actualCurve.getField()); 230 assertEquals(msgPrefix + "curve A", expectedCurve.getA(), actualCurve.getA()); 231 assertEquals(msgPrefix + "curve B", expectedCurve.getB(), actualCurve.getB()); 232 assertEquals(msgPrefix + "order", expected.getOrder(), actual.getOrder()); 233 assertEquals(msgPrefix + "generator", 234 expected.getGenerator(), actual.getGenerator()); 235 assertEquals(msgPrefix + "cofactor", expected.getCofactor(), actual.getCofactor()); 236 237 // If present, the seed must be the same 238 byte[] expectedSeed = expectedCurve.getSeed(); 239 byte[] actualSeed = expectedCurve.getSeed(); 240 if ((expectedSeed != null) && (actualSeed != null)) { 241 MoreAsserts.assertEquals(expectedSeed, actualSeed); 242 } 243 } 244 getKeyInfo(Key key)245 static KeyInfo getKeyInfo(Key key) throws InvalidKeySpecException, NoSuchAlgorithmException, 246 NoSuchProviderException { 247 if ((key instanceof PrivateKey) || (key instanceof PublicKey)) { 248 return KeyFactory.getInstance(key.getAlgorithm(), "AndroidKeyStore") 249 .getKeySpec(key, KeyInfo.class); 250 } else if (key instanceof SecretKey) { 251 return (KeyInfo) SecretKeyFactory.getInstance(key.getAlgorithm(), "AndroidKeyStore") 252 .getKeySpec((SecretKey) key, KeyInfo.class); 253 } else { 254 throw new IllegalArgumentException("Unexpected key type: " + key.getClass()); 255 } 256 } 257 assertContentsInAnyOrder(Iterable<T> actual, T... expected)258 static <T> void assertContentsInAnyOrder(Iterable<T> actual, T... expected) { 259 assertContentsInAnyOrder(null, actual, expected); 260 } 261 assertContentsInAnyOrder(String message, Iterable<T> actual, T... expected)262 static <T> void assertContentsInAnyOrder(String message, Iterable<T> actual, T... expected) { 263 Map<T, Integer> actualFreq = getFrequencyTable(actual); 264 Map<T, Integer> expectedFreq = getFrequencyTable(expected); 265 if (actualFreq.equals(expectedFreq)) { 266 return; 267 } 268 269 Map<T, Integer> extraneousFreq = new HashMap<T, Integer>(); 270 for (Map.Entry<T, Integer> actualEntry : actualFreq.entrySet()) { 271 int actualCount = actualEntry.getValue(); 272 Integer expectedCount = expectedFreq.get(actualEntry.getKey()); 273 int diff = actualCount - ((expectedCount != null) ? expectedCount : 0); 274 if (diff > 0) { 275 extraneousFreq.put(actualEntry.getKey(), diff); 276 } 277 } 278 279 Map<T, Integer> missingFreq = new HashMap<T, Integer>(); 280 for (Map.Entry<T, Integer> expectedEntry : expectedFreq.entrySet()) { 281 int expectedCount = expectedEntry.getValue(); 282 Integer actualCount = actualFreq.get(expectedEntry.getKey()); 283 int diff = expectedCount - ((actualCount != null) ? actualCount : 0); 284 if (diff > 0) { 285 missingFreq.put(expectedEntry.getKey(), diff); 286 } 287 } 288 289 List<T> extraneous = frequencyTableToValues(extraneousFreq); 290 List<T> missing = frequencyTableToValues(missingFreq); 291 StringBuilder result = new StringBuilder(); 292 String delimiter = ""; 293 if (message != null) { 294 result.append(message).append("."); 295 delimiter = " "; 296 } 297 if (!missing.isEmpty()) { 298 result.append(delimiter).append("missing: " + missing); 299 delimiter = ", "; 300 } 301 if (!extraneous.isEmpty()) { 302 result.append(delimiter).append("extraneous: " + extraneous); 303 } 304 fail(result.toString()); 305 } 306 getFrequencyTable(Iterable<T> values)307 private static <T> Map<T, Integer> getFrequencyTable(Iterable<T> values) { 308 Map<T, Integer> result = new HashMap<T, Integer>(); 309 for (T value : values) { 310 Integer count = result.get(value); 311 if (count == null) { 312 count = 1; 313 } else { 314 count++; 315 } 316 result.put(value, count); 317 } 318 return result; 319 } 320 getFrequencyTable(T... values)321 private static <T> Map<T, Integer> getFrequencyTable(T... values) { 322 Map<T, Integer> result = new HashMap<T, Integer>(); 323 for (T value : values) { 324 Integer count = result.get(value); 325 if (count == null) { 326 count = 1; 327 } else { 328 count++; 329 } 330 result.put(value, count); 331 } 332 return result; 333 } 334 335 @SuppressWarnings("rawtypes") frequencyTableToValues(Map<T, Integer> table)336 private static <T> List<T> frequencyTableToValues(Map<T, Integer> table) { 337 if (table.isEmpty()) { 338 return Collections.emptyList(); 339 } 340 341 List<T> result = new ArrayList<T>(); 342 boolean comparableValues = true; 343 for (Map.Entry<T, Integer> entry : table.entrySet()) { 344 T value = entry.getKey(); 345 if (!(value instanceof Comparable)) { 346 comparableValues = false; 347 } 348 int frequency = entry.getValue(); 349 for (int i = 0; i < frequency; i++) { 350 result.add(value); 351 } 352 } 353 354 if (comparableValues) { 355 sortAssumingComparable(result); 356 } 357 return result; 358 } 359 360 @SuppressWarnings({"rawtypes", "unchecked"}) sortAssumingComparable(List<?> values)361 private static void sortAssumingComparable(List<?> values) { 362 Collections.sort((List<Comparable>)values); 363 } 364 toLowerCase(String... values)365 static String[] toLowerCase(String... values) { 366 if (values == null) { 367 return null; 368 } 369 String[] result = new String[values.length]; 370 for (int i = 0; i < values.length; i++) { 371 String value = values[i]; 372 result[i] = (value != null) ? value.toLowerCase() : null; 373 } 374 return result; 375 } 376 getRawResPrivateKey(Context context, int resId)377 static PrivateKey getRawResPrivateKey(Context context, int resId) throws Exception { 378 byte[] pkcs8EncodedForm; 379 try (InputStream in = context.getResources().openRawResource(resId)) { 380 pkcs8EncodedForm = drain(in); 381 } 382 PKCS8EncodedKeySpec privateKeySpec = new PKCS8EncodedKeySpec(pkcs8EncodedForm); 383 384 try { 385 return KeyFactory.getInstance("EC").generatePrivate(privateKeySpec); 386 } catch (InvalidKeySpecException e) { 387 try { 388 return KeyFactory.getInstance("RSA").generatePrivate(privateKeySpec); 389 } catch (InvalidKeySpecException e2) { 390 throw new InvalidKeySpecException("The key is neither EC nor RSA", e); 391 } 392 } 393 } 394 getRawResX509Certificate(Context context, int resId)395 static X509Certificate getRawResX509Certificate(Context context, int resId) throws Exception { 396 try (InputStream in = context.getResources().openRawResource(resId)) { 397 return (X509Certificate) CertificateFactory.getInstance("X.509") 398 .generateCertificate(in); 399 } 400 } 401 importIntoAndroidKeyStore( String alias, PrivateKey privateKey, Certificate certificate, KeyProtection keyProtection)402 static KeyPair importIntoAndroidKeyStore( 403 String alias, 404 PrivateKey privateKey, 405 Certificate certificate, 406 KeyProtection keyProtection) throws Exception { 407 KeyStore keyStore = KeyStore.getInstance("AndroidKeyStore"); 408 keyStore.load(null); 409 keyStore.setEntry(alias, 410 new KeyStore.PrivateKeyEntry(privateKey, new Certificate[] {certificate}), 411 keyProtection); 412 return new KeyPair( 413 keyStore.getCertificate(alias).getPublicKey(), 414 (PrivateKey) keyStore.getKey(alias, null)); 415 } 416 importIntoAndroidKeyStore( String alias, SecretKey key, KeyProtection keyProtection)417 static ImportedKey importIntoAndroidKeyStore( 418 String alias, 419 SecretKey key, 420 KeyProtection keyProtection) throws Exception { 421 KeyStore keyStore = KeyStore.getInstance("AndroidKeyStore"); 422 keyStore.load(null); 423 keyStore.setEntry(alias, 424 new KeyStore.SecretKeyEntry(key), 425 keyProtection); 426 return new ImportedKey(alias, key, (SecretKey) keyStore.getKey(alias, null)); 427 } 428 importIntoAndroidKeyStore( String alias, Context context, int privateResId, int certResId, KeyProtection params)429 static ImportedKey importIntoAndroidKeyStore( 430 String alias, Context context, int privateResId, int certResId, KeyProtection params) 431 throws Exception { 432 Certificate originalCert = TestUtils.getRawResX509Certificate(context, certResId); 433 PublicKey originalPublicKey = originalCert.getPublicKey(); 434 PrivateKey originalPrivateKey = TestUtils.getRawResPrivateKey(context, privateResId); 435 436 // Check that the domain parameters match between the private key and the public key. This 437 // is to catch accidental errors where a test provides the wrong resource ID as one of the 438 // parameters. 439 if (!originalPublicKey.getAlgorithm().equalsIgnoreCase(originalPrivateKey.getAlgorithm())) { 440 throw new IllegalArgumentException("Key algorithm mismatch." 441 + " Public: " + originalPublicKey.getAlgorithm() 442 + ", private: " + originalPrivateKey.getAlgorithm()); 443 } 444 assertKeyPairSelfConsistent(originalPublicKey, originalPrivateKey); 445 446 KeyPair keystoreBacked = TestUtils.importIntoAndroidKeyStore( 447 alias, originalPrivateKey, originalCert, 448 params); 449 assertKeyPairSelfConsistent(keystoreBacked); 450 assertKeyPairSelfConsistent(keystoreBacked.getPublic(), originalPrivateKey); 451 return new ImportedKey( 452 alias, 453 new KeyPair(originalCert.getPublicKey(), originalPrivateKey), 454 keystoreBacked); 455 } 456 drain(InputStream in)457 static byte[] drain(InputStream in) throws IOException { 458 ByteArrayOutputStream result = new ByteArrayOutputStream(); 459 byte[] buffer = new byte[16 * 1024]; 460 int chunkSize; 461 while ((chunkSize = in.read(buffer)) != -1) { 462 result.write(buffer, 0, chunkSize); 463 } 464 return result.toByteArray(); 465 } 466 buildUpon(KeyProtection params)467 static KeyProtection.Builder buildUpon(KeyProtection params) { 468 return buildUponInternal(params, null); 469 } 470 buildUpon(KeyProtection params, int newPurposes)471 static KeyProtection.Builder buildUpon(KeyProtection params, int newPurposes) { 472 return buildUponInternal(params, newPurposes); 473 } 474 buildUpon( KeyProtection.Builder builder)475 static KeyProtection.Builder buildUpon( 476 KeyProtection.Builder builder) { 477 return buildUponInternal(builder.build(), null); 478 } 479 buildUpon( KeyProtection.Builder builder, int newPurposes)480 static KeyProtection.Builder buildUpon( 481 KeyProtection.Builder builder, int newPurposes) { 482 return buildUponInternal(builder.build(), newPurposes); 483 } 484 buildUponInternal( KeyProtection spec, Integer newPurposes)485 private static KeyProtection.Builder buildUponInternal( 486 KeyProtection spec, Integer newPurposes) { 487 int purposes = (newPurposes == null) ? spec.getPurposes() : newPurposes; 488 KeyProtection.Builder result = new KeyProtection.Builder(purposes); 489 result.setBlockModes(spec.getBlockModes()); 490 if (spec.isDigestsSpecified()) { 491 result.setDigests(spec.getDigests()); 492 } 493 result.setEncryptionPaddings(spec.getEncryptionPaddings()); 494 result.setSignaturePaddings(spec.getSignaturePaddings()); 495 result.setKeyValidityStart(spec.getKeyValidityStart()); 496 result.setKeyValidityForOriginationEnd(spec.getKeyValidityForOriginationEnd()); 497 result.setKeyValidityForConsumptionEnd(spec.getKeyValidityForConsumptionEnd()); 498 result.setRandomizedEncryptionRequired(spec.isRandomizedEncryptionRequired()); 499 result.setUserAuthenticationRequired(spec.isUserAuthenticationRequired()); 500 result.setUserAuthenticationValidityDurationSeconds( 501 spec.getUserAuthenticationValidityDurationSeconds()); 502 return result; 503 } 504 buildUpon(KeyGenParameterSpec spec)505 static KeyGenParameterSpec.Builder buildUpon(KeyGenParameterSpec spec) { 506 return buildUponInternal(spec, null); 507 } 508 buildUpon(KeyGenParameterSpec spec, int newPurposes)509 static KeyGenParameterSpec.Builder buildUpon(KeyGenParameterSpec spec, int newPurposes) { 510 return buildUponInternal(spec, newPurposes); 511 } 512 buildUpon( KeyGenParameterSpec.Builder builder)513 static KeyGenParameterSpec.Builder buildUpon( 514 KeyGenParameterSpec.Builder builder) { 515 return buildUponInternal(builder.build(), null); 516 } 517 buildUpon( KeyGenParameterSpec.Builder builder, int newPurposes)518 static KeyGenParameterSpec.Builder buildUpon( 519 KeyGenParameterSpec.Builder builder, int newPurposes) { 520 return buildUponInternal(builder.build(), newPurposes); 521 } 522 buildUponInternal( KeyGenParameterSpec spec, Integer newPurposes)523 private static KeyGenParameterSpec.Builder buildUponInternal( 524 KeyGenParameterSpec spec, Integer newPurposes) { 525 int purposes = (newPurposes == null) ? spec.getPurposes() : newPurposes; 526 KeyGenParameterSpec.Builder result = 527 new KeyGenParameterSpec.Builder(spec.getKeystoreAlias(), purposes); 528 if (spec.getKeySize() >= 0) { 529 result.setKeySize(spec.getKeySize()); 530 } 531 if (spec.getAlgorithmParameterSpec() != null) { 532 result.setAlgorithmParameterSpec(spec.getAlgorithmParameterSpec()); 533 } 534 result.setCertificateNotBefore(spec.getCertificateNotBefore()); 535 result.setCertificateNotAfter(spec.getCertificateNotAfter()); 536 result.setCertificateSerialNumber(spec.getCertificateSerialNumber()); 537 result.setCertificateSubject(spec.getCertificateSubject()); 538 result.setBlockModes(spec.getBlockModes()); 539 if (spec.isDigestsSpecified()) { 540 result.setDigests(spec.getDigests()); 541 } 542 result.setEncryptionPaddings(spec.getEncryptionPaddings()); 543 result.setSignaturePaddings(spec.getSignaturePaddings()); 544 result.setKeyValidityStart(spec.getKeyValidityStart()); 545 result.setKeyValidityForOriginationEnd(spec.getKeyValidityForOriginationEnd()); 546 result.setKeyValidityForConsumptionEnd(spec.getKeyValidityForConsumptionEnd()); 547 result.setRandomizedEncryptionRequired(spec.isRandomizedEncryptionRequired()); 548 result.setUserAuthenticationRequired(spec.isUserAuthenticationRequired()); 549 result.setUserAuthenticationValidityDurationSeconds( 550 spec.getUserAuthenticationValidityDurationSeconds()); 551 return result; 552 } 553 getKeyPairForKeyAlgorithm(String keyAlgorithm, Iterable<KeyPair> keyPairs)554 static KeyPair getKeyPairForKeyAlgorithm(String keyAlgorithm, Iterable<KeyPair> keyPairs) { 555 for (KeyPair keyPair : keyPairs) { 556 if (keyAlgorithm.equalsIgnoreCase(keyPair.getPublic().getAlgorithm())) { 557 return keyPair; 558 } 559 } 560 throw new IllegalArgumentException("No KeyPair for key algorithm " + keyAlgorithm); 561 } 562 getKeyForKeyAlgorithm(String keyAlgorithm, Iterable<? extends Key> keys)563 static Key getKeyForKeyAlgorithm(String keyAlgorithm, Iterable<? extends Key> keys) { 564 for (Key key : keys) { 565 if (keyAlgorithm.equalsIgnoreCase(key.getAlgorithm())) { 566 return key; 567 } 568 } 569 throw new IllegalArgumentException("No Key for key algorithm " + keyAlgorithm); 570 } 571 generateLargeKatMsg(byte[] seed, int msgSizeBytes)572 static byte[] generateLargeKatMsg(byte[] seed, int msgSizeBytes) throws Exception { 573 byte[] result = new byte[msgSizeBytes]; 574 MessageDigest digest = MessageDigest.getInstance("SHA-512"); 575 int resultOffset = 0; 576 int resultRemaining = msgSizeBytes; 577 while (resultRemaining > 0) { 578 seed = digest.digest(seed); 579 int chunkSize = Math.min(seed.length, resultRemaining); 580 System.arraycopy(seed, 0, result, resultOffset, chunkSize); 581 resultOffset += chunkSize; 582 resultRemaining -= chunkSize; 583 } 584 return result; 585 } 586 leftPadWithZeroBytes(byte[] array, int length)587 static byte[] leftPadWithZeroBytes(byte[] array, int length) { 588 if (array.length >= length) { 589 return array; 590 } 591 byte[] result = new byte[length]; 592 System.arraycopy(array, 0, result, result.length - array.length, array.length); 593 return result; 594 } 595 contains(int[] array, int value)596 static boolean contains(int[] array, int value) { 597 for (int element : array) { 598 if (element == value) { 599 return true; 600 } 601 } 602 return false; 603 } 604 isHmacAlgorithm(String algorithm)605 static boolean isHmacAlgorithm(String algorithm) { 606 return algorithm.toUpperCase(Locale.US).startsWith("HMAC"); 607 } 608 getHmacAlgorithmDigest(String algorithm)609 static String getHmacAlgorithmDigest(String algorithm) { 610 String algorithmUpperCase = algorithm.toUpperCase(Locale.US); 611 if (!algorithmUpperCase.startsWith("HMAC")) { 612 return null; 613 } 614 String result = algorithmUpperCase.substring("HMAC".length()); 615 if (result.startsWith("SHA")) { 616 result = "SHA-" + result.substring("SHA".length()); 617 } 618 return result; 619 } 620 getCipherKeyAlgorithm(String transformation)621 static String getCipherKeyAlgorithm(String transformation) { 622 String transformationUpperCase = transformation.toUpperCase(Locale.US); 623 if (transformationUpperCase.startsWith("AES/")) { 624 return KeyProperties.KEY_ALGORITHM_AES; 625 } else if (transformationUpperCase.startsWith("RSA/")) { 626 return KeyProperties.KEY_ALGORITHM_RSA; 627 } else { 628 throw new IllegalArgumentException("Unsupported transformation: " + transformation); 629 } 630 } 631 isCipherSymmetric(String transformation)632 static boolean isCipherSymmetric(String transformation) { 633 String transformationUpperCase = transformation.toUpperCase(Locale.US); 634 if (transformationUpperCase.startsWith("AES/")) { 635 return true; 636 } else if (transformationUpperCase.startsWith("RSA/")) { 637 return false; 638 } else { 639 throw new IllegalArgumentException("Unsupported transformation: " + transformation); 640 } 641 } 642 getCipherDigest(String transformation)643 static String getCipherDigest(String transformation) { 644 String transformationUpperCase = transformation.toUpperCase(Locale.US); 645 if (transformationUpperCase.contains("/OAEP")) { 646 if (transformationUpperCase.endsWith("/OAEPPADDING")) { 647 return KeyProperties.DIGEST_SHA1; 648 } else if (transformationUpperCase.endsWith( 649 "/OAEPWITHSHA-1ANDMGF1PADDING")) { 650 return KeyProperties.DIGEST_SHA1; 651 } else if (transformationUpperCase.endsWith( 652 "/OAEPWITHSHA-224ANDMGF1PADDING")) { 653 return KeyProperties.DIGEST_SHA224; 654 } else if (transformationUpperCase.endsWith( 655 "/OAEPWITHSHA-256ANDMGF1PADDING")) { 656 return KeyProperties.DIGEST_SHA256; 657 } else if (transformationUpperCase.endsWith( 658 "/OAEPWITHSHA-384ANDMGF1PADDING")) { 659 return KeyProperties.DIGEST_SHA384; 660 } else if (transformationUpperCase.endsWith( 661 "/OAEPWITHSHA-512ANDMGF1PADDING")) { 662 return KeyProperties.DIGEST_SHA512; 663 } else { 664 throw new RuntimeException("Unsupported OAEP padding scheme: " 665 + transformation); 666 } 667 } else { 668 return null; 669 } 670 } 671 getCipherEncryptionPadding(String transformation)672 static String getCipherEncryptionPadding(String transformation) { 673 String transformationUpperCase = transformation.toUpperCase(Locale.US); 674 if (transformationUpperCase.endsWith("/NOPADDING")) { 675 return KeyProperties.ENCRYPTION_PADDING_NONE; 676 } else if (transformationUpperCase.endsWith("/PKCS7PADDING")) { 677 return KeyProperties.ENCRYPTION_PADDING_PKCS7; 678 } else if (transformationUpperCase.endsWith("/PKCS1PADDING")) { 679 return KeyProperties.ENCRYPTION_PADDING_RSA_PKCS1; 680 } else if (transformationUpperCase.split("/")[2].startsWith("OAEP")) { 681 return KeyProperties.ENCRYPTION_PADDING_RSA_OAEP; 682 } else { 683 throw new IllegalArgumentException("Unsupported transformation: " + transformation); 684 } 685 } 686 getCipherBlockMode(String transformation)687 static String getCipherBlockMode(String transformation) { 688 return transformation.split("/")[1].toUpperCase(Locale.US); 689 } 690 getSignatureAlgorithmDigest(String algorithm)691 static String getSignatureAlgorithmDigest(String algorithm) { 692 String algorithmUpperCase = algorithm.toUpperCase(Locale.US); 693 int withIndex = algorithmUpperCase.indexOf("WITH"); 694 if (withIndex == -1) { 695 throw new IllegalArgumentException("Unsupported algorithm: " + algorithm); 696 } 697 String digest = algorithmUpperCase.substring(0, withIndex); 698 if (digest.startsWith("SHA")) { 699 digest = "SHA-" + digest.substring("SHA".length()); 700 } 701 return digest; 702 } 703 getSignatureAlgorithmPadding(String algorithm)704 static String getSignatureAlgorithmPadding(String algorithm) { 705 String algorithmUpperCase = algorithm.toUpperCase(Locale.US); 706 if (algorithmUpperCase.endsWith("WITHECDSA")) { 707 return null; 708 } else if (algorithmUpperCase.endsWith("WITHRSA")) { 709 return KeyProperties.SIGNATURE_PADDING_RSA_PKCS1; 710 } else if (algorithmUpperCase.endsWith("WITHRSA/PSS")) { 711 return KeyProperties.SIGNATURE_PADDING_RSA_PSS; 712 } else { 713 throw new IllegalArgumentException("Unsupported algorithm: " + algorithm); 714 } 715 } 716 getSignatureAlgorithmKeyAlgorithm(String algorithm)717 static String getSignatureAlgorithmKeyAlgorithm(String algorithm) { 718 String algorithmUpperCase = algorithm.toUpperCase(Locale.US); 719 if (algorithmUpperCase.endsWith("WITHECDSA")) { 720 return KeyProperties.KEY_ALGORITHM_EC; 721 } else if ((algorithmUpperCase.endsWith("WITHRSA")) 722 || (algorithmUpperCase.endsWith("WITHRSA/PSS"))) { 723 return KeyProperties.KEY_ALGORITHM_RSA; 724 } else { 725 throw new IllegalArgumentException("Unsupported algorithm: " + algorithm); 726 } 727 } 728 isKeyLongEnoughForSignatureAlgorithm(String algorithm, Key key)729 static boolean isKeyLongEnoughForSignatureAlgorithm(String algorithm, Key key) { 730 String keyAlgorithm = key.getAlgorithm(); 731 if (KeyProperties.KEY_ALGORITHM_EC.equalsIgnoreCase(keyAlgorithm)) { 732 // No length restrictions for ECDSA 733 return true; 734 } else if (KeyProperties.KEY_ALGORITHM_RSA.equalsIgnoreCase(keyAlgorithm)) { 735 // No length restrictions for RSA 736 String digest = getSignatureAlgorithmDigest(algorithm); 737 int digestOutputSizeBits = getDigestOutputSizeBits(digest); 738 if (digestOutputSizeBits == -1) { 739 // No digesting -- assume the key is long enough for the message 740 return true; 741 } 742 String paddingScheme = getSignatureAlgorithmPadding(algorithm); 743 int paddingOverheadBytes; 744 if (KeyProperties.SIGNATURE_PADDING_RSA_PKCS1.equalsIgnoreCase(paddingScheme)) { 745 paddingOverheadBytes = 30; 746 } else if (KeyProperties.SIGNATURE_PADDING_RSA_PSS.equalsIgnoreCase(paddingScheme)) { 747 paddingOverheadBytes = 22; 748 } else { 749 throw new IllegalArgumentException( 750 "Unsupported signature padding scheme: " + paddingScheme); 751 } 752 int minKeySizeBytes = paddingOverheadBytes + (digestOutputSizeBits + 7) / 8 + 1; 753 int keySizeBytes = ((RSAKey) key).getModulus().bitLength() / 8; 754 return keySizeBytes >= minKeySizeBytes; 755 } else { 756 throw new IllegalArgumentException("Unsupported key algorithm: " + keyAlgorithm); 757 } 758 } 759 getMaxSupportedPlaintextInputSizeBytes(String transformation, Key key)760 static int getMaxSupportedPlaintextInputSizeBytes(String transformation, Key key) { 761 String keyAlgorithm = getCipherKeyAlgorithm(transformation); 762 if (KeyProperties.KEY_ALGORITHM_AES.equalsIgnoreCase(keyAlgorithm)) { 763 return Integer.MAX_VALUE; 764 } else if (KeyProperties.KEY_ALGORITHM_RSA.equalsIgnoreCase(keyAlgorithm)) { 765 String encryptionPadding = getCipherEncryptionPadding(transformation); 766 int modulusSizeBytes = (getKeySizeBits(key) + 7) / 8; 767 if (KeyProperties.ENCRYPTION_PADDING_NONE.equalsIgnoreCase(encryptionPadding)) { 768 return modulusSizeBytes - 1; 769 } else if (KeyProperties.ENCRYPTION_PADDING_RSA_PKCS1.equalsIgnoreCase( 770 encryptionPadding)) { 771 return modulusSizeBytes - 11; 772 } else if (KeyProperties.ENCRYPTION_PADDING_RSA_OAEP.equalsIgnoreCase( 773 encryptionPadding)) { 774 String digest = getCipherDigest(transformation); 775 int digestOutputSizeBytes = (getDigestOutputSizeBits(digest) + 7) / 8; 776 return modulusSizeBytes - 2 * digestOutputSizeBytes - 2; 777 } else { 778 throw new IllegalArgumentException( 779 "Unsupported encryption padding scheme: " + encryptionPadding); 780 } 781 } else { 782 throw new IllegalArgumentException("Unsupported key algorithm: " + keyAlgorithm); 783 } 784 } 785 getDigestOutputSizeBits(String digest)786 static int getDigestOutputSizeBits(String digest) { 787 if (KeyProperties.DIGEST_NONE.equals(digest)) { 788 return -1; 789 } else if (KeyProperties.DIGEST_MD5.equals(digest)) { 790 return 128; 791 } else if (KeyProperties.DIGEST_SHA1.equals(digest)) { 792 return 160; 793 } else if (KeyProperties.DIGEST_SHA224.equals(digest)) { 794 return 224; 795 } else if (KeyProperties.DIGEST_SHA256.equals(digest)) { 796 return 256; 797 } else if (KeyProperties.DIGEST_SHA384.equals(digest)) { 798 return 384; 799 } else if (KeyProperties.DIGEST_SHA512.equals(digest)) { 800 return 512; 801 } else { 802 throw new IllegalArgumentException("Unsupported digest: " + digest); 803 } 804 } 805 concat(byte[] arr1, byte[] arr2)806 static byte[] concat(byte[] arr1, byte[] arr2) { 807 return concat(arr1, 0, (arr1 != null) ? arr1.length : 0, 808 arr2, 0, (arr2 != null) ? arr2.length : 0); 809 } 810 concat(byte[] arr1, int offset1, int len1, byte[] arr2, int offset2, int len2)811 static byte[] concat(byte[] arr1, int offset1, int len1, 812 byte[] arr2, int offset2, int len2) { 813 if (len1 == 0) { 814 return subarray(arr2, offset2, len2); 815 } else if (len2 == 0) { 816 return subarray(arr1, offset1, len1); 817 } 818 byte[] result = new byte[len1 + len2]; 819 if (len1 > 0) { 820 System.arraycopy(arr1, offset1, result, 0, len1); 821 } 822 if (len2 > 0) { 823 System.arraycopy(arr2, offset2, result, len1, len2); 824 } 825 return result; 826 } 827 subarray(byte[] arr, int offset, int len)828 static byte[] subarray(byte[] arr, int offset, int len) { 829 if (len == 0) { 830 return EmptyArray.BYTE; 831 } 832 if ((offset == 0) && (arr.length == len)) { 833 return arr; 834 } 835 byte[] result = new byte[len]; 836 System.arraycopy(arr, offset, result, 0, len); 837 return result; 838 } 839 getMinimalWorkingImportParametersForSigningingWith( String signatureAlgorithm)840 static KeyProtection getMinimalWorkingImportParametersForSigningingWith( 841 String signatureAlgorithm) { 842 String keyAlgorithm = getSignatureAlgorithmKeyAlgorithm(signatureAlgorithm); 843 String digest = getSignatureAlgorithmDigest(signatureAlgorithm); 844 if (KeyProperties.KEY_ALGORITHM_EC.equalsIgnoreCase(keyAlgorithm)) { 845 return new KeyProtection.Builder(KeyProperties.PURPOSE_SIGN) 846 .setDigests(digest) 847 .build(); 848 } else if (KeyProperties.KEY_ALGORITHM_RSA.equalsIgnoreCase(keyAlgorithm)) { 849 String padding = getSignatureAlgorithmPadding(signatureAlgorithm); 850 return new KeyProtection.Builder(KeyProperties.PURPOSE_SIGN) 851 .setDigests(digest) 852 .setSignaturePaddings(padding) 853 .build(); 854 } else { 855 throw new IllegalArgumentException( 856 "Unsupported signature algorithm: " + signatureAlgorithm); 857 } 858 } 859 getMinimalWorkingImportParametersForCipheringWith( String transformation, int purposes)860 static KeyProtection getMinimalWorkingImportParametersForCipheringWith( 861 String transformation, int purposes) { 862 return getMinimalWorkingImportParametersForCipheringWith(transformation, purposes, false); 863 } 864 getMinimalWorkingImportParametersForCipheringWith( String transformation, int purposes, boolean ivProvidedWhenEncrypting)865 static KeyProtection getMinimalWorkingImportParametersForCipheringWith( 866 String transformation, int purposes, boolean ivProvidedWhenEncrypting) { 867 String keyAlgorithm = TestUtils.getCipherKeyAlgorithm(transformation); 868 if (KeyProperties.KEY_ALGORITHM_AES.equalsIgnoreCase(keyAlgorithm)) { 869 String encryptionPadding = TestUtils.getCipherEncryptionPadding(transformation); 870 String blockMode = TestUtils.getCipherBlockMode(transformation); 871 boolean randomizedEncryptionRequired = true; 872 if (KeyProperties.BLOCK_MODE_ECB.equalsIgnoreCase(blockMode)) { 873 randomizedEncryptionRequired = false; 874 } else if ((ivProvidedWhenEncrypting) 875 && ((purposes & KeyProperties.PURPOSE_ENCRYPT) != 0)) { 876 randomizedEncryptionRequired = false; 877 } 878 return new KeyProtection.Builder( 879 purposes) 880 .setBlockModes(blockMode) 881 .setEncryptionPaddings(encryptionPadding) 882 .setRandomizedEncryptionRequired(randomizedEncryptionRequired) 883 .build(); 884 } else if (KeyProperties.KEY_ALGORITHM_RSA.equalsIgnoreCase(keyAlgorithm)) { 885 String digest = TestUtils.getCipherDigest(transformation); 886 String encryptionPadding = TestUtils.getCipherEncryptionPadding(transformation); 887 boolean randomizedEncryptionRequired = 888 !KeyProperties.ENCRYPTION_PADDING_NONE.equalsIgnoreCase(encryptionPadding); 889 return new KeyProtection.Builder( 890 purposes) 891 .setDigests((digest != null) ? new String[] {digest} : EmptyArray.STRING) 892 .setEncryptionPaddings(encryptionPadding) 893 .setRandomizedEncryptionRequired(randomizedEncryptionRequired) 894 .build(); 895 } else { 896 throw new IllegalArgumentException("Unsupported key algorithm: " + keyAlgorithm); 897 } 898 } 899 getBigIntegerMagnitudeBytes(BigInteger value)900 static byte[] getBigIntegerMagnitudeBytes(BigInteger value) { 901 return removeLeadingZeroByteIfPresent(value.toByteArray()); 902 } 903 removeLeadingZeroByteIfPresent(byte[] value)904 private static byte[] removeLeadingZeroByteIfPresent(byte[] value) { 905 if ((value.length < 1) || (value[0] != 0)) { 906 return value; 907 } 908 return TestUtils.subarray(value, 1, value.length - 1); 909 } 910 } 911