1 /*
2  * Copyright (C) 2016 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 com.android.server.wifi;
18 
19 import android.text.TextUtils;
20 
21 import com.android.internal.annotations.VisibleForTesting;
22 
23 /**
24  * Class for storing an IMSI (International Mobile Subscriber Identity) parameter.  The IMSI
25  * contains number (up to 15) of numerical digits.  When an IMSI ends with a '*', the specified
26  * IMSI is a prefix.
27  */
28 public class IMSIParameter {
29     /**
30      * Per 2.2 of 3GPP TS 23.003
31      * MCC (Mobile Country Code) is a 3 digit number and MNC (Mobile Network Code) is a 2
32      * or 3 digit number;
33      * The max length of IMSI is 15;
34      */
35     public static final int MCC_MNC_LENGTH_5 = 5;
36     public static final int MCC_MNC_LENGTH_6 = 6;
37     private static final int MAX_IMSI_LENGTH = 15;
38 
39     private final String mImsi;
40     private final boolean mPrefix;
41 
42     @VisibleForTesting
IMSIParameter(String imsi, boolean prefix)43     public IMSIParameter(String imsi, boolean prefix) {
44         mImsi = imsi;
45         mPrefix = prefix;
46     }
47 
48     /**
49      * Build an IMSIParameter object from the given string.  A null will be returned for a
50      * malformed string.
51      *
52      * @param imsi The IMSI string
53      * @return {@link IMSIParameter}
54      */
build(String imsi)55     public static IMSIParameter build(String imsi) {
56         if (TextUtils.isEmpty(imsi)) {
57             return null;
58         }
59         if (imsi.length() > MAX_IMSI_LENGTH) {
60             return null;
61         }
62 
63         // Detect the first non-digit character.
64         int nonDigitIndex;
65         char stopChar = '\0';
66         for (nonDigitIndex = 0; nonDigitIndex < imsi.length(); nonDigitIndex++) {
67             stopChar = imsi.charAt(nonDigitIndex);
68             if (stopChar < '0' || stopChar > '9') {
69                 break;
70             }
71         }
72 
73         if (nonDigitIndex == imsi.length()) {
74             // Full IMSI.
75             return new IMSIParameter(imsi, false);
76         } else if (nonDigitIndex == imsi.length() - 1 && stopChar == '*'
77                 && (nonDigitIndex == MCC_MNC_LENGTH_5 || nonDigitIndex == MCC_MNC_LENGTH_6)) {
78             // IMSI prefix.
79             return new IMSIParameter(imsi.substring(0, nonDigitIndex), true);
80         }
81         return null;
82     }
83 
84     /**
85      * Perform matching against the given full IMSI.
86      *
87      * @param fullIMSI The full IMSI to match against
88      * @return true if matched
89      */
matchesImsi(String fullIMSI)90     public boolean matchesImsi(String fullIMSI) {
91         if (fullIMSI == null) {
92             return false;
93         }
94 
95         if (mPrefix) {
96             // Prefix matching.
97             return mImsi.regionMatches(false, 0, fullIMSI, 0, mImsi.length());
98         } else {
99             // Exact matching.
100             return TextUtils.equals(mImsi, fullIMSI);
101         }
102     }
103 
104     /**
105      * Perform matching against the given MCC-MNC (Mobile Country Code and Mobile Network
106      * Code) combination.
107      *
108      * @param mccMnc The MCC-MNC to match against
109      * @return true if matched
110      */
matchesMccMnc(String mccMnc)111     public boolean matchesMccMnc(String mccMnc) {
112         if (mccMnc == null) {
113             return false;
114         }
115         if (mccMnc.length() != MCC_MNC_LENGTH_5 && mccMnc.length() != MCC_MNC_LENGTH_6) {
116             return false;
117         }
118         if (mPrefix && mccMnc.length() != mImsi.length()) {
119             return false;
120         }
121 
122         return mImsi.startsWith(mccMnc);
123     }
124 
125     /**
126      * If the IMSI is full length.
127      *
128      * @return true If the length of IMSI is full, false otherwise.
129      */
isFullImsi()130     public boolean isFullImsi() {
131         return !mPrefix;
132     }
133 
134     @Override
equals(Object thatObject)135     public boolean equals(Object thatObject) {
136         if (this == thatObject) {
137             return true;
138         }
139         if (!(thatObject instanceof IMSIParameter)) {
140             return false;
141         }
142 
143         IMSIParameter that = (IMSIParameter) thatObject;
144         return mPrefix == that.mPrefix && TextUtils.equals(mImsi, that.mImsi);
145     }
146 
147     @Override
hashCode()148     public int hashCode() {
149         int result = mImsi != null ? mImsi.hashCode() : 0;
150         result = 31 * result + (mPrefix ? 1 : 0);
151         return result;
152     }
153 
154     @Override
toString()155     public String toString() {
156         if (mPrefix) {
157             return mImsi + '*';
158         }
159         else {
160             return mImsi;
161         }
162     }
163 }
164