184 lines
5.7 KiB
Java
184 lines
5.7 KiB
Java
/*
|
|
* Copyright (C) 2017 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 android.telephony;
|
|
|
|
import android.annotation.NonNull;
|
|
import android.annotation.Nullable;
|
|
import android.annotation.SystemApi;
|
|
import android.os.Parcel;
|
|
import android.os.Parcelable;
|
|
import android.util.Log;
|
|
|
|
import java.security.KeyFactory;
|
|
import java.security.NoSuchAlgorithmException;
|
|
import java.security.PublicKey;
|
|
import java.security.spec.InvalidKeySpecException;
|
|
import java.security.spec.X509EncodedKeySpec;
|
|
import java.util.Date;
|
|
|
|
/**
|
|
* Class to represent information sent by the carrier, which will be used to encrypt
|
|
* the IMSI + IMPI. The ecryption is being done by WLAN, and the modem.
|
|
* @hide
|
|
*/
|
|
@SystemApi
|
|
public final class ImsiEncryptionInfo implements Parcelable {
|
|
|
|
private static final String LOG_TAG = "ImsiEncryptionInfo";
|
|
|
|
private final String mcc;
|
|
private final String mnc;
|
|
private final PublicKey publicKey;
|
|
private final String keyIdentifier;
|
|
private final int keyType;
|
|
//Date-Time in UTC when the key will expire.
|
|
private final Date expirationTime;
|
|
private final int carrierId;
|
|
|
|
/** @hide */
|
|
public ImsiEncryptionInfo(String mcc, String mnc, int keyType, String keyIdentifier,
|
|
byte[] key, Date expirationTime, int carrierId) {
|
|
this(mcc, mnc, keyType, keyIdentifier, makeKeyObject(key), expirationTime, carrierId);
|
|
}
|
|
|
|
/** @hide */
|
|
public ImsiEncryptionInfo(String mcc, String mnc, int keyType, String keyIdentifier,
|
|
PublicKey publicKey, Date expirationTime, int carrierId) {
|
|
// todo need to validate that ImsiEncryptionInfo is being created with the correct params.
|
|
// Including validating that the public key is in "X.509" format. This will be done in
|
|
// a subsequent CL.
|
|
this.mcc = mcc;
|
|
this.mnc = mnc;
|
|
this.keyType = keyType;
|
|
this.publicKey = publicKey;
|
|
this.keyIdentifier = keyIdentifier;
|
|
this.expirationTime = expirationTime;
|
|
this.carrierId = carrierId;
|
|
}
|
|
|
|
/** @hide */
|
|
public ImsiEncryptionInfo(Parcel in) {
|
|
int length = in.readInt();
|
|
byte b[] = new byte[length];
|
|
in.readByteArray(b);
|
|
publicKey = makeKeyObject(b);
|
|
mcc = in.readString();
|
|
mnc = in.readString();
|
|
keyIdentifier = in.readString();
|
|
keyType = in.readInt();
|
|
expirationTime = new Date(in.readLong());
|
|
carrierId = in.readInt();
|
|
}
|
|
|
|
/** @hide */
|
|
public String getMnc() {
|
|
return this.mnc;
|
|
}
|
|
|
|
/** @hide */
|
|
public String getMcc() {
|
|
return this.mcc;
|
|
}
|
|
|
|
/** @hide */
|
|
public int getCarrierId() {
|
|
return carrierId;
|
|
}
|
|
|
|
/**
|
|
* Returns key identifier, a string that helps the authentication server to locate the
|
|
* private key to decrypt the permanent identity, or {@code null} when uavailable.
|
|
*/
|
|
@Nullable
|
|
public String getKeyIdentifier() {
|
|
return this.keyIdentifier;
|
|
}
|
|
|
|
/** @hide */
|
|
public int getKeyType() {
|
|
return this.keyType;
|
|
}
|
|
|
|
/**
|
|
* Returns the carrier public key that is used for the IMSI encryption,
|
|
* or {@code null} when uavailable.
|
|
*/
|
|
@Nullable
|
|
public PublicKey getPublicKey() {
|
|
return this.publicKey;
|
|
}
|
|
|
|
/** @hide */
|
|
public Date getExpirationTime() {
|
|
return this.expirationTime;
|
|
}
|
|
|
|
private static PublicKey makeKeyObject(byte[] publicKeyBytes) {
|
|
try {
|
|
X509EncodedKeySpec pubKeySpec = new X509EncodedKeySpec(publicKeyBytes);
|
|
return KeyFactory.getInstance("RSA").generatePublic(pubKeySpec);
|
|
} catch (InvalidKeySpecException | NoSuchAlgorithmException ex) {
|
|
Log.e(LOG_TAG, "Error makeKeyObject: unable to convert into PublicKey", ex);
|
|
}
|
|
throw new IllegalArgumentException();
|
|
}
|
|
|
|
/** Implement the Parcelable interface */
|
|
@Override
|
|
public int describeContents() {
|
|
return 0;
|
|
}
|
|
|
|
public static final @NonNull Parcelable.Creator<ImsiEncryptionInfo> CREATOR =
|
|
new Parcelable.Creator<ImsiEncryptionInfo>() {
|
|
@Override
|
|
public ImsiEncryptionInfo createFromParcel(Parcel in) {
|
|
return new ImsiEncryptionInfo(in);
|
|
}
|
|
|
|
@Override
|
|
public ImsiEncryptionInfo[] newArray(int size) {
|
|
return new ImsiEncryptionInfo[size];
|
|
}
|
|
};
|
|
|
|
@Override
|
|
public void writeToParcel(@NonNull Parcel dest, int flags) {
|
|
byte[] b = publicKey.getEncoded();
|
|
dest.writeInt(b.length);
|
|
dest.writeByteArray(b);
|
|
dest.writeString(mcc);
|
|
dest.writeString(mnc);
|
|
dest.writeString(keyIdentifier);
|
|
dest.writeInt(keyType);
|
|
dest.writeLong(expirationTime.getTime());
|
|
dest.writeInt(carrierId);
|
|
}
|
|
|
|
@Override
|
|
public String toString(){
|
|
return "[ImsiEncryptionInfo "
|
|
+ "mcc=" + mcc
|
|
+ " mnc=" + mnc
|
|
+ ", publicKey=" + publicKey
|
|
+ ", keyIdentifier=" + keyIdentifier
|
|
+ ", keyType=" + keyType
|
|
+ ", expirationTime=" + expirationTime
|
|
+ ", carrier_id=" + carrierId
|
|
+ "]";
|
|
}
|
|
}
|