Merge changes I32c12702,I688593cc
* changes: [NS D04] Implement a simple speed comparison between scores. [NS D03] Migrate the bad wifi avoidance policy
This commit is contained in:
@@ -83,6 +83,65 @@ public final class NetworkScore implements Parcelable {
|
|||||||
uplinkBandwidthKBps = uplinkBandwidth;
|
uplinkBandwidthKBps = uplinkBandwidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evaluate whether a metrics codes for faster network is faster than another.
|
||||||
|
*
|
||||||
|
* This is a simple comparison of expected speeds. If either of the tested attributes
|
||||||
|
* are unknown, this returns zero. This implementation just assumes downlink bandwidth
|
||||||
|
* is more important than uplink bandwidth, which is more important than latency. This
|
||||||
|
* is not a very good way of evaluating network speed, but it's a start.
|
||||||
|
* TODO : do something more representative of how fast the network feels
|
||||||
|
*
|
||||||
|
* @param other the Metrics to evaluate against
|
||||||
|
* @return a negative integer, zero, or a positive integer as this metrics is worse than,
|
||||||
|
* equally good as (or unknown), or better than the passed Metrics.
|
||||||
|
* @see #compareToPreferringKnown(Metrics)
|
||||||
|
* @hide
|
||||||
|
*/
|
||||||
|
// Can't implement Comparable<Metrics> because this is @hide.
|
||||||
|
public int compareTo(@NonNull final Metrics other) {
|
||||||
|
if (downlinkBandwidthKBps != BANDWIDTH_UNKNOWN
|
||||||
|
&& other.downlinkBandwidthKBps != BANDWIDTH_UNKNOWN) {
|
||||||
|
if (downlinkBandwidthKBps > other.downlinkBandwidthKBps) return 1;
|
||||||
|
if (downlinkBandwidthKBps < other.downlinkBandwidthKBps) return -1;
|
||||||
|
}
|
||||||
|
if (uplinkBandwidthKBps != BANDWIDTH_UNKNOWN
|
||||||
|
&& other.uplinkBandwidthKBps != BANDWIDTH_UNKNOWN) {
|
||||||
|
if (uplinkBandwidthKBps > other.uplinkBandwidthKBps) return 1;
|
||||||
|
if (uplinkBandwidthKBps < other.uplinkBandwidthKBps) return -1;
|
||||||
|
}
|
||||||
|
if (latencyMs != LATENCY_UNKNOWN && other.latencyMs != LATENCY_UNKNOWN) {
|
||||||
|
// Latency : lower is better
|
||||||
|
if (latencyMs > other.latencyMs) return -1;
|
||||||
|
if (latencyMs < other.latencyMs) return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evaluate whether a metrics codes for faster network is faster than another.
|
||||||
|
*
|
||||||
|
* This is a simple comparison of expected speeds. If either of the tested attributes
|
||||||
|
* are unknown, this prefers the known attributes. This implementation just assumes
|
||||||
|
* downlink bandwidth is more important than uplink bandwidth, which is more important than
|
||||||
|
* latency. This is not a very good way of evaluating network speed, but it's a start.
|
||||||
|
* TODO : do something more representative of how fast the network feels
|
||||||
|
*
|
||||||
|
* @param other the Metrics to evaluate against
|
||||||
|
* @return a negative integer, zero, or a positive integer as this metrics is worse than,
|
||||||
|
* equally good as (or unknown), or better than the passed Metrics.
|
||||||
|
* @see #compareTo(Metrics)
|
||||||
|
* @hide
|
||||||
|
*/
|
||||||
|
public int compareToPreferringKnown(@NonNull final Metrics other) {
|
||||||
|
if (downlinkBandwidthKBps > other.downlinkBandwidthKBps) return 1;
|
||||||
|
if (downlinkBandwidthKBps < other.downlinkBandwidthKBps) return -1;
|
||||||
|
if (uplinkBandwidthKBps > other.uplinkBandwidthKBps) return 1;
|
||||||
|
if (uplinkBandwidthKBps < other.uplinkBandwidthKBps) return -1;
|
||||||
|
// Latency : lower is better
|
||||||
|
return -Integer.compare(latencyMs, other.latencyMs);
|
||||||
|
}
|
||||||
|
|
||||||
/** toString */
|
/** toString */
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "latency = " + latencyMs + " downlinkBandwidth = " + downlinkBandwidthKBps
|
return "latency = " + latencyMs + " downlinkBandwidth = " + downlinkBandwidthKBps
|
||||||
@@ -346,6 +405,33 @@ public final class NetworkScore implements Parcelable {
|
|||||||
return mLegacyScore;
|
return mLegacyScore;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the metrics to evaluate whether a network is faster than another.
|
||||||
|
*
|
||||||
|
* This is purely based on the metrics, and explicitly ignores policy or exiting. It's
|
||||||
|
* provided to get a decision on two networks when policy can not decide, or to evaluate
|
||||||
|
* how a network is expected to compare to another if it should validate.
|
||||||
|
*
|
||||||
|
* @param other the score to evaluate against
|
||||||
|
* @return whether this network is probably faster than the other
|
||||||
|
* @hide
|
||||||
|
*/
|
||||||
|
public boolean probablyFasterThan(@NonNull final NetworkScore other) {
|
||||||
|
if (mLegacyScore > other.mLegacyScore) return true;
|
||||||
|
final int atEndToEnd = mEndToEndMetrics.compareTo(other.mEndToEndMetrics);
|
||||||
|
if (atEndToEnd > 0) return true;
|
||||||
|
if (atEndToEnd < 0) return false;
|
||||||
|
final int atLinkLayer = mLinkLayerMetrics.compareTo(other.mLinkLayerMetrics);
|
||||||
|
if (atLinkLayer > 0) return true;
|
||||||
|
if (atLinkLayer < 0) return false;
|
||||||
|
final int atEndToEndPreferringKnown =
|
||||||
|
mEndToEndMetrics.compareToPreferringKnown(other.mEndToEndMetrics);
|
||||||
|
if (atEndToEndPreferringKnown > 0) return true;
|
||||||
|
if (atEndToEndPreferringKnown < 0) return false;
|
||||||
|
// If this returns 0, neither is "probably faster" so just return false.
|
||||||
|
return mLinkLayerMetrics.compareToPreferringKnown(other.mLinkLayerMetrics) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
/** Builder for NetworkScore. */
|
/** Builder for NetworkScore. */
|
||||||
public static class Builder {
|
public static class Builder {
|
||||||
private int mPolicy = 0;
|
private int mPolicy = 0;
|
||||||
|
|||||||
@@ -16,8 +16,13 @@
|
|||||||
|
|
||||||
package com.android.server.connectivity;
|
package com.android.server.connectivity;
|
||||||
|
|
||||||
|
import static android.net.NetworkScore.POLICY_IGNORE_ON_WIFI;
|
||||||
|
|
||||||
|
import static com.android.internal.util.FunctionalUtils.findFirst;
|
||||||
|
|
||||||
import android.annotation.NonNull;
|
import android.annotation.NonNull;
|
||||||
import android.annotation.Nullable;
|
import android.annotation.Nullable;
|
||||||
|
import android.net.NetworkCapabilities;
|
||||||
import android.net.NetworkRequest;
|
import android.net.NetworkRequest;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@@ -37,15 +42,33 @@ public class NetworkRanker {
|
|||||||
@NonNull final Collection<NetworkAgentInfo> nais) {
|
@NonNull final Collection<NetworkAgentInfo> nais) {
|
||||||
final ArrayList<NetworkAgentInfo> candidates = new ArrayList<>(nais);
|
final ArrayList<NetworkAgentInfo> candidates = new ArrayList<>(nais);
|
||||||
candidates.removeIf(nai -> !nai.satisfies(request));
|
candidates.removeIf(nai -> !nai.satisfies(request));
|
||||||
|
// Enforce policy.
|
||||||
|
filterBadWifiAvoidancePolicy(candidates);
|
||||||
|
|
||||||
NetworkAgentInfo bestNetwork = null;
|
NetworkAgentInfo bestNetwork = null;
|
||||||
int bestScore = Integer.MIN_VALUE;
|
int bestScore = Integer.MIN_VALUE;
|
||||||
for (final NetworkAgentInfo nai : candidates) {
|
for (final NetworkAgentInfo nai : candidates) {
|
||||||
if (nai.getCurrentScore() > bestScore) {
|
final int score = nai.getCurrentScore();
|
||||||
|
if (score > bestScore) {
|
||||||
bestNetwork = nai;
|
bestNetwork = nai;
|
||||||
bestScore = nai.getCurrentScore();
|
bestScore = score;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return bestNetwork;
|
return bestNetwork;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If some network with wifi transport is present, drop all networks with POLICY_IGNORE_ON_WIFI.
|
||||||
|
private void filterBadWifiAvoidancePolicy(
|
||||||
|
@NonNull final ArrayList<NetworkAgentInfo> candidates) {
|
||||||
|
final NetworkAgentInfo wifi = findFirst(candidates,
|
||||||
|
nai -> nai.networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI)
|
||||||
|
&& nai.everValidated
|
||||||
|
// Horrible hack : there is old UI that will let a user say they want to
|
||||||
|
// override the policy only for this network only at this time and it
|
||||||
|
// feeds into the following member. This old UI should probably be removed
|
||||||
|
// but for now keep backward compatibility.
|
||||||
|
&& !nai.avoidUnvalidated);
|
||||||
|
if (null == wifi) return; // No wifi : this policy doesn't apply
|
||||||
|
candidates.removeIf(nai -> nai.getNetworkScore().hasPolicy(POLICY_IGNORE_ON_WIFI));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
package com.android.server.connectivity
|
package com.android.server.connectivity
|
||||||
|
|
||||||
|
import android.net.NetworkCapabilities
|
||||||
import android.net.NetworkRequest
|
import android.net.NetworkRequest
|
||||||
import androidx.test.filters.SmallTest
|
import androidx.test.filters.SmallTest
|
||||||
import androidx.test.runner.AndroidJUnit4
|
import androidx.test.runner.AndroidJUnit4
|
||||||
@@ -35,6 +36,7 @@ class NetworkRankerTest {
|
|||||||
private fun makeNai(satisfy: Boolean, score: Int) = mock(NetworkAgentInfo::class.java).also {
|
private fun makeNai(satisfy: Boolean, score: Int) = mock(NetworkAgentInfo::class.java).also {
|
||||||
doReturn(satisfy).`when`(it).satisfies(any())
|
doReturn(satisfy).`when`(it).satisfies(any())
|
||||||
doReturn(score).`when`(it).currentScore
|
doReturn(score).`when`(it).currentScore
|
||||||
|
it.networkCapabilities = NetworkCapabilities()
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|||||||
Reference in New Issue
Block a user