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.Nullable; 20 import android.os.Parcel; 21 import android.os.Parcelable; 22 import android.os.UserHandle; 23 import android.util.ArraySet; 24 import android.util.Range; 25 26 import java.util.Collection; 27 import java.util.Set; 28 29 /** 30 * An inclusive range of UIDs. 31 * 32 * @hide 33 */ 34 public final class UidRange implements Parcelable { 35 public final int start; 36 public final int stop; 37 UidRange(int startUid, int stopUid)38 public UidRange(int startUid, int stopUid) { 39 if (startUid < 0) throw new IllegalArgumentException("Invalid start UID."); 40 if (stopUid < 0) throw new IllegalArgumentException("Invalid stop UID."); 41 if (startUid > stopUid) throw new IllegalArgumentException("Invalid UID range."); 42 start = startUid; 43 stop = stopUid; 44 } 45 46 /** Creates a UidRange for the specified user. */ createForUser(UserHandle user)47 public static UidRange createForUser(UserHandle user) { 48 final UserHandle nextUser = UserHandle.of(user.getIdentifier() + 1); 49 final int start = user.getUid(0 /* appId */); 50 final int end = nextUser.getUid(0 /* appId */) - 1; 51 return new UidRange(start, end); 52 } 53 54 /** Returns the smallest user Id which is contained in this UidRange */ getStartUser()55 public int getStartUser() { 56 return UserHandle.getUserHandleForUid(start).getIdentifier(); 57 } 58 59 /** Returns the largest user Id which is contained in this UidRange */ getEndUser()60 public int getEndUser() { 61 return UserHandle.getUserHandleForUid(stop).getIdentifier(); 62 } 63 64 /** Returns whether the UidRange contains the specified UID. */ contains(int uid)65 public boolean contains(int uid) { 66 return start <= uid && uid <= stop; 67 } 68 69 /** 70 * Returns the count of UIDs in this range. 71 */ count()72 public int count() { 73 return 1 + stop - start; 74 } 75 76 /** 77 * @return {@code true} if this range contains every UID contained by the {@code other} range. 78 */ containsRange(UidRange other)79 public boolean containsRange(UidRange other) { 80 return start <= other.start && other.stop <= stop; 81 } 82 83 @Override hashCode()84 public int hashCode() { 85 int result = 17; 86 result = 31 * result + start; 87 result = 31 * result + stop; 88 return result; 89 } 90 91 @Override equals(@ullable Object o)92 public boolean equals(@Nullable Object o) { 93 if (this == o) { 94 return true; 95 } 96 if (o instanceof UidRange) { 97 UidRange other = (UidRange) o; 98 return start == other.start && stop == other.stop; 99 } 100 return false; 101 } 102 103 @Override toString()104 public String toString() { 105 return start + "-" + stop; 106 } 107 108 // Implement the Parcelable interface 109 // TODO: Consider making this class no longer parcelable, since all users are likely in the 110 // system server. 111 @Override describeContents()112 public int describeContents() { 113 return 0; 114 } 115 116 @Override writeToParcel(Parcel dest, int flags)117 public void writeToParcel(Parcel dest, int flags) { 118 dest.writeInt(start); 119 dest.writeInt(stop); 120 } 121 122 public static final @android.annotation.NonNull Creator<UidRange> CREATOR = 123 new Creator<UidRange>() { 124 @Override 125 public UidRange createFromParcel(Parcel in) { 126 int start = in.readInt(); 127 int stop = in.readInt(); 128 129 return new UidRange(start, stop); 130 } 131 @Override 132 public UidRange[] newArray(int size) { 133 return new UidRange[size]; 134 } 135 }; 136 137 /** 138 * Returns whether any of the UidRange in the collection contains the specified uid 139 * 140 * @param ranges The collection of UidRange to check 141 * @param uid the uid in question 142 * @return {@code true} if the uid is contained within the ranges, {@code false} otherwise 143 * 144 * @see UidRange#contains(int) 145 */ containsUid(Collection<UidRange> ranges, int uid)146 public static boolean containsUid(Collection<UidRange> ranges, int uid) { 147 if (ranges == null) return false; 148 for (UidRange range : ranges) { 149 if (range.contains(uid)) { 150 return true; 151 } 152 } 153 return false; 154 } 155 156 /** 157 * Convert a set of {@code Range<Integer>} to a set of {@link UidRange}. 158 */ 159 @Nullable fromIntRanges(@ullable Set<Range<Integer>> ranges)160 public static ArraySet<UidRange> fromIntRanges(@Nullable Set<Range<Integer>> ranges) { 161 if (null == ranges) return null; 162 163 final ArraySet<UidRange> uids = new ArraySet<>(); 164 for (Range<Integer> range : ranges) { 165 uids.add(new UidRange(range.getLower(), range.getUpper())); 166 } 167 return uids; 168 } 169 170 /** 171 * Convert a set of {@link UidRange} to a set of {@code Range<Integer>}. 172 */ 173 @Nullable toIntRanges(@ullable Set<UidRange> ranges)174 public static ArraySet<Range<Integer>> toIntRanges(@Nullable Set<UidRange> ranges) { 175 if (null == ranges) return null; 176 177 final ArraySet<Range<Integer>> uids = new ArraySet<>(); 178 for (UidRange range : ranges) { 179 uids.add(new Range<Integer>(range.start, range.stop)); 180 } 181 return uids; 182 } 183 184 /** 185 * Compare if the given UID range sets have the same UIDs. 186 * 187 * @hide 188 */ hasSameUids(@ullable Set<UidRange> uids1, @Nullable Set<UidRange> uids2)189 public static boolean hasSameUids(@Nullable Set<UidRange> uids1, 190 @Nullable Set<UidRange> uids2) { 191 if (null == uids1) return null == uids2; 192 if (null == uids2) return false; 193 // Make a copy so it can be mutated to check that all ranges in uids2 also are in uids. 194 final Set<UidRange> remainingUids = new ArraySet<>(uids2); 195 for (UidRange range : uids1) { 196 if (!remainingUids.contains(range)) { 197 return false; 198 } 199 remainingUids.remove(range); 200 } 201 return remainingUids.isEmpty(); 202 } 203 } 204