/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.net; import android.annotation.Nullable; import android.os.Parcel; import android.os.Parcelable; import android.os.UserHandle; import android.util.ArraySet; import android.util.Range; import java.util.Collection; import java.util.Set; /** * An inclusive range of UIDs. * * @hide */ public final class UidRange implements Parcelable { public final int start; public final int stop; public UidRange(int startUid, int stopUid) { if (startUid < 0) throw new IllegalArgumentException("Invalid start UID."); if (stopUid < 0) throw new IllegalArgumentException("Invalid stop UID."); if (startUid > stopUid) throw new IllegalArgumentException("Invalid UID range."); start = startUid; stop = stopUid; } /** Creates a UidRange for the specified user. */ public static UidRange createForUser(UserHandle user) { final UserHandle nextUser = UserHandle.of(user.getIdentifier() + 1); final int start = user.getUid(0 /* appId */); final int end = nextUser.getUid(0 /* appId */) - 1; return new UidRange(start, end); } /** Returns the smallest user Id which is contained in this UidRange */ public int getStartUser() { return UserHandle.getUserHandleForUid(start).getIdentifier(); } /** Returns the largest user Id which is contained in this UidRange */ public int getEndUser() { return UserHandle.getUserHandleForUid(stop).getIdentifier(); } /** Returns whether the UidRange contains the specified UID. */ public boolean contains(int uid) { return start <= uid && uid <= stop; } /** * Returns the count of UIDs in this range. */ public int count() { return 1 + stop - start; } /** * @return {@code true} if this range contains every UID contained by the {@code other} range. */ public boolean containsRange(UidRange other) { return start <= other.start && other.stop <= stop; } @Override public int hashCode() { int result = 17; result = 31 * result + start; result = 31 * result + stop; return result; } @Override public boolean equals(@Nullable Object o) { if (this == o) { return true; } if (o instanceof UidRange) { UidRange other = (UidRange) o; return start == other.start && stop == other.stop; } return false; } @Override public String toString() { return start + "-" + stop; } // Implement the Parcelable interface // TODO: Consider making this class no longer parcelable, since all users are likely in the // system server. @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeInt(start); dest.writeInt(stop); } public static final @android.annotation.NonNull Creator CREATOR = new Creator() { @Override public UidRange createFromParcel(Parcel in) { int start = in.readInt(); int stop = in.readInt(); return new UidRange(start, stop); } @Override public UidRange[] newArray(int size) { return new UidRange[size]; } }; /** * Returns whether any of the UidRange in the collection contains the specified uid * * @param ranges The collection of UidRange to check * @param uid the uid in question * @return {@code true} if the uid is contained within the ranges, {@code false} otherwise * * @see UidRange#contains(int) */ public static boolean containsUid(Collection ranges, int uid) { if (ranges == null) return false; for (UidRange range : ranges) { if (range.contains(uid)) { return true; } } return false; } /** * Convert a set of {@code Range} to a set of {@link UidRange}. */ @Nullable public static ArraySet fromIntRanges(@Nullable Set> ranges) { if (null == ranges) return null; final ArraySet uids = new ArraySet<>(); for (Range range : ranges) { uids.add(new UidRange(range.getLower(), range.getUpper())); } return uids; } /** * Convert a set of {@link UidRange} to a set of {@code Range}. */ @Nullable public static ArraySet> toIntRanges(@Nullable Set ranges) { if (null == ranges) return null; final ArraySet> uids = new ArraySet<>(); for (UidRange range : ranges) { uids.add(new Range(range.start, range.stop)); } return uids; } /** * Compare if the given UID range sets have the same UIDs. * * @hide */ public static boolean hasSameUids(@Nullable Set uids1, @Nullable Set uids2) { if (null == uids1) return null == uids2; if (null == uids2) return false; // Make a copy so it can be mutated to check that all ranges in uids2 also are in uids. final Set remainingUids = new ArraySet<>(uids2); for (UidRange range : uids1) { if (!remainingUids.contains(range)) { return false; } remainingUids.remove(range); } return remainingUids.isEmpty(); } }