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