| /* |
| * 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 |
| + "]"; |
| } |
| } |