1 /* 2 * Copyright (C) 2015 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 package com.android.systemui.statusbar.policy; 17 18 import static com.android.systemui.statusbar.policy.NetworkControllerImpl.TAG; 19 20 import android.annotation.NonNull; 21 import android.content.Context; 22 import android.util.Log; 23 24 import com.android.systemui.statusbar.policy.NetworkController.SignalCallback; 25 26 import java.io.PrintWriter; 27 import java.text.SimpleDateFormat; 28 import java.util.BitSet; 29 30 31 /** 32 * Common base class for handling signal for both wifi and mobile data. 33 */ 34 public abstract class SignalController<T extends SignalController.State, 35 I extends SignalController.IconGroup> { 36 // Save the previous SignalController.States of all SignalControllers for dumps. 37 static final boolean RECORD_HISTORY = true; 38 // If RECORD_HISTORY how many to save, must be a power of 2. 39 static final int HISTORY_SIZE = 64; 40 41 protected static final boolean DEBUG = NetworkControllerImpl.DEBUG; 42 protected static final boolean CHATTY = NetworkControllerImpl.CHATTY; 43 44 protected final String mTag; 45 protected final T mCurrentState; 46 protected final T mLastState; 47 protected final int mTransportType; 48 protected final Context mContext; 49 // The owner of the SignalController (i.e. NetworkController will maintain the following 50 // lists and call notifyListeners whenever the list has changed to ensure everyone 51 // is aware of current state. 52 protected final NetworkControllerImpl mNetworkController; 53 54 private final CallbackHandler mCallbackHandler; 55 56 // Save the previous HISTORY_SIZE states for logging. 57 private final State[] mHistory; 58 // Where to copy the next state into. 59 private int mHistoryIndex; 60 SignalController(String tag, Context context, int type, CallbackHandler callbackHandler, NetworkControllerImpl networkController)61 public SignalController(String tag, Context context, int type, CallbackHandler callbackHandler, 62 NetworkControllerImpl networkController) { 63 mTag = TAG + "." + tag; 64 mNetworkController = networkController; 65 mTransportType = type; 66 mContext = context; 67 mCallbackHandler = callbackHandler; 68 mCurrentState = cleanState(); 69 mLastState = cleanState(); 70 if (RECORD_HISTORY) { 71 mHistory = new State[HISTORY_SIZE]; 72 for (int i = 0; i < HISTORY_SIZE; i++) { 73 mHistory[i] = cleanState(); 74 } 75 } 76 } 77 getState()78 public T getState() { 79 return mCurrentState; 80 } 81 updateConnectivity(BitSet connectedTransports, BitSet validatedTransports)82 public void updateConnectivity(BitSet connectedTransports, BitSet validatedTransports) { 83 mCurrentState.inetCondition = validatedTransports.get(mTransportType) ? 1 : 0; 84 notifyListenersIfNecessary(); 85 } 86 87 /** 88 * Used at the end of demo mode to clear out any ugly state that it has created. 89 * Since we haven't had any callbacks, then isDirty will not have been triggered, 90 * so we can just take the last good state directly from there. 91 * 92 * Used for demo mode. 93 */ resetLastState()94 public void resetLastState() { 95 mCurrentState.copyFrom(mLastState); 96 } 97 98 /** 99 * Determines if the state of this signal controller has changed and 100 * needs to trigger callbacks related to it. 101 */ isDirty()102 public boolean isDirty() { 103 if (!mLastState.equals(mCurrentState)) { 104 if (DEBUG) { 105 Log.d(mTag, "Change in state from: " + mLastState + "\n" 106 + "\tto: " + mCurrentState); 107 } 108 return true; 109 } 110 return false; 111 } 112 saveLastState()113 public void saveLastState() { 114 if (RECORD_HISTORY) { 115 recordLastState(); 116 } 117 // Updates the current time. 118 mCurrentState.time = System.currentTimeMillis(); 119 mLastState.copyFrom(mCurrentState); 120 } 121 122 /** 123 * Gets the signal icon for QS based on current state of connected, enabled, and level. 124 */ getQsCurrentIconId()125 public int getQsCurrentIconId() { 126 if (mCurrentState.connected) { 127 return getIcons().mQsIcons[mCurrentState.inetCondition][mCurrentState.level]; 128 } else if (mCurrentState.enabled) { 129 return getIcons().mQsDiscState; 130 } else { 131 return getIcons().mQsNullState; 132 } 133 } 134 135 /** 136 * Gets the signal icon for SB based on current state of connected, enabled, and level. 137 */ getCurrentIconId()138 public int getCurrentIconId() { 139 if (mCurrentState.connected) { 140 return getIcons().mSbIcons[mCurrentState.inetCondition][mCurrentState.level]; 141 } else if (mCurrentState.enabled) { 142 return getIcons().mSbDiscState; 143 } else { 144 return getIcons().mSbNullState; 145 } 146 } 147 148 /** 149 * Gets the content description id for the signal based on current state of connected and 150 * level. 151 */ getContentDescription()152 public int getContentDescription() { 153 if (mCurrentState.connected) { 154 return getIcons().mContentDesc[mCurrentState.level]; 155 } else { 156 return getIcons().mDiscContentDesc; 157 } 158 } 159 notifyListenersIfNecessary()160 public void notifyListenersIfNecessary() { 161 if (isDirty()) { 162 saveLastState(); 163 notifyListeners(); 164 } 165 } 166 167 /** 168 * Returns the resource if resId is not 0, and an empty string otherwise. 169 */ getTextIfExists(int resId)170 @NonNull CharSequence getTextIfExists(int resId) { 171 return resId != 0 ? mContext.getText(resId) : ""; 172 } 173 getIcons()174 protected I getIcons() { 175 return (I) mCurrentState.iconGroup; 176 } 177 178 /** 179 * Saves the last state of any changes, so we can log the current 180 * and last value of any state data. 181 */ recordLastState()182 protected void recordLastState() { 183 mHistory[mHistoryIndex++ & (HISTORY_SIZE - 1)].copyFrom(mLastState); 184 } 185 dump(PrintWriter pw)186 public void dump(PrintWriter pw) { 187 pw.println(" - " + mTag + " -----"); 188 pw.println(" Current State: " + mCurrentState); 189 if (RECORD_HISTORY) { 190 // Count up the states that actually contain time stamps, and only display those. 191 int size = 0; 192 for (int i = 0; i < HISTORY_SIZE; i++) { 193 if (mHistory[i].time != 0) size++; 194 } 195 // Print out the previous states in ordered number. 196 for (int i = mHistoryIndex + HISTORY_SIZE - 1; 197 i >= mHistoryIndex + HISTORY_SIZE - size; i--) { 198 pw.println(" Previous State(" + (mHistoryIndex + HISTORY_SIZE - i) + "): " 199 + mHistory[i & (HISTORY_SIZE - 1)]); 200 } 201 } 202 } 203 notifyListeners()204 public final void notifyListeners() { 205 notifyListeners(mCallbackHandler); 206 } 207 208 /** 209 * Trigger callbacks based on current state. The callbacks should be completely 210 * based on current state, and only need to be called in the scenario where 211 * mCurrentState != mLastState. 212 */ notifyListeners(SignalCallback callback)213 public abstract void notifyListeners(SignalCallback callback); 214 215 /** 216 * Generate a blank T. 217 */ cleanState()218 protected abstract T cleanState(); 219 220 /* 221 * Holds icons for a given state. Arrays are generally indexed as inet 222 * state (full connectivity or not) first, and second dimension as 223 * signal strength. 224 */ 225 static class IconGroup { 226 final int[][] mSbIcons; 227 final int[][] mQsIcons; 228 final int[] mContentDesc; 229 final int mSbNullState; 230 final int mQsNullState; 231 final int mSbDiscState; 232 final int mQsDiscState; 233 final int mDiscContentDesc; 234 // For logging. 235 final String mName; 236 IconGroup(String name, int[][] sbIcons, int[][] qsIcons, int[] contentDesc, int sbNullState, int qsNullState, int sbDiscState, int qsDiscState, int discContentDesc)237 public IconGroup(String name, int[][] sbIcons, int[][] qsIcons, int[] contentDesc, 238 int sbNullState, int qsNullState, int sbDiscState, int qsDiscState, 239 int discContentDesc) { 240 mName = name; 241 mSbIcons = sbIcons; 242 mQsIcons = qsIcons; 243 mContentDesc = contentDesc; 244 mSbNullState = sbNullState; 245 mQsNullState = qsNullState; 246 mSbDiscState = sbDiscState; 247 mQsDiscState = qsDiscState; 248 mDiscContentDesc = discContentDesc; 249 } 250 251 @Override toString()252 public String toString() { 253 return "IconGroup(" + mName + ")"; 254 } 255 } 256 257 static class State { 258 // No locale as it's only used for logging purposes 259 private static SimpleDateFormat sSDF = new SimpleDateFormat("MM-dd HH:mm:ss.SSS"); 260 boolean connected; 261 boolean enabled; 262 boolean activityIn; 263 boolean activityOut; 264 int level; 265 IconGroup iconGroup; 266 int inetCondition; 267 int rssi; // Only for logging. 268 269 // Not used for comparison, just used for logging. 270 long time; 271 copyFrom(State state)272 public void copyFrom(State state) { 273 connected = state.connected; 274 enabled = state.enabled; 275 level = state.level; 276 iconGroup = state.iconGroup; 277 inetCondition = state.inetCondition; 278 activityIn = state.activityIn; 279 activityOut = state.activityOut; 280 rssi = state.rssi; 281 time = state.time; 282 } 283 284 @Override toString()285 public String toString() { 286 if (time != 0) { 287 StringBuilder builder = new StringBuilder(); 288 toString(builder); 289 return builder.toString(); 290 } else { 291 return "Empty " + getClass().getSimpleName(); 292 } 293 } 294 toString(StringBuilder builder)295 protected void toString(StringBuilder builder) { 296 builder.append("connected=").append(connected).append(',') 297 .append("enabled=").append(enabled).append(',') 298 .append("level=").append(level).append(',') 299 .append("inetCondition=").append(inetCondition).append(',') 300 .append("iconGroup=").append(iconGroup).append(',') 301 .append("activityIn=").append(activityIn).append(',') 302 .append("activityOut=").append(activityOut).append(',') 303 .append("rssi=").append(rssi).append(',') 304 .append("lastModified=").append(sSDF.format(time)); 305 } 306 307 @Override equals(Object o)308 public boolean equals(Object o) { 309 if (!o.getClass().equals(getClass())) { 310 return false; 311 } 312 State other = (State) o; 313 return other.connected == connected 314 && other.enabled == enabled 315 && other.level == level 316 && other.inetCondition == inetCondition 317 && other.iconGroup == iconGroup 318 && other.activityIn == activityIn 319 && other.activityOut == activityOut 320 && other.rssi == rssi; 321 } 322 } 323 } 324