blob: 04c3abedd6bc57283647d27655d0c39595dfc6e8 [file] [log] [blame]
/*
* Copyright(C) 2021 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.javacard.keymaster;
import javacard.framework.ISO7816;
import javacard.framework.ISOException;
import javacard.framework.Util;
/**
* KMCosePairByteBlobTag represents a key-value type, where key can be KMInteger or KMNInteger and
* value is KMByteBlob type. struct{byte TAG_TYPE; short length; struct{short BYTE_BLOB_TYPE; short
* key; short value}}.
*/
public class KMCosePairByteBlobTag extends KMCosePairTagType {
public static Object[] keys;
private static KMCosePairByteBlobTag prototype;
private KMCosePairByteBlobTag() {}
private static KMCosePairByteBlobTag proto(short ptr) {
if (prototype == null) {
prototype = new KMCosePairByteBlobTag();
}
instanceTable[KM_COSE_KEY_BYTE_BLOB_VAL_OFFSET] = ptr;
return prototype;
}
// pointer to an empty instance used as expression
public static short exp() {
short ptr = instance(COSE_PAIR_TAG_TYPE, (short) 6);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE), KMType.COSE_PAIR_BYTE_BLOB_TAG_TYPE);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE + 2), KMType.INVALID_VALUE);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE + 4), KMByteBlob.exp());
return ptr;
}
public static short instance(short keyPtr, short valuePtr) {
if (!isKeyValueValid(keyPtr)) {
ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED);
}
if (KMType.getType(valuePtr) != BYTE_BLOB_TYPE) {
ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED);
}
short ptr = KMType.instance(COSE_PAIR_TAG_TYPE, (short) 6);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE), KMType.COSE_PAIR_BYTE_BLOB_TAG_TYPE);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE + 2), keyPtr);
Util.setShort(heap, (short) (ptr + TLV_HEADER_SIZE + 4), valuePtr);
return ptr;
}
public static KMCosePairByteBlobTag cast(short ptr) {
byte[] heap = repository.getHeap();
if (heap[ptr] != COSE_PAIR_TAG_TYPE) {
ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED);
}
// Validate the value pointer.
short valuePtr = Util.getShort(heap, (short) (ptr + TLV_HEADER_SIZE + 4));
if (KMType.getType(valuePtr) != BYTE_BLOB_TYPE) {
ISOException.throwIt(ISO7816.SW_CONDITIONS_NOT_SATISFIED);
}
return proto(ptr);
}
private static void createKeys() {
if (keys == null) {
keys =
new Object[] {
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_KEY_PUBKEY_X},
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_KEY_PUBKEY_Y},
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_KEY_PRIV_KEY},
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_LABEL_IV},
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_LABEL_KEYID},
(Object) new byte[] {(byte) 0, (byte) 0, (byte) 0, KMCose.COSE_KEY_KEY_ID},
(Object) KMCose.SUBJECT_PUBLIC_KEY,
(Object) KMCose.KEY_USAGE
};
}
}
public static boolean isKeyValueValid(short keyPtr) {
createKeys();
short type = KMType.getType(keyPtr);
short offset = 0;
if (type == INTEGER_TYPE) {
offset = KMInteger.cast(keyPtr).getStartOff();
} else if (type == NEG_INTEGER_TYPE) {
offset = KMNInteger.cast(keyPtr).getStartOff();
} else {
ISOException.throwIt(ISO7816.SW_DATA_INVALID);
}
short index = 0;
while (index < (short) keys.length) {
if (0
== Util.arrayCompare(
(byte[]) keys[index],
(short) 0,
heap,
offset,
(short) ((byte[]) keys[index]).length)) {
return true;
}
index++;
}
return false;
}
public short getValueType() {
return BYTE_BLOB_TYPE;
}
@Override
public short getKeyPtr() {
return Util.getShort(
heap, (short) (instanceTable[KM_COSE_KEY_BYTE_BLOB_VAL_OFFSET] + TLV_HEADER_SIZE + 2));
}
@Override
public short getValuePtr() {
return Util.getShort(
heap, (short) (instanceTable[KM_COSE_KEY_BYTE_BLOB_VAL_OFFSET] + TLV_HEADER_SIZE + 4));
}
}