1 /* 2 * Copyright (C) 2014 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 android.net; 18 19 import android.annotation.SystemApi; 20 import android.os.Parcel; 21 import android.os.Parcelable; 22 23 import java.util.Arrays; 24 import java.util.Objects; 25 26 /** 27 * A curve defining the network score over a range of RSSI values. 28 * 29 * <p>For each RSSI bucket, the score may be any byte. Scores have no absolute meaning and are only 30 * considered relative to other scores assigned by the same scorer. Networks with no score are 31 * treated equivalently to a network with score {@link Byte#MIN_VALUE}, and will not be used. 32 * 33 * <p>For example, consider a curve starting at -110 dBm with a bucket width of 10 and the 34 * following buckets: {@code [-20, -10, 0, 10, 20, 30, 40, 50, 60, 70, 80, 90, 100, 110, 120]}. 35 * This represents a linear curve between -110 dBm and 30 dBm. It scores progressively higher at 36 * stronger signal strengths. 37 * 38 * <p>A network can be assigned a fixed score independent of RSSI by setting 39 * {@link #rssiBuckets} to a one-byte array whose element is the fixed score. {@link #start} 40 * should be set to the lowest RSSI value at which this fixed score should apply, and 41 * {@link #bucketWidth} should be set such that {@code start + bucketWidth} is equal to the 42 * highest RSSI value at which this fixed score should apply. 43 * 44 * <p>Note that RSSI values below -110 dBm or above 30 dBm are unlikely to cause any difference 45 * in connectivity behavior from those endpoints. That is, the connectivity framework will treat 46 * a network with a -120 dBm signal exactly as it would treat one with a -110 dBm signal. 47 * Therefore, graphs which specify scores outside this range may be truncated to this range by 48 * the system. 49 * 50 * @see ScoredNetwork 51 * @hide 52 */ 53 @SystemApi 54 public class RssiCurve implements Parcelable { 55 private static final int DEFAULT_ACTIVE_NETWORK_RSSI_BOOST = 25; 56 57 /** The starting dBm of the curve. */ 58 public final int start; 59 60 /** The width of each RSSI bucket, in dBm. */ 61 public final int bucketWidth; 62 63 /** The score for each RSSI bucket. */ 64 public final byte[] rssiBuckets; 65 66 /** 67 * The RSSI boost to give this network when active, in dBm. 68 * 69 * <p>When the system is connected to this network, it will pretend that the network has this 70 * much higher of an RSSI. This is to avoid switching networks when another network has only a 71 * slightly higher score. 72 */ 73 public final int activeNetworkRssiBoost; 74 75 /** 76 * Construct a new {@link RssiCurve}. 77 * 78 * @param start the starting dBm of the curve. 79 * @param bucketWidth the width of each RSSI bucket, in dBm. 80 * @param rssiBuckets the score for each RSSI bucket. 81 */ RssiCurve(int start, int bucketWidth, byte[] rssiBuckets)82 public RssiCurve(int start, int bucketWidth, byte[] rssiBuckets) { 83 this(start, bucketWidth, rssiBuckets, DEFAULT_ACTIVE_NETWORK_RSSI_BOOST); 84 } 85 86 /** 87 * Construct a new {@link RssiCurve}. 88 * 89 * @param start the starting dBm of the curve. 90 * @param bucketWidth the width of each RSSI bucket, in dBm. 91 * @param rssiBuckets the score for each RSSI bucket. 92 * @param activeNetworkRssiBoost the RSSI boost to apply when this network is active, in dBm. 93 */ RssiCurve(int start, int bucketWidth, byte[] rssiBuckets, int activeNetworkRssiBoost)94 public RssiCurve(int start, int bucketWidth, byte[] rssiBuckets, int activeNetworkRssiBoost) { 95 this.start = start; 96 this.bucketWidth = bucketWidth; 97 if (rssiBuckets == null || rssiBuckets.length == 0) { 98 throw new IllegalArgumentException("rssiBuckets must be at least one element large."); 99 } 100 this.rssiBuckets = rssiBuckets; 101 this.activeNetworkRssiBoost = activeNetworkRssiBoost; 102 } 103 RssiCurve(Parcel in)104 private RssiCurve(Parcel in) { 105 start = in.readInt(); 106 bucketWidth = in.readInt(); 107 int bucketCount = in.readInt(); 108 rssiBuckets = new byte[bucketCount]; 109 in.readByteArray(rssiBuckets); 110 activeNetworkRssiBoost = in.readInt(); 111 } 112 113 @Override describeContents()114 public int describeContents() { 115 return 0; 116 } 117 118 @Override writeToParcel(Parcel out, int flags)119 public void writeToParcel(Parcel out, int flags) { 120 out.writeInt(start); 121 out.writeInt(bucketWidth); 122 out.writeInt(rssiBuckets.length); 123 out.writeByteArray(rssiBuckets); 124 out.writeInt(activeNetworkRssiBoost); 125 } 126 127 /** 128 * Lookup the score for a given RSSI value. 129 * 130 * @param rssi The RSSI to lookup. If the RSSI falls below the start of the curve, the score at 131 * the start of the curve will be returned. If it falls after the end of the curve, the 132 * score at the end of the curve will be returned. 133 * @return the score for the given RSSI. 134 */ lookupScore(int rssi)135 public byte lookupScore(int rssi) { 136 return lookupScore(rssi, false /* isActiveNetwork */); 137 } 138 139 /** 140 * Lookup the score for a given RSSI value. 141 * 142 * @param rssi The RSSI to lookup. If the RSSI falls below the start of the curve, the score at 143 * the start of the curve will be returned. If it falls after the end of the curve, the 144 * score at the end of the curve will be returned. 145 * @param isActiveNetwork Whether this network is currently active. 146 * @return the score for the given RSSI. 147 */ lookupScore(int rssi, boolean isActiveNetwork)148 public byte lookupScore(int rssi, boolean isActiveNetwork) { 149 if (isActiveNetwork) { 150 rssi += activeNetworkRssiBoost; 151 } 152 153 int index = (rssi - start) / bucketWidth; 154 155 // Snap the index to the closest bucket if it falls outside the curve. 156 if (index < 0) { 157 index = 0; 158 } else if (index > rssiBuckets.length - 1) { 159 index = rssiBuckets.length - 1; 160 } 161 162 return rssiBuckets[index]; 163 } 164 165 /** 166 * Determine if two RSSI curves are defined in the same way. 167 * 168 * <p>Note that two curves can be equivalent but defined differently, e.g. if one bucket in one 169 * curve is split into two buckets in another. For the purpose of this method, these curves are 170 * not considered equal to each other. 171 */ 172 @Override equals(Object o)173 public boolean equals(Object o) { 174 if (this == o) return true; 175 if (o == null || getClass() != o.getClass()) return false; 176 177 RssiCurve rssiCurve = (RssiCurve) o; 178 179 return start == rssiCurve.start && 180 bucketWidth == rssiCurve.bucketWidth && 181 Arrays.equals(rssiBuckets, rssiCurve.rssiBuckets) && 182 activeNetworkRssiBoost == rssiCurve.activeNetworkRssiBoost; 183 } 184 185 @Override hashCode()186 public int hashCode() { 187 return Objects.hash(start, bucketWidth, activeNetworkRssiBoost) ^ Arrays.hashCode(rssiBuckets); 188 } 189 190 @Override toString()191 public String toString() { 192 StringBuilder sb = new StringBuilder(); 193 sb.append("RssiCurve[start=") 194 .append(start) 195 .append(",bucketWidth=") 196 .append(bucketWidth) 197 .append(",activeNetworkRssiBoost=") 198 .append(activeNetworkRssiBoost); 199 200 sb.append(",buckets="); 201 for (int i = 0; i < rssiBuckets.length; i++) { 202 sb.append(rssiBuckets[i]); 203 if (i < rssiBuckets.length - 1) { 204 sb.append(","); 205 } 206 } 207 sb.append("]"); 208 209 return sb.toString(); 210 } 211 212 public static final Creator<RssiCurve> CREATOR = 213 new Creator<RssiCurve>() { 214 @Override 215 public RssiCurve createFromParcel(Parcel in) { 216 return new RssiCurve(in); 217 } 218 219 @Override 220 public RssiCurve[] newArray(int size) { 221 return new RssiCurve[size]; 222 } 223 }; 224 } 225