blob: b1b6e7533cc05bc8cea5c80de26993f5d75504e0 [file] [log] [blame]
/*
* Copyright (C) 2006 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.internal.telephony.uicc;
import android.os.Build;
/**
* {@hide}
*/
public class
IccIoResult {
private static final String UNKNOWN_ERROR = "unknown";
private String getErrorString() {
// Errors from 3gpp 11.11 9.4.1
// Additional Errors from ETSI 102.221
//
// All error codes below are copied directly from their respective specification
// without modification except in cases where necessary string formatting has been omitted.
switch(sw1) {
case 0x62:
switch(sw2) {
case 0x00: return "No information given,"
+ " state of non volatile memory unchanged";
case 0x81: return "Part of returned data may be corrupted";
case 0x82: return "End of file/record reached before reading Le bytes";
case 0x83: return "Selected file invalidated";
case 0x84: return "Selected file in termination state";
case 0xF1: return "More data available";
case 0xF2: return "More data available and proactive command pending";
case 0xF3: return "Response data available";
}
break;
case 0x63:
if (sw2 >> 4 == 0x0C) {
return "Command successful but after using an internal"
+ "update retry routine but Verification failed";
}
switch(sw2) {
case 0xF1: return "More data expected";
case 0xF2: return "More data expected and proactive command pending";
}
break;
case 0x64:
switch(sw2) {
case 0x00: return "No information given,"
+ " state of non-volatile memory unchanged";
}
break;
case 0x65:
switch(sw2) {
case 0x00: return "No information given, state of non-volatile memory changed";
case 0x81: return "Memory problem";
}
break;
case 0x67:
switch(sw2) {
case 0x00: return "incorrect parameter P3";
default: return "The interpretation of this status word is command dependent";
}
// break;
case 0x6B: return "incorrect parameter P1 or P2";
case 0x6D: return "unknown instruction code given in the command";
case 0x6E: return "wrong instruction class given in the command";
case 0x6F:
switch(sw2) {
case 0x00: return "technical problem with no diagnostic given";
default: return "The interpretation of this status word is command dependent";
}
// break;
case 0x68:
switch(sw2) {
case 0x00: return "No information given";
case 0x81: return "Logical channel not supported";
case 0x82: return "Secure messaging not supported";
}
break;
case 0x69:
switch(sw2) {
case 0x00: return "No information given";
case 0x81: return "Command incompatible with file structure";
case 0x82: return "Security status not satisfied";
case 0x83: return "Authentication/PIN method blocked";
case 0x84: return "Referenced data invalidated";
case 0x85: return "Conditions of use not satisfied";
case 0x86: return "Command not allowed (no EF selected)";
case 0x89: return "Command not allowed - secure channel -"
+ " security not satisfied";
}
break;
case 0x6A:
switch(sw2) {
case 0x80: return "Incorrect parameters in the data field";
case 0x81: return "Function not supported";
case 0x82: return "File not found";
case 0x83: return "Record not found";
case 0x84: return "Not enough memory space";
case 0x86: return "Incorrect parameters P1 to P2";
case 0x87: return "Lc inconsistent with P1 to P2";
case 0x88: return "Referenced data not found";
}
break;
case 0x90: return null; // success
case 0x91: return null; // success
//Status Code 0x92 has contradictory meanings from 11.11 and 102.221 10.2.1.1
case 0x92:
if (sw2 >> 4 == 0) {
return "command successful but after using an internal update retry routine";
}
switch(sw2) {
case 0x40: return "memory problem";
}
break;
case 0x93:
switch(sw2) {
case 0x00:
return "SIM Application Toolkit is busy. Command cannot be executed"
+ " at present, further normal commands are allowed.";
}
break;
case 0x94:
switch(sw2) {
case 0x00: return "no EF selected";
case 0x02: return "out f range (invalid address)";
case 0x04: return "file ID not found/pattern not found";
case 0x08: return "file is inconsistent with the command";
}
break;
case 0x98:
switch(sw2) {
case 0x02: return "no CHV initialized";
case 0x04: return "access condition not fulfilled/"
+ "unsuccessful CHV verification, at least one attempt left/"
+ "unsuccessful UNBLOCK CHV verification, at least one attempt left/"
+ "authentication failed";
case 0x08: return "in contradiction with CHV status";
case 0x10: return "in contradiction with invalidation status";
case 0x40: return "unsuccessful CHV verification, no attempt left/"
+ "unsuccessful UNBLOCK CHV verification, no attempt left/"
+ "CHV blocked"
+ "UNBLOCK CHV blocked";
case 0x50: return "increase cannot be performed, Max value reached";
// The definition for these status codes can be found in TS 31.102 7.3.1
case 0x62: return "authentication error, application specific";
case 0x64: return "authentication error, security context not supported";
case 0x65: return "key freshness failure";
case 0x66: return "authentication error, no memory space available";
case 0x67: return "authentication error, no memory space available in EF_MUK";
}
break;
case 0x9E: return null; // success
case 0x9F: return null; // success
}
return UNKNOWN_ERROR;
}
public int sw1;
public int sw2;
public byte[] payload;
public IccIoResult(int sw1, int sw2, byte[] payload) {
this.sw1 = sw1;
this.sw2 = sw2;
this.payload = payload;
}
public IccIoResult(int sw1, int sw2, String hexString) {
this(sw1, sw2, IccUtils.hexStringToBytes(hexString));
}
@Override
public String toString() {
return "IccIoResult sw1:0x" + Integer.toHexString(sw1) + " sw2:0x"
+ Integer.toHexString(sw2) + " Payload: "
+ ((Build.IS_DEBUGGABLE && Build.IS_ENG) ? payload : "*******")
+ ((!success()) ? " Error: " + getErrorString() : "");
}
/**
* true if this operation was successful
* See GSM 11.11 Section 9.4
* (the fun stuff is absent in 51.011)
*/
public boolean success() {
return sw1 == 0x90 || sw1 == 0x91 || sw1 == 0x9e || sw1 == 0x9f;
}
/**
* Returns exception on error or null if success
*/
public IccException getException() {
if (success()) return null;
switch (sw1) {
case 0x94:
if (sw2 == 0x08) {
return new IccFileTypeMismatch();
} else {
return new IccFileNotFound();
}
default:
return new IccException("sw1:" + sw1 + " sw2:" + sw2);
}
}
}