1 /* 2 * Copyright (C) 2006 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.internal.telephony.uicc; 18 19 import android.os.AsyncResult; 20 import android.os.Handler; 21 import android.os.Message; 22 import android.util.SparseArray; 23 24 import com.android.internal.telephony.gsm.UsimPhoneBookManager; 25 26 import java.util.ArrayList; 27 import java.util.Iterator; 28 29 /** 30 * {@hide} 31 */ 32 public class AdnRecordCache extends Handler implements IccConstants { 33 //***** Instance Variables 34 35 private IccFileHandler mFh; 36 private UsimPhoneBookManager mUsimPhoneBookManager; 37 38 // Indexed by EF ID 39 SparseArray<ArrayList<AdnRecord>> mAdnLikeFiles 40 = new SparseArray<ArrayList<AdnRecord>>(); 41 42 // People waiting for ADN-like files to be loaded 43 SparseArray<ArrayList<Message>> mAdnLikeWaiters 44 = new SparseArray<ArrayList<Message>>(); 45 46 // People waiting for adn record to be updated 47 SparseArray<Message> mUserWriteResponse = new SparseArray<Message>(); 48 49 //***** Event Constants 50 51 static final int EVENT_LOAD_ALL_ADN_LIKE_DONE = 1; 52 static final int EVENT_UPDATE_ADN_DONE = 2; 53 54 //***** Constructor 55 56 57 AdnRecordCache(IccFileHandler fh)58 AdnRecordCache(IccFileHandler fh) { 59 mFh = fh; 60 mUsimPhoneBookManager = new UsimPhoneBookManager(mFh, this); 61 } 62 63 //***** Called from SIMRecords 64 65 /** 66 * Called from SIMRecords.onRadioNotAvailable and SIMRecords.handleSimRefresh. 67 */ reset()68 public void reset() { 69 mAdnLikeFiles.clear(); 70 mUsimPhoneBookManager.reset(); 71 72 clearWaiters(); 73 clearUserWriters(); 74 75 } 76 clearWaiters()77 private void clearWaiters() { 78 int size = mAdnLikeWaiters.size(); 79 for (int i = 0; i < size; i++) { 80 ArrayList<Message> waiters = mAdnLikeWaiters.valueAt(i); 81 AsyncResult ar = new AsyncResult(null, null, new RuntimeException("AdnCache reset")); 82 notifyWaiters(waiters, ar); 83 } 84 mAdnLikeWaiters.clear(); 85 } 86 clearUserWriters()87 private void clearUserWriters() { 88 int size = mUserWriteResponse.size(); 89 for (int i = 0; i < size; i++) { 90 sendErrorResponse(mUserWriteResponse.valueAt(i), "AdnCace reset"); 91 } 92 mUserWriteResponse.clear(); 93 } 94 95 /** 96 * @return List of AdnRecords for efid if we've already loaded them this 97 * radio session, or null if we haven't 98 */ 99 public ArrayList<AdnRecord> getRecordsIfLoaded(int efid)100 getRecordsIfLoaded(int efid) { 101 return mAdnLikeFiles.get(efid); 102 } 103 104 /** 105 * Returns extension ef associated with ADN-like EF or -1 if 106 * we don't know. 107 * 108 * See 3GPP TS 51.011 for this mapping 109 */ extensionEfForEf(int efid)110 public int extensionEfForEf(int efid) { 111 switch (efid) { 112 case EF_MBDN: return EF_EXT6; 113 case EF_ADN: return EF_EXT1; 114 case EF_SDN: return EF_EXT3; 115 case EF_FDN: return EF_EXT2; 116 case EF_MSISDN: return EF_EXT1; 117 case EF_PBR: return 0; // The EF PBR doesn't have an extension record 118 default: return -1; 119 } 120 } 121 sendErrorResponse(Message response, String errString)122 private void sendErrorResponse(Message response, String errString) { 123 if (response != null) { 124 Exception e = new RuntimeException(errString); 125 AsyncResult.forMessage(response).exception = e; 126 response.sendToTarget(); 127 } 128 } 129 130 /** 131 * Update an ADN-like record in EF by record index 132 * 133 * @param efid must be one among EF_ADN, EF_FDN, and EF_SDN 134 * @param adn is the new adn to be stored 135 * @param recordIndex is the 1-based adn record index 136 * @param pin2 is required to update EF_FDN, otherwise must be null 137 * @param response message to be posted when done 138 * response.exception hold the exception in error 139 */ updateAdnByIndex(int efid, AdnRecord adn, int recordIndex, String pin2, Message response)140 public void updateAdnByIndex(int efid, AdnRecord adn, int recordIndex, String pin2, 141 Message response) { 142 143 int extensionEF = extensionEfForEf(efid); 144 if (extensionEF < 0) { 145 sendErrorResponse(response, "EF is not known ADN-like EF:0x" + 146 Integer.toHexString(efid).toUpperCase()); 147 return; 148 } 149 150 Message pendingResponse = mUserWriteResponse.get(efid); 151 if (pendingResponse != null) { 152 sendErrorResponse(response, "Have pending update for EF:0x" + 153 Integer.toHexString(efid).toUpperCase()); 154 return; 155 } 156 157 mUserWriteResponse.put(efid, response); 158 159 new AdnRecordLoader(mFh).updateEF(adn, efid, extensionEF, 160 recordIndex, pin2, 161 obtainMessage(EVENT_UPDATE_ADN_DONE, efid, recordIndex, adn)); 162 } 163 164 /** 165 * Replace oldAdn with newAdn in ADN-like record in EF 166 * 167 * The ADN-like records must be read through requestLoadAllAdnLike() before 168 * 169 * @param efid must be one of EF_ADN, EF_FDN, and EF_SDN 170 * @param oldAdn is the adn to be replaced 171 * If oldAdn.isEmpty() is ture, it insert the newAdn 172 * @param newAdn is the adn to be stored 173 * If newAdn.isEmpty() is true, it delete the oldAdn 174 * @param pin2 is required to update EF_FDN, otherwise must be null 175 * @param response message to be posted when done 176 * response.exception hold the exception in error 177 */ updateAdnBySearch(int efid, AdnRecord oldAdn, AdnRecord newAdn, String pin2, Message response)178 public void updateAdnBySearch(int efid, AdnRecord oldAdn, AdnRecord newAdn, 179 String pin2, Message response) { 180 181 int extensionEF; 182 extensionEF = extensionEfForEf(efid); 183 184 if (extensionEF < 0) { 185 sendErrorResponse(response, "EF is not known ADN-like EF:0x" + 186 Integer.toHexString(efid).toUpperCase()); 187 return; 188 } 189 190 ArrayList<AdnRecord> oldAdnList; 191 192 if (efid == EF_PBR) { 193 oldAdnList = mUsimPhoneBookManager.loadEfFilesFromUsim(); 194 } else { 195 oldAdnList = getRecordsIfLoaded(efid); 196 } 197 198 if (oldAdnList == null) { 199 sendErrorResponse(response, "Adn list not exist for EF:0x" + 200 Integer.toHexString(efid).toUpperCase()); 201 return; 202 } 203 204 int index = -1; 205 int count = 1; 206 for (Iterator<AdnRecord> it = oldAdnList.iterator(); it.hasNext(); ) { 207 if (oldAdn.isEqual(it.next())) { 208 index = count; 209 break; 210 } 211 count++; 212 } 213 214 if (index == -1) { 215 sendErrorResponse(response, "Adn record don't exist for " + oldAdn); 216 return; 217 } 218 219 if (efid == EF_PBR) { 220 AdnRecord foundAdn = oldAdnList.get(index-1); 221 efid = foundAdn.mEfid; 222 extensionEF = foundAdn.mExtRecord; 223 index = foundAdn.mRecordNumber; 224 225 newAdn.mEfid = efid; 226 newAdn.mExtRecord = extensionEF; 227 newAdn.mRecordNumber = index; 228 } 229 230 Message pendingResponse = mUserWriteResponse.get(efid); 231 232 if (pendingResponse != null) { 233 sendErrorResponse(response, "Have pending update for EF:0x" + 234 Integer.toHexString(efid).toUpperCase()); 235 return; 236 } 237 238 mUserWriteResponse.put(efid, response); 239 240 new AdnRecordLoader(mFh).updateEF(newAdn, efid, extensionEF, 241 index, pin2, 242 obtainMessage(EVENT_UPDATE_ADN_DONE, efid, index, newAdn)); 243 } 244 245 246 /** 247 * Responds with exception (in response) if efid is not a known ADN-like 248 * record 249 */ 250 public void requestLoadAllAdnLike(int efid, int extensionEf, Message response)251 requestLoadAllAdnLike (int efid, int extensionEf, Message response) { 252 ArrayList<Message> waiters; 253 ArrayList<AdnRecord> result; 254 255 if (efid == EF_PBR) { 256 result = mUsimPhoneBookManager.loadEfFilesFromUsim(); 257 } else { 258 result = getRecordsIfLoaded(efid); 259 } 260 261 // Have we already loaded this efid? 262 if (result != null) { 263 if (response != null) { 264 AsyncResult.forMessage(response).result = result; 265 response.sendToTarget(); 266 } 267 268 return; 269 } 270 271 // Have we already *started* loading this efid? 272 273 waiters = mAdnLikeWaiters.get(efid); 274 275 if (waiters != null) { 276 // There's a pending request for this EF already 277 // just add ourselves to it 278 279 waiters.add(response); 280 return; 281 } 282 283 // Start loading efid 284 285 waiters = new ArrayList<Message>(); 286 waiters.add(response); 287 288 mAdnLikeWaiters.put(efid, waiters); 289 290 291 if (extensionEf < 0) { 292 // respond with error if not known ADN-like record 293 294 if (response != null) { 295 AsyncResult.forMessage(response).exception 296 = new RuntimeException("EF is not known ADN-like EF:0x" + 297 Integer.toHexString(efid).toUpperCase()); 298 response.sendToTarget(); 299 } 300 301 return; 302 } 303 304 new AdnRecordLoader(mFh).loadAllFromEF(efid, extensionEf, 305 obtainMessage(EVENT_LOAD_ALL_ADN_LIKE_DONE, efid, 0)); 306 } 307 308 //***** Private methods 309 310 private void notifyWaiters(ArrayList<Message> waiters, AsyncResult ar)311 notifyWaiters(ArrayList<Message> waiters, AsyncResult ar) { 312 313 if (waiters == null) { 314 return; 315 } 316 317 for (int i = 0, s = waiters.size() ; i < s ; i++) { 318 Message waiter = waiters.get(i); 319 320 AsyncResult.forMessage(waiter, ar.result, ar.exception); 321 waiter.sendToTarget(); 322 } 323 } 324 325 //***** Overridden from Handler 326 327 @Override 328 public void handleMessage(Message msg)329 handleMessage(Message msg) { 330 AsyncResult ar; 331 int efid; 332 333 switch(msg.what) { 334 case EVENT_LOAD_ALL_ADN_LIKE_DONE: 335 /* arg1 is efid, obj.result is ArrayList<AdnRecord>*/ 336 ar = (AsyncResult) msg.obj; 337 efid = msg.arg1; 338 ArrayList<Message> waiters; 339 340 waiters = mAdnLikeWaiters.get(efid); 341 mAdnLikeWaiters.delete(efid); 342 343 if (ar.exception == null) { 344 mAdnLikeFiles.put(efid, (ArrayList<AdnRecord>) ar.result); 345 } 346 notifyWaiters(waiters, ar); 347 break; 348 case EVENT_UPDATE_ADN_DONE: 349 ar = (AsyncResult)msg.obj; 350 efid = msg.arg1; 351 int index = msg.arg2; 352 AdnRecord adn = (AdnRecord) (ar.userObj); 353 354 if (ar.exception == null) { 355 mAdnLikeFiles.get(efid).set(index - 1, adn); 356 mUsimPhoneBookManager.invalidateCache(); 357 } 358 359 Message response = mUserWriteResponse.get(efid); 360 mUserWriteResponse.delete(efid); 361 362 // response may be cleared when simrecord is reset, 363 // so we should check if it is null. 364 if (response != null) { 365 AsyncResult.forMessage(response, null, ar.exception); 366 response.sendToTarget(); 367 } 368 break; 369 } 370 } 371 } 372