From 4977e6544758447dff3e5ff753f6efb06c70215f Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Sat, 16 Apr 2022 00:22:56 +0000 Subject: [PATCH 1/9] 1. Removed attestKey function and combined inside generateKey itself. 2. Made some memory optimizations. --- .../android/javacard/keymaster/KMEncoder.java | 60 +-- .../keymaster/KMKeyCharacteristics.java | 10 +- .../javacard/keymaster/KMKeymasterApplet.java | 410 +++++++++--------- .../javacard/keymaster/KMRepository.java | 64 ++- HAL/JavacardKeyMintDevice.cpp | 79 +--- 5 files changed, 284 insertions(+), 339 deletions(-) diff --git a/Applet/src/com/android/javacard/keymaster/KMEncoder.java b/Applet/src/com/android/javacard/keymaster/KMEncoder.java index e47d799d..4cfc9be3 100644 --- a/Applet/src/com/android/javacard/keymaster/KMEncoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMEncoder.java @@ -77,6 +77,20 @@ private void encode(short obj) { push(obj); } + // Use this function, when the max len + public short encode(short object, byte[] buffer, short startOff, short maxLength) { + scratchBuf[STACK_PTR_OFFSET] = 0; + bufferRef[0] = buffer; + scratchBuf[START_OFFSET] = startOff; + if ((short) (startOff + maxLength) > buffer.length) { + ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); + } + scratchBuf[LEN_OFFSET] = (short) (startOff + maxLength); + push(object); + encode(); + return (short) (scratchBuf[START_OFFSET] - startOff); + } + public short encode(short object, byte[] buffer, short startOff) { scratchBuf[STACK_PTR_OFFSET] = 0; bufferRef[0] = buffer; @@ -87,60 +101,32 @@ public short encode(short object, byte[] buffer, short startOff) { } else { scratchBuf[LEN_OFFSET] = (short) buffer.length; } - //this.length = (short)(startOff + length); push(object); encode(); return (short) (scratchBuf[START_OFFSET] - startOff); } - // array{KMError.OK,Array{KMByteBlobs}} - public void encodeCertChain(byte[] buffer, short offset, short length, short errInt32Ptr) { - bufferRef[0] = buffer; - scratchBuf[START_OFFSET] = offset; - scratchBuf[LEN_OFFSET] = (short) (offset + 1); - //Total length is ArrayHeader + [UIntHeader + length(errInt32Ptr)] - scratchBuf[LEN_OFFSET] += (short) (1 + getEncodedIntegerLength(errInt32Ptr)); - - writeMajorTypeWithLength(ARRAY_TYPE, (short) 2); // Array of 2 elements - encodeUnsignedInteger(errInt32Ptr); - } - //array{KMError.OK,Array{KMByteBlobs}} - public short encodeCert(byte[] certBuffer, short bufferStart, short certStart, short certLength, short errInt32Ptr) { + public short encodeCert(byte[] certBuffer, short bufferStart, short certStart, short certLength) { + if (bufferStart > certStart) { + ISOException.throwIt(ISO7816.SW_DATA_INVALID); + } bufferRef[0] = certBuffer; scratchBuf[START_OFFSET] = certStart; scratchBuf[LEN_OFFSET] = (short) (certStart + 1); - //Array header - 2 elements i.e. 1 byte - scratchBuf[START_OFFSET]--; - // errInt32Ptr - PowerResetStatus + ErrorCode - 4 bytes - // Integer header - 1 byte - scratchBuf[START_OFFSET] -= getEncodedIntegerLength(errInt32Ptr); - //Array header - 2 elements i.e. 1 byte + // Byte Header + cert length + scratchBuf[START_OFFSET] -= getEncodedBytesLength(certLength); + //Array header - 1 elements i.e. 1 byte scratchBuf[START_OFFSET]--; - // Cert Byte blob - typically 2 bytes length i.e. 3 bytes header - scratchBuf[START_OFFSET] -= 2; - if (certLength >= SHORT_PAYLOAD) { - scratchBuf[START_OFFSET]--; - } if (scratchBuf[START_OFFSET] < bufferStart) { ISOException.throwIt(ISO7816.SW_WRONG_LENGTH); } bufferStart = scratchBuf[START_OFFSET]; - writeMajorTypeWithLength(ARRAY_TYPE, (short) 2); // Array of 2 elements - encodeUnsignedInteger(errInt32Ptr); //PowerResetStatus + ErrorCode - writeMajorTypeWithLength(ARRAY_TYPE, (short) 1); // Array of 1 element + writeMajorTypeWithLength(ARRAY_TYPE, (short) 1); // Array of 1 elements writeMajorTypeWithLength(BYTES_TYPE, certLength); // Cert Byte Blob of length return bufferStart; } - public short encodeError(short errInt32Ptr, byte[] buffer, short startOff, short length) { - bufferRef[0] = buffer; - scratchBuf[START_OFFSET] = startOff; - scratchBuf[LEN_OFFSET] = (short) (startOff + length + 1); - encodeUnsignedInteger(errInt32Ptr); - return (short) (scratchBuf[START_OFFSET] - startOff); - } - private void encode() { while (scratchBuf[STACK_PTR_OFFSET] > 0) { short exp = pop(); @@ -637,7 +623,7 @@ private short getEncodedArrayLen(short obj) { return len; } - private short getEncodedBytesLength(short len) { + public short getEncodedBytesLength(short len) { short ret = 0; if (len < KMEncoder.UINT8_LENGTH && len >= 0) { ret = 1; diff --git a/Applet/src/com/android/javacard/keymaster/KMKeyCharacteristics.java b/Applet/src/com/android/javacard/keymaster/KMKeyCharacteristics.java index be073140..654dae3d 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeyCharacteristics.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeyCharacteristics.java @@ -37,15 +37,13 @@ private KMKeyCharacteristics() { } public static short exp() { - short sb = KMKeyParameters.exp(); - short tee = KMKeyParameters.exp(); - short keystore = KMKeyParameters.exp(); + short keyParamExp = KMKeyParameters.exp(); short arrPtr = KMArray.instance((short) 3); KMArray arr = KMArray.cast(arrPtr); - arr.add(STRONGBOX_ENFORCED, sb); - arr.add(TEE_ENFORCED, tee); - arr.add(KEYSTORE_ENFORCED, keystore); + arr.add(STRONGBOX_ENFORCED, keyParamExp); + arr.add(TEE_ENFORCED, keyParamExp); + arr.add(KEYSTORE_ENFORCED, keyParamExp); return instance(arrPtr); } diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index e78e7bc6..cf7212a2 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -146,8 +146,9 @@ public class KMKeymasterApplet extends Applet implements AppletEvent, ExtendedLe public static final byte INS_UPDATE_CHALLENGE_CMD = KEYMINT_CMD_APDU_START + 32; //0x40 public static final byte INS_FINISH_SEND_DATA_CMD = KEYMINT_CMD_APDU_START + 33; //0x41 public static final byte INS_GET_RESPONSE_CMD = KEYMINT_CMD_APDU_START + 34; //0x42 - private static final byte INS_GET_HEAP_PROFILE_DATA = KEYMINT_CMD_APDU_START + 35; //0x43 - private static final byte KEYMINT_CMD_APDU_END = KEYMINT_CMD_APDU_START + 36; //0x44 + private static final byte KEYMINT_CMD_APDU_END = KEYMINT_CMD_APDU_START + 35; //0x43 + // Enable this when doing heap profiling. + //private static final byte INS_GET_HEAP_PROFILE_DATA = KEYMINT_CMD_APDU_START + 36; //0x44 private static final byte INS_END_KM_CMD = 0x7F; @@ -210,7 +211,9 @@ public class KMKeymasterApplet extends Applet implements AppletEvent, ExtendedLe public static final byte AES_GCM_NONCE_LENGTH = 12; // ComputeHMAC constants private static final short HMAC_SHARED_PARAM_MAX_SIZE = 64; - protected static final short MAX_CERT_SIZE = 2048; + protected static final short MAX_CERT_SIZE = 2500; + protected static final short MAX_KEY_CHARS_SIZE = 512; + protected static final short MAX_KEYBLOB_SIZE = 1024; // KEYBLOB_CURRENT_VERSION goes into KeyBlob and will affect all // the KeyBlobs if it is changed. please increment this // version number whenever you change anything related to @@ -415,9 +418,6 @@ public void process(APDU apdu) { case INS_GENERATE_KEY_CMD: processGenerateKey(apdu); break; - case INS_ATTEST_KEY_CMD: - processAttestKeyCmd(apdu); - break; case INS_IMPORT_KEY_CMD: processImportKeyCmd(apdu); break; @@ -491,9 +491,10 @@ public void process(APDU apdu) { case INS_GET_RKP_HARDWARE_INFO: rkp.process(apduIns, apdu); break; - case INS_GET_HEAP_PROFILE_DATA: + // Enable this when doing heap profiling. + /*case INS_GET_HEAP_PROFILE_DATA: processGetHeapProfileData(apdu); - break; + break;*/ default: ISOException.throwIt(ISO7816.SW_INS_NOT_SUPPORTED); } @@ -607,7 +608,8 @@ protected void resetData() { */ public static void sendOutgoing(APDU apdu, short resp) { //TODO handle the extended buffer stuff. We can reuse this. - short usedHeap = repository.getHeapIndex(); + // Enable this when doing heap profiling. + //short usedHeap = repository.getHeapIndex(); short bufferStartOffset = repository.allocAvailableMemory(); byte[] buffer = repository.getHeap(); // TODO we can change the following to incremental send. @@ -616,7 +618,64 @@ public static void sendOutgoing(APDU apdu, short resp) { .getHeap().length)) { ISOException.throwIt(ISO7816.SW_WRONG_LENGTH); } - repository.updateHeapProfileData((short)(usedHeap + bufferLength)); + // Enable this when doing heap profiling. + //repository.updateHeapProfileData((short) (bufferLength + usedHeap)); + // Send data + apdu.setOutgoing(); + apdu.setOutgoingLength(bufferLength); + apdu.sendBytesLong(buffer, bufferStartOffset, bufferLength); + } + + public void sendOutgoing(APDU apdu, KMAttestationCert cert, short certStart, short keyblob, + short keyChars) { + // This is the special case where the output is encoded manually without using + // the encoder algorithm. Encoder creates a duplicate copy for each KMType Object. + // The output of the generateKey, importKey and importWrappedKey commands are huge so + // by manually encoding we can avoid duplicate copies. + // The output data is directly written to the end of heap in the below order + // output = [ + // errorCode : uint // ErrorCode + // keyBlob : bstr // KeyBlob. + // keyChars + // certifcate + // ] + // certificate = [ + // x509_cert : bstr // X509 certificate + // ] + // keyChars = { // Map + // } + byte[] buffer = repository.getHeap(); + + if (cert == null) { + // This happens for Symmetric keys. + short bufferStart = repository.allocReclaimableMemory((short) 1); + buffer[bufferStart] = (byte) 0x80; // Array of 0 length. + } else { + // Encode the certificate into cbor data at the end of the heap + // certData = [ + // x509_cert : bstr // X509 certificate + // ] + short bufferStart = encoder.encodeCert( + repository.getHeap(), + certStart, + cert.getCertStart(), + cert.getCertLength()); + // reclaim the unused memory in the certificate. + repository.reclaimMemory((short) (bufferStart - certStart)); + } + + // Encode KeyCharacteristics at the end of heap just before data[CERTIFICATE] + encodeKeyCharacteristics(keyChars); + // and encode it to the end of the buffer before KEY_CHARACTERISTICS + encodeKeyBlob(keyblob); + // Write Error code before data[KEY_BLOB] + short bufferStartOffset = repository.allocReclaimableMemory((short) 1); + buffer[bufferStartOffset] = 0x00; + // Write Array header before ErrorCode. + bufferStartOffset = repository.allocReclaimableMemory((short) 1); + buffer[bufferStartOffset] = (byte) 0x84; + + short bufferLength = (short) (KMRepository.HEAP_SIZE - bufferStartOffset); // Send data apdu.setOutgoing(); apdu.setOutgoingLength(bufferLength); @@ -1027,6 +1086,10 @@ private void processUpgradeKeyCmd(APDU apdu) { upgradeKeyBlobKeyCharacteristics(data[HW_PARAMETERS], scratchPad); // create new key blob with current os version etc. createEncryptedKeyBlob(scratchPad); + // allocate reclaimable memory. + short buffer = repository.alloc(MAX_KEYBLOB_SIZE); + data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer); + data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, data[KEY_BLOB]); } else { data[KEY_BLOB] = KMByteBlob.instance((short) 0); } @@ -1231,9 +1294,7 @@ private void processFinishImportWrappedKeyCmd(APDU apdu){ } private KMAttestationCert makeCommonCert(byte[] scratchPad) { - // The Algorithm value can be read from either data[HW_PARAMETERS] or data[KEY_PARAMETERS] - // as the values will be same, and they are cryptographically bound. - short alg = KMKeyParameters.findTag(KMType.ENUM_TAG, KMType.ALGORITHM, data[HW_PARAMETERS]); + short alg = KMKeyParameters.findTag(KMType.ENUM_TAG, KMType.ALGORITHM, data[KEY_PARAMETERS]); boolean rsaCert = KMEnumTag.cast(alg).getValue() == KMType.RSA; KMAttestationCert cert = KMAttestationCertImpl.instance(rsaCert, seProvider); @@ -1290,26 +1351,33 @@ private KMAttestationCert makeCommonCert(byte[] scratchPad) { return cert; } + private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyParam, - short attChallenge, short issuer, short hwParameters, byte[] scratchPad) { + short attChallenge, short issuer, byte[] scratchPad) { KMAttestationCert cert = makeCommonCert(scratchPad); - // App Id and App Data, + // Read App Id and App Data. short appId = getApplicationId(attKeyParam); short appData = getApplicationData(attKeyParam); + // Take backup of the required global variables KEY_BLOB, PUB_KEY, SECRET, KEY_CHAR + // and HW_PARAMS before they get overridden by isKeyUpgradeRequired() function. short origBlob = data[KEY_BLOB]; short pubKey = data[PUB_KEY]; short privKey = data[SECRET]; - // Check if key requires upgrade. The KeyBlob is parsed inside isKeyUpgradeRequired - // function itself. + short hwParams = data[HW_PARAMETERS]; + short keyChars = data[KEY_CHARACTERISTICS]; + // Check if key requires upgrade for attestKeyBlob. The KeyBlob is parsed inside + // isKeyUpgradeRequired function itself. if (isKeyUpgradeRequired(attKeyBlob, appId, appData, scratchPad)) { KMException.throwIt(KMError.KEY_REQUIRES_UPGRADE); } + // Get the private key of the attest key. short attestationKeySecret = KMArray.cast(data[KEY_BLOB]).get(KEY_BLOB_SECRET); + // Get the KeyCharacteristics and SB param of the attest key short attestKeyCharacteristics = KMArray.cast(data[KEY_BLOB]).get(KEY_BLOB_PARAMS); short attestKeySbParams = KMKeyCharacteristics.cast(attestKeyCharacteristics).getStrongboxEnforced(); - short attKeyPurpose = KMKeyParameters.findTag(KMType.ENUM_ARRAY_TAG, KMType.PURPOSE, attestKeySbParams); // If the attest key's purpose is not "attest key" then error. + short attKeyPurpose = KMKeyParameters.findTag(KMType.ENUM_ARRAY_TAG, KMType.PURPOSE, attestKeySbParams); if (!KMEnumArrayTag.cast(attKeyPurpose).contains(KMType.ATTEST_KEY)) { KMException.throwIt(KMError.INCOMPATIBLE_PURPOSE); } @@ -1328,9 +1396,16 @@ private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyPara } cert.attestationChallenge(attChallenge); cert.issuer(issuer); + + // Restore back the global variables. data[PUB_KEY] = pubKey; data[SECRET] = privKey; data[KEY_BLOB] = origBlob; + data[HW_PARAMETERS] = hwParams; + data[KEY_CHARACTERISTICS] = keyChars; + data[SW_PARAMETERS] = KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).getKeystoreEnforced(); + data[TEE_PARAMETERS] = KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).getTeeEnforced(); + data[SB_PARAMETERS] = KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).getStrongboxEnforced(); cert.publicKey(data[PUB_KEY]); // Save attestation application id - must be present. @@ -1348,26 +1423,21 @@ private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyPara addAttestationIds(cert, scratchPad); // Add Tags - addTags(hwParameters, true, cert); - short swParams = KMKeyParameters.makeKeystoreEnforced(data[KEY_PARAMETERS], scratchPad); - addTags(swParams, false, cert); + addTags(data[HW_PARAMETERS], true, cert); + addTags(data[SW_PARAMETERS], false, cert); // Add Device Boot locked status cert.deviceLocked(kmDataStore.isDeviceBootLocked()); // VB data cert.verifiedBootHash(getVerifiedBootHash(scratchPad)); cert.verifiedBootKey(getBootKey(scratchPad)); cert.verifiedBootState((byte) kmDataStore.getBootState()); - data[SECRET] = privKey; - data[KEY_BLOB] = origBlob; return cert; } private KMAttestationCert makeSelfSignedCert(short attPrivKey, short attPubKey, short mode, byte[] scratchPad) { KMAttestationCert cert = makeCommonCert(scratchPad); - // The Algorithm value can be read from either data[HW_PARAMETERS] or data[KEY_PARAMETERS] - // as the values will be same, and they are cryptographically bound. - short alg = KMEnumTag.getValue(KMType.ALGORITHM, data[HW_PARAMETERS]); + short alg = KMEnumTag.getValue(KMType.ALGORITHM, data[KEY_PARAMETERS]); short subject = KMKeyParameters.findTag(KMType.BYTES_TAG, KMType.CERTIFICATE_SUBJECT_NAME, data[KEY_PARAMETERS]); // If no subject name is specified then use the default subject name. @@ -2892,12 +2962,15 @@ private boolean validateHwToken(short hwToken, byte[] scratchPad) { } private short importKeyCmd(APDU apdu){ - short cmd = KMArray.instance((short) 3); + short cmd = KMArray.instance((short) 6); // Arguments short params = KMKeyParameters.expAny(); KMArray.cast(cmd).add((short) 0, params); KMArray.cast(cmd).add((short) 1, KMEnum.instance(KMType.KEY_FORMAT)); KMArray.cast(cmd).add((short) 2, KMByteBlob.exp()); + KMArray.cast(cmd).add((short) 3, KMByteBlob.exp()); //attest key + KMArray.cast(cmd).add((short) 4, params); //attest key params + KMArray.cast(cmd).add((short) 5, KMByteBlob.exp()); //issuer return receiveIncoming(apdu, cmd); } @@ -2908,6 +2981,9 @@ private void processImportKeyCmd(APDU apdu) { data[KEY_PARAMETERS] = KMArray.cast(cmd).get((short) 0); short keyFmt = KMArray.cast(cmd).get((short) 1); data[IMPORTED_KEY_BLOB] = KMArray.cast(cmd).get((short) 2); + data[ATTEST_KEY_BLOB] = KMArray.cast(cmd).get((short) 3); + data[ATTEST_KEY_PARAMS] = KMArray.cast(cmd).get((short) 4); + data[ATTEST_KEY_ISSUER] = KMArray.cast(cmd).get((short) 5); keyFmt = KMEnum.cast(keyFmt).getVal(); data[CERTIFICATE] = KMArray.instance((short)0); //by default the cert is empty. @@ -2973,21 +3049,14 @@ private void importKey(APDU apdu, short keyFmt, byte[] scratchPad) { break; } makeKeyCharacteristics( scratchPad); + KMAttestationCert cert = generateAttestation(data[ATTEST_KEY_BLOB], data[ATTEST_KEY_PARAMS],scratchPad); createEncryptedKeyBlob(scratchPad); - // MAC the KeyParameters. - short keyParamsMac = macKeyParams(keyParams, scratchPad); // Remove custom tags from key characteristics short teeParams = KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).getTeeEnforced(); if(teeParams != KMType.INVALID_VALUE) { KMKeyParameters.cast(teeParams).deleteCustomTags(); } - // prepare the response - short resp = KMArray.instance((short) 4); - KMArray.cast(resp).add((short) 0, KMInteger.uint_16(KMError.OK)); - KMArray.cast(resp).add((short) 1, data[KEY_BLOB]); - KMArray.cast(resp).add((short) 2, data[KEY_CHARACTERISTICS]); - KMArray.cast(resp).add((short) 3, keyParamsMac); - sendOutgoing(apdu, resp); + sendOutgoing(apdu, cert, data[CERTIFICATE], data[KEY_BLOB], data[KEY_CHARACTERISTICS]); } private void importECKeys(byte[] scratchPad) { @@ -3363,9 +3432,13 @@ protected void setVendorPatchLevel(short patch){ private short generateKeyCmd(APDU apdu){ short params = KMKeyParameters.expAny(); + short blob = KMByteBlob.exp(); // Array of expected arguments - short cmd = KMArray.instance((short) 1); + short cmd = KMArray.instance((short) 4); KMArray.cast(cmd).add((short) 0, params); //key params + KMArray.cast(cmd).add((short) 1, blob); //attest key + KMArray.cast(cmd).add((short) 2, params); //attest key params + KMArray.cast(cmd).add((short) 3, blob); //issuer return receiveIncoming(apdu, cmd); } @@ -3375,25 +3448,27 @@ private void processGenerateKey(APDU apdu) { // Re-purpose the apdu buffer as scratch pad. byte[] scratchPad = apdu.getBuffer(); data[KEY_PARAMETERS] = KMArray.cast(cmd).get((short) 0); + data[ATTEST_KEY_BLOB] = KMArray.cast(cmd).get((short) 1); + data[ATTEST_KEY_PARAMS] = KMArray.cast(cmd).get((short) 2); + data[ATTEST_KEY_ISSUER] = KMArray.cast(cmd).get((short) 3); + data[CERTIFICATE] = KMType.INVALID_VALUE; // by default the cert is empty. // ROLLBACK_RESISTANCE not supported. - KMTag.assertAbsence(data[KEY_PARAMETERS], KMType.BOOL_TAG,KMType.ROLLBACK_RESISTANCE, - KMError.ROLLBACK_RESISTANCE_UNAVAILABLE); + KMTag.assertAbsence(data[KEY_PARAMETERS], KMType.BOOL_TAG,KMType.ROLLBACK_RESISTANCE, KMError.ROLLBACK_RESISTANCE_UNAVAILABLE); // As per specification Early boot keys may be created after early boot ended. // Algorithm must be present - KMTag.assertPresence(data[KEY_PARAMETERS], KMType.ENUM_TAG, KMType.ALGORITHM, - KMError.INVALID_ARGUMENT); + KMTag.assertPresence(data[KEY_PARAMETERS], KMType.ENUM_TAG, KMType.ALGORITHM, KMError.INVALID_ARGUMENT); //Check if the tags are supported. if (KMKeyParameters.hasUnsupportedTags(data[KEY_PARAMETERS])) { KMException.throwIt(KMError.UNSUPPORTED_TAG); } short attKeyPurpose = - KMKeyParameters.findTag(KMType.ENUM_ARRAY_TAG, KMType.PURPOSE, data[KEY_PARAMETERS]); + KMKeyParameters.findTag(KMType.ENUM_ARRAY_TAG, KMType.PURPOSE, data[KEY_PARAMETERS]); // ATTEST_KEY cannot be combined with any other purpose. if (attKeyPurpose != KMType.INVALID_VALUE - && KMEnumArrayTag.cast(attKeyPurpose).contains(KMType.ATTEST_KEY) - && KMEnumArrayTag.cast(attKeyPurpose).length() > 1) { + && KMEnumArrayTag.cast(attKeyPurpose).contains(KMType.ATTEST_KEY) + && KMEnumArrayTag.cast(attKeyPurpose).length() > 1) { KMException.throwIt(KMError.INCOMPATIBLE_PURPOSE); } short alg = KMEnumTag.getValue(KMType.ALGORITHM, data[KEY_PARAMETERS]); @@ -3421,21 +3496,15 @@ private void processGenerateKey(APDU apdu) { // create key blob and associated attestation. data[ORIGIN] = KMType.GENERATED; makeKeyCharacteristics(scratchPad); + // construct the certificate and place the encoded data in data[CERTIFICATE] + KMAttestationCert cert = generateAttestation(data[ATTEST_KEY_BLOB], data[ATTEST_KEY_PARAMS], scratchPad); createEncryptedKeyBlob(scratchPad); - // MAC the KeyParameters. - short keyParamsMac = macKeyParams(data[KEY_PARAMETERS], scratchPad); // Remove custom tags from key characteristics short teeParams = KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).getTeeEnforced(); if(teeParams != KMType.INVALID_VALUE) { KMKeyParameters.cast(teeParams).deleteCustomTags(); } - // prepare the response - short resp = KMArray.instance((short) 4); - KMArray.cast(resp).add((short) 0, KMInteger.uint_16(KMError.OK)); - KMArray.cast(resp).add((short) 1, data[KEY_BLOB]); - KMArray.cast(resp).add((short) 2, data[KEY_CHARACTERISTICS]); - KMArray.cast(resp).add((short) 3, keyParamsMac); - sendOutgoing(apdu, resp); + sendOutgoing(apdu, cert, data[CERTIFICATE], data[KEY_BLOB], data[KEY_CHARACTERISTICS]); } private short generateAttestKeyCmd(APDU apdu) { @@ -3453,16 +3522,6 @@ private short generateAttestKeyCmd(APDU apdu) { return receiveIncoming(apdu, cmd); } - public void getAttestKeyInputParameters(short arrPtr, short[] data, byte keyBlobOff, - byte keyParametersOff, - byte attestKeyBlobOff, byte attestKeyParamsOff, byte attestKeyIssuerOff) { - data[keyBlobOff] = KMArray.cast(arrPtr).get((short) 0); - data[keyParametersOff] = KMArray.cast(arrPtr).get((short) 1); - data[attestKeyBlobOff] = KMType.INVALID_VALUE; - data[attestKeyParamsOff] = KMType.INVALID_VALUE; - data[attestKeyIssuerOff] = KMType.INVALID_VALUE; - } - private short getApplicationId(short params) { short appId = KMKeyParameters.findTag(KMType.BYTES_TAG, KMType.APPLICATION_ID, params); if (appId != KMTag.INVALID_VALUE) { @@ -3487,52 +3546,13 @@ private short getApplicationData(short params) { return appData; } - private void processAttestKeyCmd(APDU apdu) { - // Receive the incoming request fully from the master into buffer. - short cmd = generateAttestKeyCmd(apdu); - // Re-purpose the apdu buffer as scratch pad. - byte[] scratchPad = apdu.getBuffer(); - data[KEY_BLOB] = KMArray.cast(cmd).get((short) 0); - data[KEY_PARAMETERS] = KMArray.cast(cmd).get((short) 1); - data[ATTEST_KEY_BLOB] = KMArray.cast(cmd).get((short) 2); - data[ATTEST_KEY_PARAMS] = KMArray.cast(cmd).get((short) 3); - data[ATTEST_KEY_ISSUER] = KMArray.cast(cmd).get((short) 4); - short keyParamsMac = KMArray.cast(cmd).get((short) 5); - - data[CERTIFICATE] = KMArray.instance((short) 0); // by default the cert is empty. - - // Check for app id and app data. - data[APP_ID] = getApplicationId(data[KEY_PARAMETERS]); - data[APP_DATA] = getApplicationData(data[KEY_PARAMETERS]); - // Check if key requires upgrade. The KeyBlob is parsed inside isKeyUpgradeRequired - // function itself. - if (isKeyUpgradeRequired(data[KEY_BLOB], data[APP_ID], data[APP_DATA], scratchPad)) { - // This condition occurs if either any of the system properties (OsVersion, OsPatchLevel, - // VendorPatchLevel or BootPatchLevel) changes or KeyBlob format changed. So return - // KEY_REQUIRES_UPGRADE error as this scenario is application to ATTEST_KEY_BLOB as well - // as ATTEST_KEY_BLOB got generated before the KEY_BLOB. - KMException.throwIt(KMError.KEY_REQUIRES_UPGRADE); - } - // Validate KeyParams Mac - if (!validateKeyParamsMac(data[KEY_PARAMETERS], keyParamsMac, scratchPad)) { - KMException.throwIt(KMError.INVALID_KEY_BLOB); - } - // The key which is being attested should be asymmetric i.e. RSA or EC - // The Algorithm value can be read from either data[HW_PARAMETERS] or data[KEY_PARAMETERS] - // as the values will be same, and they are cryptographically bound. - short alg = KMEnumTag.getValue(KMType.ALGORITHM, data[HW_PARAMETERS]); - if (alg == KMType.RSA || alg == KMType.EC) { - // Build certificate - generateAttestation(data[ATTEST_KEY_BLOB], data[ATTEST_KEY_PARAMS], scratchPad); - } - short resp = KMArray.instance((short) 2); - KMArray.cast(resp).add((short) 0, KMInteger.uint_16(KMError.OK)); - KMArray.cast(resp).add((short) 1, data[CERTIFICATE]); - sendOutgoing(apdu, resp); - } - private short getAttestationMode(short attKeyBlob, short attChallenge) { + short alg = KMKeyParameters.findTag(KMType.ENUM_TAG, KMType.ALGORITHM, data[KEY_PARAMETERS]); short mode = KMType.NO_CERT; + if(KMEnumTag.cast(alg).getValue() != KMType.RSA && + KMEnumTag.cast(alg).getValue() != KMType.EC) { + return mode; + } // If attestation keyblob preset if (attKeyBlob != KMType.INVALID_VALUE && KMByteBlob.cast(attKeyBlob).length() > 0) { // No attestation challenge present then it is an error @@ -3557,8 +3577,15 @@ private short getAttestationMode(short attKeyBlob, short attChallenge) { return mode; } - private void generateAttestation(short attKeyBlob, short attKeyParam, byte[] scratchPad){ + private KMAttestationCert generateAttestation(short attKeyBlob, short attKeyParam, byte[] scratchPad){ + // 1) If attestation key is present and attestation challenge is absent then it is an error. + // 2) If attestation key is absent and attestation challenge is present then it is an error as + // factory provisioned attestation key is not supported. + // 3) If both are present and issuer is absent or attest key purpose is not ATTEST_KEY then it is an error. + // 4) If the generated/imported keys are RSA or EC then validity period must be specified. + // Device Unique Attestation is not supported. // Device unique attestation not supported + short heapStart = repository.getHeapIndex(); KMTag.assertAbsence(data[KEY_PARAMETERS], KMType.BOOL_TAG, KMType.DEVICE_UNIQUE_ATTESTATION, KMError.CANNOT_ATTEST_IDS); // Read attestation challenge if present @@ -3574,7 +3601,7 @@ private void generateAttestation(short attKeyBlob, short attKeyParam, byte[] sc switch (mode){ case KMType.ATTESTATION_CERT: cert = makeAttestationCert(attKeyBlob, attKeyParam, attChallenge, data[ATTEST_KEY_ISSUER], - data[HW_PARAMETERS], scratchPad); + scratchPad); break; case KMType.SELF_SIGNED_CERT: cert = makeSelfSignedCert(data[SECRET], data[PUB_KEY], mode, scratchPad); @@ -3583,35 +3610,22 @@ private void generateAttestation(short attKeyBlob, short attKeyParam, byte[] sc cert = makeSelfSignedCert(KMType.INVALID_VALUE, data[PUB_KEY], mode, scratchPad); break; default: - data[CERTIFICATE] = KMArray.instance((short)0); - return; + data[CERTIFICATE] = KMType.INVALID_VALUE; + return null; } - // Allocate memory - short certData = KMByteBlob.instance(MAX_CERT_SIZE); - - cert.buffer(KMByteBlob.cast(certData).getBuffer(), - KMByteBlob.cast(certData).getStartOff(), - KMByteBlob.cast(certData).length()); - + // Certificate Data is converted to cbor and written to the end of the stack. + short certData = repository.allocReclaimableMemory(MAX_CERT_SIZE); + // Leave first 4 bytes for Array header and ByteBlob header. + cert.buffer(repository.getHeap(), (short) (certData + 4), (short) (MAX_CERT_SIZE - 4)); // Build the certificate - this will sign the cert cert.build(); - // Adjust the start and length of the certificate in the blob - KMByteBlob.cast(certData).setStartOff(cert.getCertStart()); - KMByteBlob.cast(certData).setLength(cert.getCertLength()); - // Initialize the certificate as array of blob - data[CERTIFICATE] = KMArray.instance((short)1); - KMArray.cast(data[CERTIFICATE]).add((short)0, certData); + // Certificate is now built so the data in the heap starting from heapStart to the current + // heap index can be reused. So resetting the heap index to heapStart. + repository.setHeapIndex(heapStart); + data[CERTIFICATE] = certData; + return cert; } - /** - * 1) If attestation key is present and attestation challenge is absent then it is an error. - * 2) If attestation key is absent and attestation challenge is present then it is an error as - * factory provisioned attestation key is not supported. - * 3) If both are present and issuer is absent or attest key purpose is not ATTEST_KEY then it is an error. - * 4) If the generated/imported keys are RSA or EC then validity period must be specified. - * Device Unique Attestation is not supported. - */ - private static void validateRSAKey(byte[] scratchPad) { // Read key size if(!KMTag.isValidKeySize(data[KEY_PARAMETERS])){ @@ -3800,50 +3814,6 @@ private static void makeKeyCharacteristics(byte[] scratchPad) { KMKeyCharacteristics.cast(data[KEY_CHARACTERISTICS]).setTeeEnforced(data[TEE_PARAMETERS]); } - private short macKeyParams(short keyParams, byte[] scratchPad) { - // For Symmetric Keys no need to Mac the Key parameters. As for symmetric keys - // attestation is not done. - if (SYM_KEY_TYPE == getKeyType(keyParams)) { - return KMByteBlob.instance((short) 0); - } - short offset = repository.allocReclaimableMemory(MAX_KEY_PARAMS_BUF_SIZE); - short len = encoder.encode(keyParams, repository.getHeap(), offset); - if (len > MAX_KEY_PARAMS_BUF_SIZE) { - // KeyParamters exceeded the maximum allowed size. - KMException.throwIt(KMError.INSUFFICIENT_BUFFER_SPACE); - } - - short signLen = seProvider.hmacSign( - KMByteBlob.cast(data[AUTH_TAG]).getBuffer(), - KMByteBlob.cast(data[AUTH_TAG]).getStartOff(), - KMByteBlob.cast(data[AUTH_TAG]).length(), - repository.getHeap(), - offset, - len, - scratchPad, - (short) 0); - //release memory - repository.reclaimMemory(MAX_KEY_PARAMS_BUF_SIZE); - - return KMByteBlob.instance(scratchPad, (short) 0, signLen); - } - - private boolean validateKeyParamsMac(short keyParams, short keyParamsMac, byte[] scratchPad) { - short ptr = macKeyParams(keyParams, scratchPad); - if (KMByteBlob.cast(ptr).length() != KMByteBlob.cast(keyParamsMac).length()) { - return false; - } - if (0 != Util.arrayCompare( - KMByteBlob.cast(ptr).getBuffer(), - KMByteBlob.cast(ptr).getStartOff(), - KMByteBlob.cast(keyParamsMac).getBuffer(), - KMByteBlob.cast(keyParamsMac).getStartOff(), - KMByteBlob.cast(keyParamsMac).length())) { - return false; - } - return true; - } - private static void createEncryptedKeyBlob(byte[] scratchPad) { // make root of trust blob data[ROT] = readROT(scratchPad); @@ -3871,11 +3841,35 @@ private static void createEncryptedKeyBlob(byte[] scratchPad) { KMKeyCharacteristics.cast(tempChar).setKeystoreEnforced(emptyParam); KMKeyCharacteristics.cast(tempChar).setTeeEnforced(data[TEE_PARAMETERS]); KMArray.cast(data[KEY_BLOB]).add(KEY_BLOB_PARAMS, tempChar); + } + // Encodes KeyCharacteristics at the end of the heap + private void encodeKeyCharacteristics(short keyChars) { + byte[] buffer = repository.getHeap(); + short ptr = repository.allocReclaimableMemory(MAX_KEY_CHARS_SIZE); + short len = encoder.encode(keyChars, buffer, ptr, MAX_KEY_CHARS_SIZE); + // shift the encoded KeyCharacteristics data towards the right till the data[CERTIFICATE] offset. + Util.arrayCopyNonAtomic(buffer, ptr, buffer, (short) (ptr + (MAX_KEY_CHARS_SIZE - len)), len); + // Reclaim the unused memory. + repository.reclaimMemory((short) (MAX_KEY_CHARS_SIZE - len)); + } + + // Encodes KeyBlob at the end of the heap + private void encodeKeyBlob(short keyBlobPtr) { // allocate reclaimable memory. - short buffer = repository.alloc((short) 1024); - short keyBlob = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer); - data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, keyBlob); + byte[] buffer = repository.getHeap(); + short top = repository.allocReclaimableMemory(MAX_KEYBLOB_SIZE); + short keyBlob = encoder.encode(keyBlobPtr, buffer, top, MAX_KEYBLOB_SIZE); + Util.arrayCopyNonAtomic(repository.getHeap(), top, repository.getHeap(), + (short) (top + MAX_KEYBLOB_SIZE - keyBlob), keyBlob); + short newTop = (short) (top + MAX_KEYBLOB_SIZE - keyBlob); + // Encode the KeyBlob array inside a ByteString. Get the length of + // the ByteString header. + short encodedBytesLength = encoder.getEncodedBytesLength(keyBlob); + newTop -= encodedBytesLength; + encoder.encodeByteBlobHeader(keyBlob, buffer, newTop, encodedBytesLength); + // Reclaim unused memory. + repository.reclaimMemory((short) (newTop - top)); } private short readKeyBlobVersion(short keyBlob) { @@ -4202,6 +4196,10 @@ public static void generateRkpKey(byte[] scratchPad, short keyParams) { data[ORIGIN] = KMType.GENERATED; makeKeyCharacteristics(scratchPad); createEncryptedKeyBlob(scratchPad); + // allocate reclaimable memory. + short buffer = repository.alloc((short) 1024); + short keyBlob = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer); + data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, keyBlob); } public static short getPubKey() { return data[PUB_KEY]; @@ -4223,7 +4221,7 @@ public static short getPivateKey() { public static short encodeToApduBuffer(short object, byte[] apduBuf, short apduOff, short maxLen) { short offset = repository.allocReclaimableMemory(maxLen); - short len = encoder.encode(object, repository.getHeap(), offset); + short len = encoder.encode(object, repository.getHeap(), offset, maxLen); Util.arrayCopyNonAtomic(repository.getHeap(), offset, apduBuf, apduOff, len); //release memory repository.reclaimMemory(maxLen); @@ -4402,36 +4400,40 @@ public static short generateBcc(boolean testMode, byte[] scratchPad) { } private void updateTrustedConfirmationOperation(KMOperationState op) { - if (op.isTrustedConfirmationRequired()) { - op.getTrustedConfirmationSigner().update(KMByteBlob.cast(data[INPUT_DATA]).getBuffer(), - KMByteBlob.cast(data[INPUT_DATA]).getStartOff(), KMByteBlob.cast(data[INPUT_DATA]).length()); - } + if (op.isTrustedConfirmationRequired()) { + op.getTrustedConfirmationSigner().update(KMByteBlob.cast(data[INPUT_DATA]).getBuffer(), + KMByteBlob.cast(data[INPUT_DATA]).getStartOff(), + KMByteBlob.cast(data[INPUT_DATA]).length()); + } } private void finishTrustedConfirmationOperation(KMOperationState op) { - // Perform trusted confirmation if required - if (op.isTrustedConfirmationRequired()) { - if (0 == KMByteBlob.cast(data[CONFIRMATION_TOKEN]).length()) { - KMException.throwIt(KMError.NO_USER_CONFIRMATION); - } - - boolean verified = op.getTrustedConfirmationSigner().verify(KMByteBlob.cast(data[INPUT_DATA]).getBuffer(), - KMByteBlob.cast(data[INPUT_DATA]).getStartOff(), KMByteBlob.cast(data[INPUT_DATA]).length(), - KMByteBlob.cast(data[CONFIRMATION_TOKEN]).getBuffer(), - KMByteBlob.cast(data[CONFIRMATION_TOKEN]).getStartOff(), - KMByteBlob.cast(data[CONFIRMATION_TOKEN]).length()); - if (!verified) { - KMException.throwIt(KMError.NO_USER_CONFIRMATION); - } - } - } - - private void processGetHeapProfileData(APDU apdu) { + // Perform trusted confirmation if required + if (op.isTrustedConfirmationRequired()) { + if (0 == KMByteBlob.cast(data[CONFIRMATION_TOKEN]).length()) { + KMException.throwIt(KMError.NO_USER_CONFIRMATION); + } + + boolean verified = op.getTrustedConfirmationSigner() + .verify(KMByteBlob.cast(data[INPUT_DATA]).getBuffer(), + KMByteBlob.cast(data[INPUT_DATA]).getStartOff(), + KMByteBlob.cast(data[INPUT_DATA]).length(), + KMByteBlob.cast(data[CONFIRMATION_TOKEN]).getBuffer(), + KMByteBlob.cast(data[CONFIRMATION_TOKEN]).getStartOff(), + KMByteBlob.cast(data[CONFIRMATION_TOKEN]).length()); + if (!verified) { + KMException.throwIt(KMError.NO_USER_CONFIRMATION); + } + } + } + + // Enable this when doing heap profiling. + /*private void processGetHeapProfileData(APDU apdu) { // No Arguments // prepare the response short resp = KMArray.instance((short) 2); KMArray.cast(resp).add((short) 0, KMInteger.uint_16(KMError.OK)); KMArray.cast(resp).add((short) 1, KMInteger.uint_16(repository.getMaxHeapUsed())); sendOutgoing(apdu, resp); - } + }*/ } diff --git a/Applet/src/com/android/javacard/keymaster/KMRepository.java b/Applet/src/com/android/javacard/keymaster/KMRepository.java index e5065d47..98733975 100644 --- a/Applet/src/com/android/javacard/keymaster/KMRepository.java +++ b/Applet/src/com/android/javacard/keymaster/KMRepository.java @@ -16,6 +16,7 @@ package com.android.javacard.keymaster; +import java.util.Base64.Decoder; import org.globalplatform.upgrade.Element; import com.android.javacard.seprovider.KMException; import com.android.javacard.seprovider.KMUpgradable; @@ -32,14 +33,14 @@ */ public class KMRepository { - public static final short HEAP_SIZE = 10000; + public static final short HEAP_SIZE = 8500; // Class Attributes private byte[] heap; private short[] heapIndex; - private short reclaimIndex; - //used for heap profiling - public static short[] maxHeapUsage; + private static short[] reclaimIndex; + // Enable this when doing heap profiling. + //public static short maxHeapUsage; // Singleton instance private static KMRepository repository; @@ -51,8 +52,8 @@ public static KMRepository instance() { public KMRepository(boolean isUpgrading) { heap = JCSystem.makeTransientByteArray(HEAP_SIZE, JCSystem.CLEAR_ON_RESET); heapIndex = JCSystem.makeTransientShortArray((short)1, JCSystem.CLEAR_ON_RESET); - maxHeapUsage = JCSystem.makeTransientShortArray((short)1, JCSystem.CLEAR_ON_RESET); - reclaimIndex = HEAP_SIZE; + reclaimIndex = JCSystem.makeTransientShortArray((short) 1, JCSystem.CLEAR_ON_RESET); + reclaimIndex[0] = HEAP_SIZE; repository = this; } @@ -65,9 +66,9 @@ public void onProcess() { } public void clean() { - Util.arrayFillNonAtomic(heap, (short) 0, heapIndex[0], (byte) 0); + Util.arrayFillNonAtomic(heap, (short) 0, HEAP_SIZE, (byte) 0); heapIndex[0] = 0; - reclaimIndex = HEAP_SIZE; + reclaimIndex[0] = HEAP_SIZE; } public void onDeselect() { @@ -80,21 +81,33 @@ public void onSelect() { // This function uses memory from the back of the heap(transient memory). Call // reclaimMemory function immediately after the use. public short allocReclaimableMemory(short length) { - if ((((short) (reclaimIndex - length)) <= heapIndex[0]) + if ((((short) (reclaimIndex[0] - length)) <= heapIndex[0]) || (length >= HEAP_SIZE / 2)) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } - reclaimIndex -= length; - return reclaimIndex; + reclaimIndex[0] -= length; + // Enable this when doing heap profiling. + //updateHeapProfileData((short) 0); + return reclaimIndex[0]; + } + + // Use this function to reset the heapIndex to its previous state. + // Some of the data might be lost so use it carefully. + public void setHeapIndex(short offset) { + if (offset > heapIndex[0] || offset < 0) { + ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); + } + Util.arrayFillNonAtomic(heap, offset, (short) (heapIndex[0] - offset), (byte) 0); + heapIndex[0] = offset; } // Reclaims the memory back. public void reclaimMemory(short length) { - if (reclaimIndex < heapIndex[0]) { + if (reclaimIndex[0] < heapIndex[0]) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } - Util.arrayFillNonAtomic(heap, reclaimIndex, length, (byte) 0); - reclaimIndex += length; + Util.arrayFillNonAtomic(heap, reclaimIndex[0], length, (byte) 0); + reclaimIndex[0] += length; } public short allocAvailableMemory() { @@ -102,16 +115,18 @@ public short allocAvailableMemory() { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } short index = heapIndex[0]; - heapIndex[0] = (short) heap.length; + heapIndex[0] = reclaimIndex[0]; return index; } public short alloc(short length) { if ((((short) (heapIndex[0] + length)) > heap.length) || - (((short) (heapIndex[0] + length)) > reclaimIndex)) { + (((short) (heapIndex[0] + length)) > reclaimIndex[0])) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } heapIndex[0] += length; + // Enable this when doing heap profiling. + //updateHeapProfileData((short) 0); return (short) (heapIndex[0] - length); } @@ -123,14 +138,23 @@ public short getHeapIndex() { return heapIndex[0]; } + // Enable this when doing heap profiling. +/* public void updateHeapProfileData(short size) { - if(size > maxHeapUsage[0]) { - maxHeapUsage[0] = size; + short totalHeapConsumed = 0; + if (size > 0) { + // This case comes while sending output. + totalHeapConsumed = size; + } else { + totalHeapConsumed = (short) (heapIndex[0] + (HEAP_SIZE - reclaimIndex[0])); + } + if(totalHeapConsumed > maxHeapUsage) { + maxHeapUsage = totalHeapConsumed; } } public short getMaxHeapUsed() { - return maxHeapUsage[0]; + return maxHeapUsage; } - + */ } diff --git a/HAL/JavacardKeyMintDevice.cpp b/HAL/JavacardKeyMintDevice.cpp index 03a1b004..d8cb427c 100644 --- a/HAL/JavacardKeyMintDevice.cpp +++ b/HAL/JavacardKeyMintDevice.cpp @@ -72,10 +72,11 @@ ScopedAStatus JavacardKeyMintDevice::getHardwareInfo(KeyMintHardwareInfo* info) ScopedAStatus JavacardKeyMintDevice::generateKey(const vector& keyParams, const optional& attestationKey, KeyCreationResult* creationResult) { - vector keyParamsMac; cppbor::Array array; // add key params cbor_.addKeyparameters(array, keyParams); + // add attestation key if any + cbor_.addAttestationKey(array, attestationKey); auto [item, err] = card_->sendRequest(Instruction::INS_GENERATE_KEY_CMD, array); if (err != KM_ERROR_OK) { LOG(ERROR) << "Error in sending generateKey."; @@ -83,32 +84,10 @@ ScopedAStatus JavacardKeyMintDevice::generateKey(const vector& key } if (!cbor_.getBinaryArray(item, 1, creationResult->keyBlob) || !cbor_.getKeyCharacteristics(item, 2, creationResult->keyCharacteristics) || - !cbor_.getBinaryArray(item, 3, keyParamsMac)) { + !cbor_.getCertificateChain(item, 3, creationResult->certificateChain)) { LOG(ERROR) << "Error in decoding og response in generateKey."; return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); } - - AuthorizationSet paramSet; - paramSet.Reinitialize(KmParamSet(keyParams)); - // Call attestKey only Asymmetric algorithms. - keymaster_algorithm_t algorithm; - paramSet.GetTagValue(TAG_ALGORITHM, &algorithm); - if (algorithm == KM_ALGORITHM_RSA || algorithm == KM_ALGORITHM_EC) { - cppbor::Array attestKeyArray; - attestKeyArray.add(creationResult->keyBlob); - cbor_.addKeyparameters(attestKeyArray, keyParams); - cbor_.addAttestationKey(attestKeyArray, attestationKey); - attestKeyArray.add(keyParamsMac); - auto [certItem, error] = card_->sendRequest(Instruction::INS_ATTEST_KEY_CMD, attestKeyArray); - if (error != KM_ERROR_OK) { - LOG(ERROR) << "Failed in attestKey err: " << error; - return km_utils::kmError2ScopedAStatus(error); - } - if (!cbor_.getCertificateChain(certItem, 1, creationResult->certificateChain)) { - LOG(ERROR) << "Error in decoding og response in generateKey."; - return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); - } - } return ScopedAStatus::ok(); } @@ -129,7 +108,6 @@ ScopedAStatus JavacardKeyMintDevice::importKey(const vector& keyPa const optional& attestationKey, KeyCreationResult* creationResult) { - vector keyParamsMac; cppbor::Array request; // add key params cbor_.addKeyparameters(request, keyParams); @@ -137,6 +115,8 @@ ScopedAStatus JavacardKeyMintDevice::importKey(const vector& keyPa request.add(Uint(static_cast(keyFormat))); // add key data request.add(Bstr(keyData)); + // add attestation key if any + cbor_.addAttestationKey(request, attestationKey); auto [item, err] = card_->sendRequest(Instruction::INS_IMPORT_KEY_CMD, request); if (err != KM_ERROR_OK) { @@ -145,32 +125,10 @@ ScopedAStatus JavacardKeyMintDevice::importKey(const vector& keyPa } if (!cbor_.getBinaryArray(item, 1, creationResult->keyBlob) || !cbor_.getKeyCharacteristics(item, 2, creationResult->keyCharacteristics) || - !cbor_.getBinaryArray(item, 3, keyParamsMac)) { + !cbor_.getCertificateChain(item, 3, creationResult->certificateChain)) { LOG(ERROR) << "Error in decoding response in importKey."; return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); } - - AuthorizationSet paramSet; - paramSet.Reinitialize(KmParamSet(keyParams)); - // Call attestKey only Asymmetric algorithms. - keymaster_algorithm_t algorithm; - paramSet.GetTagValue(TAG_ALGORITHM, &algorithm); - if (algorithm == KM_ALGORITHM_RSA || algorithm == KM_ALGORITHM_EC) { - cppbor::Array attestKeyArray; - attestKeyArray.add(creationResult->keyBlob); - cbor_.addKeyparameters(attestKeyArray, keyParams); - cbor_.addAttestationKey(attestKeyArray, attestationKey); - attestKeyArray.add(keyParamsMac); - auto [certItem, error] = card_->sendRequest(Instruction::INS_ATTEST_KEY_CMD, attestKeyArray); - if (error != KM_ERROR_OK) { - LOG(ERROR) << "Failed in attestKey err: " << error; - return km_utils::kmError2ScopedAStatus(error); - } - if (!cbor_.getCertificateChain(certItem, 1, creationResult->certificateChain)) { - LOG(ERROR) << "Error in decoding of response in importKey."; - return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); - } - } return ScopedAStatus::ok(); } @@ -192,7 +150,6 @@ ScopedAStatus JavacardKeyMintDevice::importWrappedKey(const vector& wra std::vector tag; vector authList; KeyFormat keyFormat; - vector keyParamsMac; std::vector wrappedKeyDescription; keymaster_error_t errorCode = parseWrappedKey(wrappedKeyData, iv, transitKey, secureKey, tag, authList, keyFormat, wrappedKeyDescription); @@ -217,32 +174,10 @@ ScopedAStatus JavacardKeyMintDevice::importWrappedKey(const vector& wra } if (!cbor_.getBinaryArray(item, 1, creationResult->keyBlob) || !cbor_.getKeyCharacteristics(item, 2, creationResult->keyCharacteristics) || - !cbor_.getBinaryArray(item, 3, keyParamsMac)) { + !cbor_.getCertificateChain(item, 3, creationResult->certificateChain)) { LOG(ERROR) << "Error in decoding the response in importWrappedKey."; return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); } - - AuthorizationSet paramSet; - paramSet.Reinitialize(KmParamSet(authList)); - // Call attestKey only Asymmetric algorithms. - keymaster_algorithm_t algorithm; - paramSet.GetTagValue(TAG_ALGORITHM, &algorithm); - if (algorithm == KM_ALGORITHM_RSA || algorithm == KM_ALGORITHM_EC) { - cppbor::Array attestKeyArray; - attestKeyArray.add(creationResult->keyBlob); - cbor_.addKeyparameters(attestKeyArray, authList); - attestKeyArray.add(keyParamsMac); - cbor_.addAttestationKey(attestKeyArray, std::nullopt); - auto [certItem, error] = card_->sendRequest(Instruction::INS_ATTEST_KEY_CMD, attestKeyArray); - if (error != KM_ERROR_OK) { - LOG(ERROR) << "Failed in attestKey err: " << error; - return km_utils::kmError2ScopedAStatus(error); - } - if (!cbor_.getCertificateChain(certItem, 1, creationResult->certificateChain)) { - LOG(ERROR) << "Error in decoding of response in importWrappedKey."; - return km_utils::kmError2ScopedAStatus(KM_ERROR_UNKNOWN_ERROR); - } - } return ScopedAStatus::ok(); } From dd579c7ebbf72a7714b579cf2db0edac58792a52 Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Mon, 18 Apr 2022 04:55:31 +0000 Subject: [PATCH 2/9] Length validations --- .../javacard/keymaster/KMConfigurations.java | 1 + ...{KMPKCS8Decoder.java => KMAsn1Parser.java} | 36 +++++++++++++-- .../android/javacard/keymaster/KMByteTag.java | 46 +++++++++++++------ .../javacard/keymaster/KMKeyParameters.java | 6 +++ .../javacard/keymaster/KMKeymasterApplet.java | 13 ++++-- .../android/javacard/keymaster/KMType.java | 10 ++-- .../RemotelyProvisionedComponentDevice.java | 2 +- 7 files changed, 83 insertions(+), 31 deletions(-) rename Applet/src/com/android/javacard/keymaster/{KMPKCS8Decoder.java => KMAsn1Parser.java} (88%) diff --git a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java index 3eb4a2e4..0ced4d70 100644 --- a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,4 +20,5 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; + public static final byte MAX_ATTESTATION_IDS_SIZE = 48; } diff --git a/Applet/src/com/android/javacard/keymaster/KMPKCS8Decoder.java b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java similarity index 88% rename from Applet/src/com/android/javacard/keymaster/KMPKCS8Decoder.java rename to Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java index 3040d480..bfe1aaa3 100644 --- a/Applet/src/com/android/javacard/keymaster/KMPKCS8Decoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java @@ -3,13 +3,16 @@ import com.android.javacard.seprovider.KMException; import javacard.framework.Util; -public class KMPKCS8Decoder { +public class KMAsn1Parser { public static final byte ASN1_OCTET_STRING= 0x04; public static final byte ASN1_SEQUENCE= 0x30; + public static final byte ASN1_SET= 0x31; public static final byte ASN1_INTEGER= 0x02; + public static final byte OBJECT_IDENTIFIER = 0x06; public static final byte ASN1_A0_TAG = (byte) 0xA0; public static final byte ASN1_A1_TAG = (byte) 0xA1; public static final byte ASN1_BIT_STRING = 0x03; + public static final byte ASN1_UTF8_STRING = 0x0C; public static final byte[] EC_CURVE = { 0x06,0x08,0x2a,(byte)0x86,0x48,(byte)0xce,0x3d,0x03, 0x01,0x07 @@ -23,12 +26,15 @@ public class KMPKCS8Decoder { 0x3d,0x02,0x01,0x06,0x08,0x2a,(byte)0x86,0x48, (byte)0xce,0x3d,0x03,0x01,0x07 }; + public static final byte[] COMMON_NAME_OID = { + 0x55, 0x04, 0x03 + }; private byte[] data; private short start; private short length; private short cur; - private static KMPKCS8Decoder inst; - private KMPKCS8Decoder(){ + private static KMAsn1Parser inst; + private KMAsn1Parser(){ start = 0; length = 0; cur = 0; @@ -45,6 +51,15 @@ public short decodeEc(short blob){ decodeCommon((short)0, EC_ALGORITHM); return decodeEcPrivateKey((short)1); } + + public short decodeSubject(short blob) { + init(blob); + header(ASN1_SEQUENCE); + header(ASN1_SET); + header(ASN1_SEQUENCE); + objectIdentifier(COMMON_NAME_OID); + return header(ASN1_UTF8_STRING); + } public short decodeEcSubjectPublicKeyInfo(short blob) { init(blob); @@ -183,6 +198,17 @@ private void validateTag0IfPresent(){ if(Util.arrayCompare(data, cur, EC_CURVE, (short)0, len) != 0) KMException.throwIt(KMError.UNKNOWN_ERROR); incrementCursor(len); } + + private short objectIdentifier(byte[] oid) { + short length = header(OBJECT_IDENTIFIER); + if (length != oid.length) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + if(Util.arrayCompare(data, cur, oid, (short)0, length) != 0) KMException.throwIt(KMError.UNKNOWN_ERROR); + incrementCursor(length); + return length; + } + private short header(short tag){ short t = getByte(); if(t != tag) KMException.throwIt(KMError.UNKNOWN_ERROR); @@ -222,9 +248,9 @@ private short getLength(){ else KMException.throwIt(KMError.UNKNOWN_ERROR); return KMType.INVALID_VALUE; //should not come here } - public static KMPKCS8Decoder instance() { + public static KMAsn1Parser instance() { if (inst == null) { - inst = new KMPKCS8Decoder(); + inst = new KMAsn1Parser(); } return inst; } diff --git a/Applet/src/com/android/javacard/keymaster/KMByteTag.java b/Applet/src/com/android/javacard/keymaster/KMByteTag.java index b7b8decd..3cb5d34f 100644 --- a/Applet/src/com/android/javacard/keymaster/KMByteTag.java +++ b/Applet/src/com/android/javacard/keymaster/KMByteTag.java @@ -95,26 +95,23 @@ public short length() { return KMByteBlob.cast(blobPtr).length(); } + // TODO Review this function private static boolean validateKey(short key, short byteBlob) { short valueLen = KMByteBlob.cast(byteBlob).length(); switch (key) { - case ROOT_OF_TRUST: - case UNIQUE_ID: case ATTESTATION_APPLICATION_ID: - case ATTESTATION_ID_BRAND: - case ATTESTATION_ID_DEVICE: - case ATTESTATION_ID_PRODUCT: - case ATTESTATION_ID_SERIAL: - case ATTESTATION_ID_IMEI: - case ATTESTATION_ID_MEID: - case ATTESTATION_ID_MANUFACTURER: - case ATTESTATION_ID_MODEL: - case ASSOCIATED_DATA: - case NONCE: - case CONFIRMATION_TOKEN: - case VERIFIED_BOOT_KEY: - case VERIFIED_BOOT_HASH: + if (valueLen > MAX_ATTESTATION_APP_ID_SIZE) { + return false; + } + break; case CERTIFICATE_SUBJECT_NAME: + { + KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); + short length = asn1Decoder.decodeSubject(byteBlob); + if (length > MAX_SUBJECT_CN_LEN) { + return false; + } + } break; case APPLICATION_ID: case APPLICATION_DATA: @@ -127,6 +124,25 @@ private static boolean validateKey(short key, short byteBlob) { return false; } break; + case ATTESTATION_ID_BRAND: + case ATTESTATION_ID_DEVICE: + case ATTESTATION_ID_PRODUCT: + case ATTESTATION_ID_SERIAL: + case ATTESTATION_ID_IMEI: + case ATTESTATION_ID_MEID: + case ATTESTATION_ID_MANUFACTURER: + case ATTESTATION_ID_MODEL: + if (valueLen > KMConfigurations.MAX_ATTESTATION_IDS_SIZE) { + return false; + } + break; + case ROOT_OF_TRUST: // TODO : Not adding it as ByteTag in HiddenParamters. + //case UNIQUE_ID: This tag never used in keyParamters. + case NONCE: // Validation of nonce happends in begin operation. + // Below two tags are obsolete in keymint. + //case ASSOCIATED_DATA: + //case CONFIRMATION_TOKEN: + break; default: return false; } diff --git a/Applet/src/com/android/javacard/keymaster/KMKeyParameters.java b/Applet/src/com/android/javacard/keymaster/KMKeyParameters.java index 73eee8e1..9b2e4dad 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeyParameters.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeyParameters.java @@ -403,11 +403,17 @@ public static short makeHidden(short keyParamsPtr, short rootOfTrustBlob, byte[] short appId = KMKeyParameters.findTag(KMType.BYTES_TAG, KMType.APPLICATION_ID, keyParamsPtr); if (appId != KMTag.INVALID_VALUE) { appId = KMByteTag.cast(appId).getValue(); + if (KMByteBlob.cast(appId).length() == 0) { + appId = KMTag.INVALID_VALUE; + } } short appData = KMKeyParameters.findTag(KMType.BYTES_TAG, KMType.APPLICATION_DATA, keyParamsPtr); if (appData != KMTag.INVALID_VALUE) { appData = KMByteTag.cast(appData).getValue(); + if (KMByteBlob.cast(appData).length() == 0) { + appData = KMTag.INVALID_VALUE; + } } return makeHidden(appId, appData, rootOfTrustBlob, scratchPad); } diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index cf7212a2..b128e974 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -1381,6 +1381,11 @@ private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyPara if (!KMEnumArrayTag.cast(attKeyPurpose).contains(KMType.ATTEST_KEY)) { KMException.throwIt(KMError.INCOMPATIBLE_PURPOSE); } + KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); + short length = asn1Decoder.decodeSubject(issuer); + if (length > KMType.MAX_SUBJECT_CN_LEN) { + KMException.throwIt(KMError.INVALID_ISSUER_SUBJECT_NAME); + } // If issuer is not present then it is an error if (KMByteBlob.cast(issuer).length() <= 0) { KMException.throwIt(KMError.MISSING_ISSUER_SUBJECT_NAME); @@ -1719,7 +1724,7 @@ private void finishAesDesOperation(KMOperationState op){ private void finishKeyAgreementOperation(KMOperationState op, byte[] scratchPad) { try { - KMPKCS8Decoder pkcs8 = KMPKCS8Decoder.instance(); + KMAsn1Parser pkcs8 = KMAsn1Parser.instance(); short blob = pkcs8.decodeEcSubjectPublicKeyInfo(data[INPUT_DATA]); short len = op.getOperation().finish( KMByteBlob.cast(blob).getBuffer(), @@ -3061,7 +3066,7 @@ private void importKey(APDU apdu, short keyFmt, byte[] scratchPad) { private void importECKeys(byte[] scratchPad) { // Decode key material - KMPKCS8Decoder pkcs8 = KMPKCS8Decoder.instance(); + KMAsn1Parser pkcs8 = KMAsn1Parser.instance(); short keyBlob = pkcs8.decodeEc(data[IMPORTED_KEY_BLOB]); data[PUB_KEY] = KMArray.cast(keyBlob).get((short) 0); data[SECRET] = KMArray.cast(keyBlob).get((short) 1); @@ -3253,7 +3258,7 @@ private void importAESKey(byte[] scratchPad) { private void importRSAKey(byte[] scratchPad) { // Decode key material - KMPKCS8Decoder pkcs8 = KMPKCS8Decoder.instance(); + KMAsn1Parser pkcs8 = KMAsn1Parser.instance(); short keyblob = pkcs8.decodeRsa(data[IMPORTED_KEY_BLOB]); data[PUB_KEY] = KMArray.cast(keyblob).get((short) 0); short pubKeyExp = KMArray.cast(keyblob).get((short)1); @@ -4384,7 +4389,7 @@ public static short generateBcc(boolean testMode, byte[] scratchPad) { scratchPad, temp ); - len = KMPKCS8Decoder.instance(). + len = KMAsn1Parser.instance(). decodeEcdsa256Signature(KMByteBlob.instance(scratchPad, temp, len), scratchPad, temp); coseSignStructure = KMByteBlob.instance(scratchPad, temp, len); diff --git a/Applet/src/com/android/javacard/keymaster/KMType.java b/Applet/src/com/android/javacard/keymaster/KMType.java index df78ba4e..bc5849ab 100644 --- a/Applet/src/com/android/javacard/keymaster/KMType.java +++ b/Applet/src/com/android/javacard/keymaster/KMType.java @@ -145,12 +145,6 @@ public abstract class KMType { public static final byte UNVERIFIED_BOOT = 0x02; public static final byte FAILED_BOOT = 0x03; - // Verified Boot Key - public static final short VERIFIED_BOOT_KEY = (short) 0xF004; - - // Verified Boot Hash - public static final short VERIFIED_BOOT_HASH = (short) 0xF005; - // Device Locked public static final short DEVICE_LOCKED = (short) 0xF006; public static final byte DEVICE_LOCKED_TRUE = 0x01; @@ -364,6 +358,10 @@ public abstract class KMType { public static final short MAX_ATTESTATION_CHALLENGE_SIZE = 128; // Max certificate serial size. public static final short MAX_CERTIFICATE_SERIAL_SIZE = 20; + // Attestation Application ID + public static final short MAX_ATTESTATION_APP_ID_SIZE = 1024; + // Maximum Certificate Subject Common name length. + public static final short MAX_SUBJECT_CN_LEN = 64; protected static KMRepository repository; diff --git a/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java b/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java index 19ac6a82..1682146d 100644 --- a/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java +++ b/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java @@ -785,7 +785,7 @@ private short createSignedMac(KMDeviceUniqueKeyPair deviceUniqueKeyPair, byte[] scratchPad, signStructure ); - len = KMPKCS8Decoder.instance(). + len = KMAsn1Parser.instance(). decodeEcdsa256Signature(KMByteBlob.instance(scratchPad, signStructure, len), scratchPad, signStructure); signStructure = KMByteBlob.instance(scratchPad, signStructure, len); From 1ac564a3eb8b455a494a0e285e4e97a8e497285d Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Mon, 18 Apr 2022 05:07:39 +0000 Subject: [PATCH 3/9] update the length in KmConfigurations --- .../src/com/android/javacard/keymaster/KMConfigurations.java | 1 + 1 file changed, 1 insertion(+) diff --git a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java index 3eb4a2e4..f18506e5 100644 --- a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,4 +20,5 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; + public static final byte MAX_ATTESTATION_IDS_SIZE = 48; } From 185b6cff4723205780ef615bd65bf68bff967db0 Mon Sep 17 00:00:00 2001 From: "avinash.hedage" Date: Mon, 18 Apr 2022 18:17:50 +0000 Subject: [PATCH 4/9] Updated cert subject decoding --- .../javacard/keymaster/KMAsn1Parser.java | 20 ++++++++++++++++++- .../android/javacard/keymaster/KMDecoder.java | 11 ++++++---- .../android/javacard/keymaster/KMEncoder.java | 2 +- .../javacard/keymaster/KMKeymasterApplet.java | 8 ++++++-- 4 files changed, 33 insertions(+), 8 deletions(-) diff --git a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java index bfe1aaa3..6dcaf4f3 100644 --- a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java +++ b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java @@ -12,7 +12,13 @@ public class KMAsn1Parser { public static final byte ASN1_A0_TAG = (byte) 0xA0; public static final byte ASN1_A1_TAG = (byte) 0xA1; public static final byte ASN1_BIT_STRING = 0x03; + public static final byte ASN1_UTF8_STRING = 0x0C; + public static final byte ASN1_TELETEX_STRING = 0x14; + public static final byte ASN1_PRINTABLE_STRING = 0x13; + public static final byte ASN1_UNIVERSAL_STRING = 0x1C; + public static final byte ASN1_BMP_STRING = 0x1E; + public static final byte[] EC_CURVE = { 0x06,0x08,0x2a,(byte)0x86,0x48,(byte)0xce,0x3d,0x03, 0x01,0x07 @@ -58,7 +64,7 @@ public short decodeSubject(short blob) { header(ASN1_SET); header(ASN1_SEQUENCE); objectIdentifier(COMMON_NAME_OID); - return header(ASN1_UTF8_STRING); + return subjectHeader(); } public short decodeEcSubjectPublicKeyInfo(short blob) { @@ -215,6 +221,18 @@ private short header(short tag){ return getLength(); } + private short subjectHeader(){ + short t = getByte(); + if(t != ASN1_UTF8_STRING && + t != ASN1_TELETEX_STRING && + t != ASN1_PRINTABLE_STRING && + t != ASN1_UNIVERSAL_STRING && + t != ASN1_BMP_STRING) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + return getLength(); + } + private byte getByte(){ byte d = data[cur]; incrementCursor((short)1); diff --git a/Applet/src/com/android/javacard/keymaster/KMDecoder.java b/Applet/src/com/android/javacard/keymaster/KMDecoder.java index ac17b9eb..e3c7bc45 100644 --- a/Applet/src/com/android/javacard/keymaster/KMDecoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMDecoder.java @@ -363,10 +363,13 @@ private short decodeKeyParam(short exp) { obj = decode(tagClass); KMArray.cast(vals).add(arrPos++, obj); break; - }catch(KMException e){ - if(KMException.reason() == KMError.INVALID_TAG && - !ignoreInvalidTags){ - KMException.throwIt(KMError.INVALID_TAG); + } catch(KMException e){ + if (KMException.reason() == KMError.INVALID_TAG) { + if(!ignoreInvalidTags){ + KMException.throwIt(KMError.INVALID_TAG); + } + }else { + KMException.throwIt(KMException.reason()); } break; } diff --git a/Applet/src/com/android/javacard/keymaster/KMEncoder.java b/Applet/src/com/android/javacard/keymaster/KMEncoder.java index 4cfc9be3..9b53dbdf 100644 --- a/Applet/src/com/android/javacard/keymaster/KMEncoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMEncoder.java @@ -77,7 +77,7 @@ private void encode(short obj) { push(obj); } - // Use this function, when the max len + // Use this function, when the max len is given public short encode(short object, byte[] buffer, short startOff, short maxLength) { scratchBuf[STACK_PTR_OFFSET] = 0; bufferRef[0] = buffer; diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index b128e974..3d8369a3 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -1351,7 +1351,6 @@ private KMAttestationCert makeCommonCert(byte[] scratchPad) { return cert; } - private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyParam, short attChallenge, short issuer, byte[] scratchPad) { KMAttestationCert cert = makeCommonCert(scratchPad); @@ -1382,7 +1381,12 @@ private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyPara KMException.throwIt(KMError.INCOMPATIBLE_PURPOSE); } KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); - short length = asn1Decoder.decodeSubject(issuer); + short length = 0; + try { + length = asn1Decoder.decodeSubject(issuer); + } catch (KMException e) { + KMException.throwIt(KMError.INVALID_ISSUER_SUBJECT_NAME); + } if (length > KMType.MAX_SUBJECT_CN_LEN) { KMException.throwIt(KMError.INVALID_ISSUER_SUBJECT_NAME); } From c78e66a039b6b128d9bedc2a35144aeca5640af9 Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Tue, 19 Apr 2022 03:56:44 +0000 Subject: [PATCH 5/9] Added Subject DER validation --- .../keymaster/KMAttestationCertImpl.java | 2 +- .../javacard/keymaster/KMAsn1Parser.java | 90 +++++++++++++------ .../android/javacard/keymaster/KMByteTag.java | 4 +- .../javacard/keymaster/KMKeymasterApplet.java | 4 +- .../android/javacard/keymaster/KMType.java | 4 +- 5 files changed, 70 insertions(+), 34 deletions(-) diff --git a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAttestationCertImpl.java b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAttestationCertImpl.java index a5e4953b..0da291eb 100644 --- a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAttestationCertImpl.java +++ b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAttestationCertImpl.java @@ -110,7 +110,7 @@ public class KMAttestationCertImpl implements KMAttestationCert { KMType.ATTESTATION_ID_SERIAL, KMType.ATTESTATION_ID_PRODUCT, KMType.ATTESTATION_ID_DEVICE, KMType.ATTESTATION_ID_BRAND, KMType.OS_PATCH_LEVEL, KMType.OS_VERSION, KMType.ROOT_OF_TRUST, - KMType.ORIGIN, KMType.UNLOCKED_DEVICE_REQUIRED, + KMType.ORIGIN, KMType.UNLOCKED_DEVICE_REQUIRED, KMType.TRUSTED_CONFIRMATION_REQUIRED, KMType.AUTH_TIMEOUT, KMType.USER_AUTH_TYPE, KMType.NO_AUTH_REQUIRED, KMType.EARLY_BOOT_ONLY, diff --git a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java index 6dcaf4f3..07f9ed2e 100644 --- a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java +++ b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java @@ -32,8 +32,24 @@ public class KMAsn1Parser { 0x3d,0x02,0x01,0x06,0x08,0x2a,(byte)0x86,0x48, (byte)0xce,0x3d,0x03,0x01,0x07 }; - public static final byte[] COMMON_NAME_OID = { - 0x55, 0x04, 0x03 + + //https://datatracker.ietf.org/doc/html/rfc5280, RFC 5280, Page 21 + public byte[] COMMON_OID = new byte[] { + 0x06, 0x03, 0x55, 0x04 + }; + // This array contains the last byte of OID for each oid type. + // The first 4 bytes are common as shown above in COMMON_OID + private static final byte[] attributeOIds1 = { + 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x0A, 0x0B, 0x0C, 0x2A, + 0x2B, 0x2C, 0x2E, 0x41, + }; + // https://datatracker.ietf.org/doc/html/rfc5280, RFC 5280, Page 124 + // TODO Specification does not mention about the DN_QUALIFIER_OID max length. + private static final byte[] attributeValueMaxLen = { + 0x40/*64 commonName*/, 0x28/*40 surname*/, 0x40/*64 serial*/, 0x02/*64 country*/, + (byte)0x80/*128 locality*/, (byte)0x80/*128 state*/, 0x40/*64 organization*/, 0x40/*64 organization unit*/, + 0x40/*64 title*/, 0x10/*16 givenName*/, 0x05/* initials*/, 0x03/* gen qualifier*/, 0x40,/*64 dn-qualifier*/ + (byte)0x80/*128 pseudonym*/ }; private byte[] data; private short start; @@ -58,13 +74,16 @@ public short decodeEc(short blob){ return decodeEcPrivateKey((short)1); } - public short decodeSubject(short blob) { + public void validateDerSubject(short blob) { init(blob); header(ASN1_SEQUENCE); - header(ASN1_SET); - header(ASN1_SEQUENCE); - objectIdentifier(COMMON_NAME_OID); - return subjectHeader(); + while (cur < ((short) (start + length))) { + header(ASN1_SET); + header(ASN1_SEQUENCE); + // Parse and validate OBJECT-IDENTIFIER and Value fields + // Cursor is incremented in validateAttributeTypeAndValue. + validateAttributeTypeAndValue(); + } } public short decodeEcSubjectPublicKeyInfo(short blob) { @@ -205,14 +224,43 @@ private void validateTag0IfPresent(){ incrementCursor(len); } - private short objectIdentifier(byte[] oid) { - short length = header(OBJECT_IDENTIFIER); - if (length != oid.length) { - KMException.throwIt(KMError.UNKNOWN_ERROR); - } - if(Util.arrayCompare(data, cur, oid, (short)0, length) != 0) KMException.throwIt(KMError.UNKNOWN_ERROR); - incrementCursor(length); - return length; + private void validateAttributeTypeAndValue() { + short start = cur; + if (getByte() != OBJECT_IDENTIFIER) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + short length = getLength(); + if (length != 3) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + cur = start; + boolean found = false; + for(short i = 0; i < (short) attributeOIds1.length; i++) { + if ((Util.arrayCompare(data, cur, COMMON_OID, (short)0, (short) COMMON_OID.length) == 0) && + (attributeOIds1[i] == data[(short)(cur + COMMON_OID.length)])) { + incrementCursor((short) (COMMON_OID.length + 1)); + // Validate the length of the attribute value. + short tag = getByte(); + if(tag != ASN1_UTF8_STRING && + tag != ASN1_TELETEX_STRING && + tag != ASN1_PRINTABLE_STRING && + tag != ASN1_UNIVERSAL_STRING && + tag != ASN1_BMP_STRING) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + length = getLength(); + if (length > attributeValueMaxLen[i]) { + KMException.throwIt(KMError.UNKNOWN_ERROR); + } + incrementCursor(length); + found = true; + break; + } + } + if (!found) { + // None of the attributes matches. + KMException.throwIt(KMError.UNKNOWN_ERROR); + } } private short header(short tag){ @@ -221,18 +269,6 @@ private short header(short tag){ return getLength(); } - private short subjectHeader(){ - short t = getByte(); - if(t != ASN1_UTF8_STRING && - t != ASN1_TELETEX_STRING && - t != ASN1_PRINTABLE_STRING && - t != ASN1_UNIVERSAL_STRING && - t != ASN1_BMP_STRING) { - KMException.throwIt(KMError.UNKNOWN_ERROR); - } - return getLength(); - } - private byte getByte(){ byte d = data[cur]; incrementCursor((short)1); diff --git a/Applet/src/com/android/javacard/keymaster/KMByteTag.java b/Applet/src/com/android/javacard/keymaster/KMByteTag.java index 3cb5d34f..88f3ea63 100644 --- a/Applet/src/com/android/javacard/keymaster/KMByteTag.java +++ b/Applet/src/com/android/javacard/keymaster/KMByteTag.java @@ -107,8 +107,8 @@ private static boolean validateKey(short key, short byteBlob) { case CERTIFICATE_SUBJECT_NAME: { KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); - short length = asn1Decoder.decodeSubject(byteBlob); - if (length > MAX_SUBJECT_CN_LEN) { + asn1Decoder.validateDerSubject(byteBlob); + if (valueLen > MAX_SUBJECT_DER_LEN) { return false; } } diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index 3d8369a3..b6825173 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -1383,11 +1383,11 @@ private KMAttestationCert makeAttestationCert(short attKeyBlob, short attKeyPara KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); short length = 0; try { - length = asn1Decoder.decodeSubject(issuer); + asn1Decoder.validateDerSubject(issuer); } catch (KMException e) { KMException.throwIt(KMError.INVALID_ISSUER_SUBJECT_NAME); } - if (length > KMType.MAX_SUBJECT_CN_LEN) { + if (KMByteBlob.cast(issuer).length() > KMType.MAX_SUBJECT_DER_LEN) { KMException.throwIt(KMError.INVALID_ISSUER_SUBJECT_NAME); } // If issuer is not present then it is an error diff --git a/Applet/src/com/android/javacard/keymaster/KMType.java b/Applet/src/com/android/javacard/keymaster/KMType.java index bc5849ab..95806ad3 100644 --- a/Applet/src/com/android/javacard/keymaster/KMType.java +++ b/Applet/src/com/android/javacard/keymaster/KMType.java @@ -360,8 +360,8 @@ public abstract class KMType { public static final short MAX_CERTIFICATE_SERIAL_SIZE = 20; // Attestation Application ID public static final short MAX_ATTESTATION_APP_ID_SIZE = 1024; - // Maximum Certificate Subject Common name length. - public static final short MAX_SUBJECT_CN_LEN = 64; + // DER subject max length. + public static final short MAX_SUBJECT_DER_LEN = 1095; protected static KMRepository repository; From 4941008037b374de2d0298412c3c36bb3e008f45 Mon Sep 17 00:00:00 2001 From: "avinash.hedage" Date: Tue, 19 Apr 2022 07:04:03 +0000 Subject: [PATCH 6/9] added max length validation for attestation ids --- .../src/com/android/javacard/keymaster/KMAndroidSEApplet.java | 3 +++ 1 file changed, 3 insertions(+) diff --git a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAndroidSEApplet.java b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAndroidSEApplet.java index 3a3d4983..d024fb83 100644 --- a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAndroidSEApplet.java +++ b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMAndroidSEApplet.java @@ -288,6 +288,9 @@ public void setAttestationIds(short attIdVals) { KMException.throwIt(KMError.INVALID_ARGUMENT); } obj = KMByteTag.cast(obj).getValue(); + if (KMByteBlob.cast(obj).length() > KMConfigurations.MAX_ATTESTATION_IDS_SIZE) { + KMException.throwIt(KMError.INVALID_INPUT_LENGTH); + } kmDataStore.setAttestationId(key, KMByteBlob.cast(obj).getBuffer(), KMByteBlob.cast(obj).getStartOff(), KMByteBlob.cast(obj).length()); index++; From 58c1df9d18d08bc16041953511cbe7d0b5c87ded Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Tue, 19 Apr 2022 07:51:00 +0000 Subject: [PATCH 7/9] Incorporated review comments --- .../javacard/keymaster/KMConfigurations.java | 2 +- .../javacard/keymaster/KMConfigurations.java | 2 +- .../javacard/keymaster/KMAsn1Parser.java | 44 ++++++++++++++----- .../android/javacard/keymaster/KMByteTag.java | 13 ++---- .../javacard/keymaster/KMKeymasterApplet.java | 9 ++-- 5 files changed, 43 insertions(+), 27 deletions(-) diff --git a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java index f18506e5..3085a4f1 100644 --- a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,5 +20,5 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; - public static final byte MAX_ATTESTATION_IDS_SIZE = 48; + public static final byte MAX_ATTESTATION_IDS_SIZE = 64; } diff --git a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java index 0ced4d70..3085a4f1 100644 --- a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,5 +20,5 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; - public static final byte MAX_ATTESTATION_IDS_SIZE = 48; + public static final byte MAX_ATTESTATION_IDS_SIZE = 64; } diff --git a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java index 07f9ed2e..bb0c82ba 100644 --- a/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java +++ b/Applet/src/com/android/javacard/keymaster/KMAsn1Parser.java @@ -34,22 +34,46 @@ public class KMAsn1Parser { }; //https://datatracker.ietf.org/doc/html/rfc5280, RFC 5280, Page 21 + // 2.5.4 public byte[] COMMON_OID = new byte[] { 0x06, 0x03, 0x55, 0x04 }; // This array contains the last byte of OID for each oid type. // The first 4 bytes are common as shown above in COMMON_OID - private static final byte[] attributeOIds1 = { - 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x0A, 0x0B, 0x0C, 0x2A, - 0x2B, 0x2C, 0x2E, 0x41, + private static final byte[] attributeOIds = { + 0x03, /* commonName COMMON_OID.3 */ + 0x04, /* surName COMMON_OID.4*/ + 0x05, /* serialNumber COMMON_OID.5 */ + 0x06, /* countryName COMMON_OID.6 */ + 0x07, /* locality COMMON_OID.7 */ + 0x08, /* stateOrProviince COMMON_OID.8 */ + 0x0A, /* organizationName COMMON_OID.10 */ + 0x0B, /* organizationalUnitName COMMON_OID.11 */ + 0x0C, /* title COMMON_OID.10 */ + 0x2A, /* givenName COMMON_OID.42 */ + 0x2B, /* initials COMMON_OID.43 */ + 0x2C, /* generationQualifier COMMON_OID.44 */ + 0x2E, /* dnQualifer COMMON_OID.46 */ + 0x41, /* pseudonym COMMON_OID.65 */ }; // https://datatracker.ietf.org/doc/html/rfc5280, RFC 5280, Page 124 // TODO Specification does not mention about the DN_QUALIFIER_OID max length. + // So the max limit is set at 64. private static final byte[] attributeValueMaxLen = { - 0x40/*64 commonName*/, 0x28/*40 surname*/, 0x40/*64 serial*/, 0x02/*64 country*/, - (byte)0x80/*128 locality*/, (byte)0x80/*128 state*/, 0x40/*64 organization*/, 0x40/*64 organization unit*/, - 0x40/*64 title*/, 0x10/*16 givenName*/, 0x05/* initials*/, 0x03/* gen qualifier*/, 0x40,/*64 dn-qualifier*/ - (byte)0x80/*128 pseudonym*/ + 0x40, /* 1-64 commonName */ + 0x28, /* 1-40 surname */ + 0x40, /* 1-64 serial */ + 0x02, /* 1-2 country */ + (byte) 0x80, /* 1-128 locality */ + (byte) 0x80, /* 1-128 state */ + 0x40, /* 1-64 organization */ + 0x40, /* 1-64 organization unit*/ + 0x40, /* 1-64 title */ + 0x10, /* 1-16 givenName */ + 0x05, /* 1-5 initials */ + 0x03, /* 1-3 gen qualifier */ + 0x40, /* 1-64 dn-qualifier */ + (byte) 0x80 /* 1-128 pseudonym */ }; private byte[] data; private short start; @@ -235,9 +259,9 @@ private void validateAttributeTypeAndValue() { } cur = start; boolean found = false; - for(short i = 0; i < (short) attributeOIds1.length; i++) { + for(short i = 0; i < (short) attributeOIds.length; i++) { if ((Util.arrayCompare(data, cur, COMMON_OID, (short)0, (short) COMMON_OID.length) == 0) && - (attributeOIds1[i] == data[(short)(cur + COMMON_OID.length)])) { + (attributeOIds[i] == data[(short)(cur + COMMON_OID.length)])) { incrementCursor((short) (COMMON_OID.length + 1)); // Validate the length of the attribute value. short tag = getByte(); @@ -249,7 +273,7 @@ private void validateAttributeTypeAndValue() { KMException.throwIt(KMError.UNKNOWN_ERROR); } length = getLength(); - if (length > attributeValueMaxLen[i]) { + if (length <= 0 && length > attributeValueMaxLen[i]) { KMException.throwIt(KMError.UNKNOWN_ERROR); } incrementCursor(length); diff --git a/Applet/src/com/android/javacard/keymaster/KMByteTag.java b/Applet/src/com/android/javacard/keymaster/KMByteTag.java index 88f3ea63..70fa8959 100644 --- a/Applet/src/com/android/javacard/keymaster/KMByteTag.java +++ b/Applet/src/com/android/javacard/keymaster/KMByteTag.java @@ -95,7 +95,6 @@ public short length() { return KMByteBlob.cast(blobPtr).length(); } - // TODO Review this function private static boolean validateKey(short key, short byteBlob) { short valueLen = KMByteBlob.cast(byteBlob).length(); switch (key) { @@ -106,11 +105,11 @@ private static boolean validateKey(short key, short byteBlob) { break; case CERTIFICATE_SUBJECT_NAME: { - KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); - asn1Decoder.validateDerSubject(byteBlob); if (valueLen > MAX_SUBJECT_DER_LEN) { return false; } + KMAsn1Parser asn1Decoder = KMAsn1Parser.instance(); + asn1Decoder.validateDerSubject(byteBlob); } break; case APPLICATION_ID: @@ -136,12 +135,8 @@ private static boolean validateKey(short key, short byteBlob) { return false; } break; - case ROOT_OF_TRUST: // TODO : Not adding it as ByteTag in HiddenParamters. - //case UNIQUE_ID: This tag never used in keyParamters. - case NONCE: // Validation of nonce happends in begin operation. - // Below two tags are obsolete in keymint. - //case ASSOCIATED_DATA: - //case CONFIRMATION_TOKEN: + case ROOT_OF_TRUST: + case NONCE: break; default: return false; diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index b6825173..4761687a 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -668,12 +668,9 @@ public void sendOutgoing(APDU apdu, KMAttestationCert cert, short certStart, sho encodeKeyCharacteristics(keyChars); // and encode it to the end of the buffer before KEY_CHARACTERISTICS encodeKeyBlob(keyblob); - // Write Error code before data[KEY_BLOB] - short bufferStartOffset = repository.allocReclaimableMemory((short) 1); - buffer[bufferStartOffset] = 0x00; - // Write Array header before ErrorCode. - bufferStartOffset = repository.allocReclaimableMemory((short) 1); - buffer[bufferStartOffset] = (byte) 0x84; + // Write Array header and ErrorCode before data[KEY_BLOB] + short bufferStartOffset = repository.allocReclaimableMemory((short) 2); + Util.setShort(buffer, bufferStartOffset, (short) 0x8400); short bufferLength = (short) (KMRepository.HEAP_SIZE - bufferStartOffset); // Send data From 6edc56e9780c1252d1ed787c9d4cd0ac27c8e7ff Mon Sep 17 00:00:00 2001 From: "avinash.hedage" Date: Tue, 19 Apr 2022 18:06:06 +0000 Subject: [PATCH 8/9] encode function changes + removed heap profiling --- .../android/javacard/keymaster/KMCoseMap.java | 4 +- .../android/javacard/keymaster/KMEncoder.java | 21 +++------- .../javacard/keymaster/KMKeymasterApplet.java | 41 ++++++------------- .../javacard/keymaster/KMRepository.java | 28 ++----------- .../RemotelyProvisionedComponentDevice.java | 2 +- 5 files changed, 23 insertions(+), 73 deletions(-) diff --git a/Applet/src/com/android/javacard/keymaster/KMCoseMap.java b/Applet/src/com/android/javacard/keymaster/KMCoseMap.java index ceb67fb2..2f57be40 100644 --- a/Applet/src/com/android/javacard/keymaster/KMCoseMap.java +++ b/Applet/src/com/android/javacard/keymaster/KMCoseMap.java @@ -127,8 +127,8 @@ private static boolean compareAndSwap(short ptr, short index) { firstKey = KMMap.cast(ptr).getKey(index); secondKey = KMMap.cast(ptr).getKey((short) (index + 1)); } - firstKeyLen = KMKeymasterApplet.encoder.encode(firstKey, scratchpad, (short) 0); - secondKeyLen = KMKeymasterApplet.encoder.encode(secondKey, scratchpad, firstKeyLen); + firstKeyLen = KMKeymasterApplet.encoder.encode(firstKey, scratchpad, (short) 0, (short) scratchpad.length); + secondKeyLen = KMKeymasterApplet.encoder.encode(secondKey, scratchpad, firstKeyLen, (short) scratchpad.length); if ((firstKeyLen > secondKeyLen) || ((firstKeyLen == secondKeyLen) && (0 < Util.arrayCompare(scratchpad, (short) 0, scratchpad, firstKeyLen, firstKeyLen)))) { diff --git a/Applet/src/com/android/javacard/keymaster/KMEncoder.java b/Applet/src/com/android/javacard/keymaster/KMEncoder.java index 9b53dbdf..a8d9e328 100644 --- a/Applet/src/com/android/javacard/keymaster/KMEncoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMEncoder.java @@ -78,32 +78,21 @@ private void encode(short obj) { } // Use this function, when the max len is given - public short encode(short object, byte[] buffer, short startOff, short maxLength) { + public short encode(short object, byte[] buffer, short startOff, short bufLen, short encoderOutLimitLen) { scratchBuf[STACK_PTR_OFFSET] = 0; bufferRef[0] = buffer; scratchBuf[START_OFFSET] = startOff; - if ((short) (startOff + maxLength) > buffer.length) { + if ((short) (startOff + encoderOutLimitLen) > bufLen) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } - scratchBuf[LEN_OFFSET] = (short) (startOff + maxLength); + scratchBuf[LEN_OFFSET] = (short) (startOff + encoderOutLimitLen); push(object); encode(); return (short) (scratchBuf[START_OFFSET] - startOff); } - public short encode(short object, byte[] buffer, short startOff) { - scratchBuf[STACK_PTR_OFFSET] = 0; - bufferRef[0] = buffer; - scratchBuf[START_OFFSET] = startOff; - short len = (short) (buffer.length - startOff); - if ((len < 0) || len > KMRepository.HEAP_SIZE) { - scratchBuf[LEN_OFFSET] = KMRepository.HEAP_SIZE; - } else { - scratchBuf[LEN_OFFSET] = (short) buffer.length; - } - push(object); - encode(); - return (short) (scratchBuf[START_OFFSET] - startOff); + public short encode(short object, byte[] buffer, short startOff, short bufLen) { + return encode(object, buffer, startOff, bufLen, (short) (bufLen - startOff)); } //array{KMError.OK,Array{KMByteBlobs}} diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index 4761687a..c6d5f238 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -147,9 +147,6 @@ public class KMKeymasterApplet extends Applet implements AppletEvent, ExtendedLe public static final byte INS_FINISH_SEND_DATA_CMD = KEYMINT_CMD_APDU_START + 33; //0x41 public static final byte INS_GET_RESPONSE_CMD = KEYMINT_CMD_APDU_START + 34; //0x42 private static final byte KEYMINT_CMD_APDU_END = KEYMINT_CMD_APDU_START + 35; //0x43 - // Enable this when doing heap profiling. - //private static final byte INS_GET_HEAP_PROFILE_DATA = KEYMINT_CMD_APDU_START + 36; //0x44 - private static final byte INS_END_KM_CMD = 0x7F; // Data Dictionary items @@ -211,7 +208,7 @@ public class KMKeymasterApplet extends Applet implements AppletEvent, ExtendedLe public static final byte AES_GCM_NONCE_LENGTH = 12; // ComputeHMAC constants private static final short HMAC_SHARED_PARAM_MAX_SIZE = 64; - protected static final short MAX_CERT_SIZE = 2500; + protected static final short MAX_CERT_SIZE = 3000; protected static final short MAX_KEY_CHARS_SIZE = 512; protected static final short MAX_KEYBLOB_SIZE = 1024; // KEYBLOB_CURRENT_VERSION goes into KeyBlob and will affect all @@ -491,10 +488,6 @@ public void process(APDU apdu) { case INS_GET_RKP_HARDWARE_INFO: rkp.process(apduIns, apdu); break; - // Enable this when doing heap profiling. - /*case INS_GET_HEAP_PROFILE_DATA: - processGetHeapProfileData(apdu); - break;*/ default: ISOException.throwIt(ISO7816.SW_INS_NOT_SUPPORTED); } @@ -608,18 +601,14 @@ protected void resetData() { */ public static void sendOutgoing(APDU apdu, short resp) { //TODO handle the extended buffer stuff. We can reuse this. - // Enable this when doing heap profiling. - //short usedHeap = repository.getHeapIndex(); short bufferStartOffset = repository.allocAvailableMemory(); byte[] buffer = repository.getHeap(); // TODO we can change the following to incremental send. - short bufferLength = encoder.encode(resp, buffer, bufferStartOffset); + short bufferLength = encoder.encode(resp, buffer, bufferStartOffset, repository.getHeapReclaimIndex()); if (((short) (bufferLength + bufferStartOffset)) > ((short) repository .getHeap().length)) { ISOException.throwIt(ISO7816.SW_WRONG_LENGTH); } - // Enable this when doing heap profiling. - //repository.updateHeapProfileData((short) (bufferLength + usedHeap)); // Send data apdu.setOutgoing(); apdu.setOutgoingLength(bufferLength); @@ -1085,7 +1074,7 @@ private void processUpgradeKeyCmd(APDU apdu) { createEncryptedKeyBlob(scratchPad); // allocate reclaimable memory. short buffer = repository.alloc(MAX_KEYBLOB_SIZE); - data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer); + data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer, repository.getHeapReclaimIndex()); data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, data[KEY_BLOB]); } else { data[KEY_BLOB] = KMByteBlob.instance((short) 0); @@ -3852,8 +3841,9 @@ private static void createEncryptedKeyBlob(byte[] scratchPad) { // Encodes KeyCharacteristics at the end of the heap private void encodeKeyCharacteristics(short keyChars) { byte[] buffer = repository.getHeap(); + short prevReclaimIndex = repository.getHeapReclaimIndex(); short ptr = repository.allocReclaimableMemory(MAX_KEY_CHARS_SIZE); - short len = encoder.encode(keyChars, buffer, ptr, MAX_KEY_CHARS_SIZE); + short len = encoder.encode(keyChars, buffer, ptr, prevReclaimIndex, MAX_KEY_CHARS_SIZE); // shift the encoded KeyCharacteristics data towards the right till the data[CERTIFICATE] offset. Util.arrayCopyNonAtomic(buffer, ptr, buffer, (short) (ptr + (MAX_KEY_CHARS_SIZE - len)), len); // Reclaim the unused memory. @@ -3864,8 +3854,9 @@ private void encodeKeyCharacteristics(short keyChars) { private void encodeKeyBlob(short keyBlobPtr) { // allocate reclaimable memory. byte[] buffer = repository.getHeap(); + short prevReclaimIndex = repository.getHeapReclaimIndex(); short top = repository.allocReclaimableMemory(MAX_KEYBLOB_SIZE); - short keyBlob = encoder.encode(keyBlobPtr, buffer, top, MAX_KEYBLOB_SIZE); + short keyBlob = encoder.encode(keyBlobPtr, buffer, top, prevReclaimIndex, MAX_KEYBLOB_SIZE); Util.arrayCopyNonAtomic(repository.getHeap(), top, repository.getHeap(), (short) (top + MAX_KEYBLOB_SIZE - keyBlob), keyBlob); short newTop = (short) (top + MAX_KEYBLOB_SIZE - keyBlob); @@ -4100,13 +4091,14 @@ private static void makeAuthData(short version, byte[] scratchPad) { default: KMException.throwIt(KMError.INVALID_KEY_BLOB); } + short prevReclaimIndex = repository.getHeapReclaimIndex(); short authIndex = repository.allocReclaimableMemory(MAX_AUTH_DATA_SIZE); index = 0; short len = 0; Util.arrayFillNonAtomic(repository.getHeap(), authIndex, MAX_AUTH_DATA_SIZE, (byte) 0); while (index < numParams) { short tag = Util.getShort(scratchPad, (short) (index * 2)); - len = encoder.encode(tag, repository.getHeap(), (short) (authIndex + 32)); + len = encoder.encode(tag, repository.getHeap(), (short) (authIndex + 32), prevReclaimIndex); Util.arrayCopyNonAtomic(repository.getHeap(), authIndex, repository.getHeap(), (short) (authIndex + len + 32), (short) 32); len = seProvider.messageDigest256(repository.getHeap(), @@ -4204,7 +4196,7 @@ public static void generateRkpKey(byte[] scratchPad, short keyParams) { createEncryptedKeyBlob(scratchPad); // allocate reclaimable memory. short buffer = repository.alloc((short) 1024); - short keyBlob = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer); + short keyBlob = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer, repository.getHeapReclaimIndex()); data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, keyBlob); } public static short getPubKey() { @@ -4226,8 +4218,9 @@ public static short getPivateKey() { */ public static short encodeToApduBuffer(short object, byte[] apduBuf, short apduOff, short maxLen) { + short prevReclaimIndex = repository.getHeapReclaimIndex(); short offset = repository.allocReclaimableMemory(maxLen); - short len = encoder.encode(object, repository.getHeap(), offset, maxLen); + short len = encoder.encode(object, repository.getHeap(), offset, prevReclaimIndex, maxLen); Util.arrayCopyNonAtomic(repository.getHeap(), offset, apduBuf, apduOff, len); //release memory repository.reclaimMemory(maxLen); @@ -4432,14 +4425,4 @@ private void finishTrustedConfirmationOperation(KMOperationState op) { } } } - - // Enable this when doing heap profiling. - /*private void processGetHeapProfileData(APDU apdu) { - // No Arguments - // prepare the response - short resp = KMArray.instance((short) 2); - KMArray.cast(resp).add((short) 0, KMInteger.uint_16(KMError.OK)); - KMArray.cast(resp).add((short) 1, KMInteger.uint_16(repository.getMaxHeapUsed())); - sendOutgoing(apdu, resp); - }*/ } diff --git a/Applet/src/com/android/javacard/keymaster/KMRepository.java b/Applet/src/com/android/javacard/keymaster/KMRepository.java index 98733975..380d63dc 100644 --- a/Applet/src/com/android/javacard/keymaster/KMRepository.java +++ b/Applet/src/com/android/javacard/keymaster/KMRepository.java @@ -33,14 +33,12 @@ */ public class KMRepository { - public static final short HEAP_SIZE = 8500; + public static final short HEAP_SIZE = 10000; // Class Attributes private byte[] heap; private short[] heapIndex; private static short[] reclaimIndex; - // Enable this when doing heap profiling. - //public static short maxHeapUsage; // Singleton instance private static KMRepository repository; @@ -86,8 +84,6 @@ public short allocReclaimableMemory(short length) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } reclaimIndex[0] -= length; - // Enable this when doing heap profiling. - //updateHeapProfileData((short) 0); return reclaimIndex[0]; } @@ -125,8 +121,6 @@ public short alloc(short length) { ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED); } heapIndex[0] += length; - // Enable this when doing heap profiling. - //updateHeapProfileData((short) 0); return (short) (heapIndex[0] - length); } @@ -137,24 +131,8 @@ public byte[] getHeap() { public short getHeapIndex() { return heapIndex[0]; } - - // Enable this when doing heap profiling. -/* - public void updateHeapProfileData(short size) { - short totalHeapConsumed = 0; - if (size > 0) { - // This case comes while sending output. - totalHeapConsumed = size; - } else { - totalHeapConsumed = (short) (heapIndex[0] + (HEAP_SIZE - reclaimIndex[0])); - } - if(totalHeapConsumed > maxHeapUsage) { - maxHeapUsage = totalHeapConsumed; - } - } - public short getMaxHeapUsed() { - return maxHeapUsage; + public short getHeapReclaimIndex() { + return reclaimIndex[0]; } - */ } diff --git a/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java b/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java index 1682146d..87ad8541 100644 --- a/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java +++ b/Applet/src/com/android/javacard/keymaster/RemotelyProvisionedComponentDevice.java @@ -737,7 +737,7 @@ private void constructPartialPubKeysToSignMac(byte[] scratchPad, short arrayLeng bufIndex, (short) 3); partialPayloadLen += - encoder.encode(arrPtr, repository.getHeap(), (short) (bufIndex + partialPayloadLen)); + encoder.encode(arrPtr, repository.getHeap(), (short) (bufIndex + partialPayloadLen), repository.getHeapReclaimIndex()); Util.arrayCopyNonAtomic(repository.getHeap(), bufIndex, scratchPad, len, partialPayloadLen); ((KMOperation) operation[0]).update(scratchPad, (short) 0, (short) (len + partialPayloadLen)); } From cf7dd4a9cf4a0c128eaa3ac519e5ed16df5180a0 Mon Sep 17 00:00:00 2001 From: Subrahmanyaman Date: Tue, 19 Apr 2022 19:21:48 +0000 Subject: [PATCH 9/9] Added Comments --- .../javacard/keymaster/KMConfigurations.java | 2 ++ .../javacard/keymaster/KMConfigurations.java | 2 ++ .../android/javacard/keymaster/KMEncoder.java | 15 ++++++++++++-- .../javacard/keymaster/KMKeymasterApplet.java | 20 +++++++++++-------- 4 files changed, 29 insertions(+), 10 deletions(-) diff --git a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java index 3085a4f1..b706d5ed 100644 --- a/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/AndroidSEProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,5 +20,7 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; + // If the size of the attestation ids is known and lesser than 64 + // then reduce the size here. It reduces the heap memory usage. public static final byte MAX_ATTESTATION_IDS_SIZE = 64; } diff --git a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java index 3085a4f1..b706d5ed 100644 --- a/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java +++ b/Applet/JCardSimProvider/src/com/android/javacard/keymaster/KMConfigurations.java @@ -20,5 +20,7 @@ public class KMConfigurations { public static final byte LITTLE_ENDIAN = 0x00; public static final byte BIG_ENDIAN = 0x01; public static final byte TEE_MACHINE_TYPE = LITTLE_ENDIAN; + // If the size of the attestation ids is known and lesser than 64 + // then reduce the size here. It reduces the heap memory usage. public static final byte MAX_ATTESTATION_IDS_SIZE = 64; } diff --git a/Applet/src/com/android/javacard/keymaster/KMEncoder.java b/Applet/src/com/android/javacard/keymaster/KMEncoder.java index a8d9e328..25158a33 100644 --- a/Applet/src/com/android/javacard/keymaster/KMEncoder.java +++ b/Applet/src/com/android/javacard/keymaster/KMEncoder.java @@ -77,8 +77,19 @@ private void encode(short obj) { push(obj); } - // Use this function, when the max len is given - public short encode(short object, byte[] buffer, short startOff, short bufLen, short encoderOutLimitLen) { + /** + * This functions encodes the given object into the provider buffer space + * in cbor format. + * + * @param object Object to be encoded into cbor data. + * @param buffer Output where cbor data is copied. + * @param startOff is the start offset of the buffer. + * @param bufLen length of the buffer + * @param encoderOutLimitLen excepted encoded output length. + * @return length of the encoded buffer. + */ + public short encode(short object, byte[] buffer, short startOff, short bufLen, + short encoderOutLimitLen) { scratchBuf[STACK_PTR_OFFSET] = 0; bufferRef[0] = buffer; scratchBuf[START_OFFSET] = startOff; diff --git a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java index c6d5f238..c4632d39 100644 --- a/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java +++ b/Applet/src/com/android/javacard/keymaster/KMKeymasterApplet.java @@ -1072,10 +1072,12 @@ private void processUpgradeKeyCmd(APDU apdu) { upgradeKeyBlobKeyCharacteristics(data[HW_PARAMETERS], scratchPad); // create new key blob with current os version etc. createEncryptedKeyBlob(scratchPad); - // allocate reclaimable memory. - short buffer = repository.alloc(MAX_KEYBLOB_SIZE); - data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer, repository.getHeapReclaimIndex()); - data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, data[KEY_BLOB]); + short prevReclaimIndex = repository.getHeapReclaimIndex(); + short offset = repository.allocReclaimableMemory(MAX_KEYBLOB_SIZE); + data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), offset, + prevReclaimIndex, MAX_KEYBLOB_SIZE); + data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), offset, data[KEY_BLOB]); + repository.reclaimMemory(MAX_KEYBLOB_SIZE); } else { data[KEY_BLOB] = KMByteBlob.instance((short) 0); } @@ -4194,10 +4196,12 @@ public static void generateRkpKey(byte[] scratchPad, short keyParams) { data[ORIGIN] = KMType.GENERATED; makeKeyCharacteristics(scratchPad); createEncryptedKeyBlob(scratchPad); - // allocate reclaimable memory. - short buffer = repository.alloc((short) 1024); - short keyBlob = encoder.encode(data[KEY_BLOB], repository.getHeap(), buffer, repository.getHeapReclaimIndex()); - data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), buffer, keyBlob); + short prevReclaimIndex = repository.getHeapReclaimIndex(); + short offset = repository.allocReclaimableMemory(MAX_KEYBLOB_SIZE); + data[KEY_BLOB] = encoder.encode(data[KEY_BLOB], repository.getHeap(), offset, + prevReclaimIndex, MAX_KEYBLOB_SIZE); + data[KEY_BLOB] = KMByteBlob.instance(repository.getHeap(), offset, data[KEY_BLOB]); + repository.reclaimMemory(MAX_KEYBLOB_SIZE); } public static short getPubKey() { return data[PUB_KEY];