1 /*
2  * Copyright (C) 2012 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.telephony;
18 
19 import android.os.Parcel;
20 import android.os.Parcelable;
21 import android.text.format.Time;
22 
23 import com.android.internal.telephony.uicc.IccUtils;
24 
25 import java.util.Arrays;
26 
27 /**
28  * Contains information elements for a GSM or UMTS ETWS warning notification.
29  * Supported values for each element are defined in 3GPP TS 23.041.
30  *
31  * {@hide}
32  */
33 public class SmsCbEtwsInfo implements Parcelable {
34 
35     /** ETWS warning type for earthquake. */
36     public static final int ETWS_WARNING_TYPE_EARTHQUAKE = 0x00;
37 
38     /** ETWS warning type for tsunami. */
39     public static final int ETWS_WARNING_TYPE_TSUNAMI = 0x01;
40 
41     /** ETWS warning type for earthquake and tsunami. */
42     public static final int ETWS_WARNING_TYPE_EARTHQUAKE_AND_TSUNAMI = 0x02;
43 
44     /** ETWS warning type for test messages. */
45     public static final int ETWS_WARNING_TYPE_TEST_MESSAGE = 0x03;
46 
47     /** ETWS warning type for other emergency types. */
48     public static final int ETWS_WARNING_TYPE_OTHER_EMERGENCY = 0x04;
49 
50     /** Unknown ETWS warning type. */
51     public static final int ETWS_WARNING_TYPE_UNKNOWN = -1;
52 
53     /** One of the ETWS warning type constants defined in this class. */
54     private final int mWarningType;
55 
56     /** Whether or not to activate the emergency user alert tone and vibration. */
57     private final boolean mEmergencyUserAlert;
58 
59     /** Whether or not to activate a popup alert. */
60     private final boolean mActivatePopup;
61 
62     /**
63      * 50-byte security information (ETWS primary notification for GSM only). As of Release 10,
64      * 3GPP TS 23.041 states that the UE shall ignore the ETWS primary notification timestamp
65      * and digital signature if received. Therefore it is treated as a raw byte array and
66      * parceled with the broadcast intent if present, but the timestamp is only computed if an
67      * application asks for the individual components.
68      */
69     private final byte[] mWarningSecurityInformation;
70 
71     /** Create a new SmsCbEtwsInfo object with the specified values. */
SmsCbEtwsInfo(int warningType, boolean emergencyUserAlert, boolean activatePopup, byte[] warningSecurityInformation)72     public SmsCbEtwsInfo(int warningType, boolean emergencyUserAlert, boolean activatePopup,
73             byte[] warningSecurityInformation) {
74         mWarningType = warningType;
75         mEmergencyUserAlert = emergencyUserAlert;
76         mActivatePopup = activatePopup;
77         mWarningSecurityInformation = warningSecurityInformation;
78     }
79 
80     /** Create a new SmsCbEtwsInfo object from a Parcel. */
SmsCbEtwsInfo(Parcel in)81     SmsCbEtwsInfo(Parcel in) {
82         mWarningType = in.readInt();
83         mEmergencyUserAlert = (in.readInt() != 0);
84         mActivatePopup = (in.readInt() != 0);
85         mWarningSecurityInformation = in.createByteArray();
86     }
87 
88     /**
89      * Flatten this object into a Parcel.
90      *
91      * @param dest  The Parcel in which the object should be written.
92      * @param flags Additional flags about how the object should be written (ignored).
93      */
94     @Override
writeToParcel(Parcel dest, int flags)95     public void writeToParcel(Parcel dest, int flags) {
96         dest.writeInt(mWarningType);
97         dest.writeInt(mEmergencyUserAlert ? 1 : 0);
98         dest.writeInt(mActivatePopup ? 1 : 0);
99         dest.writeByteArray(mWarningSecurityInformation);
100     }
101 
102     /**
103      * Returns the ETWS warning type.
104      * @return a warning type such as {@link #ETWS_WARNING_TYPE_EARTHQUAKE}
105      */
getWarningType()106     public int getWarningType() {
107         return mWarningType;
108     }
109 
110     /**
111      * Returns the ETWS emergency user alert flag.
112      * @return true to notify terminal to activate emergency user alert; false otherwise
113      */
isEmergencyUserAlert()114     public boolean isEmergencyUserAlert() {
115         return mEmergencyUserAlert;
116     }
117 
118     /**
119      * Returns the ETWS activate popup flag.
120      * @return true to notify terminal to activate display popup; false otherwise
121      */
isPopupAlert()122     public boolean isPopupAlert() {
123         return mActivatePopup;
124     }
125 
126     /**
127      * Returns the Warning-Security-Information timestamp (GSM primary notifications only).
128      * As of Release 10, 3GPP TS 23.041 states that the UE shall ignore this value if received.
129      * @return a UTC timestamp in System.currentTimeMillis() format, or 0 if not present
130      */
getPrimaryNotificationTimestamp()131     public long getPrimaryNotificationTimestamp() {
132         if (mWarningSecurityInformation == null || mWarningSecurityInformation.length < 7) {
133             return 0;
134         }
135 
136         int year = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[0]);
137         int month = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[1]);
138         int day = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[2]);
139         int hour = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[3]);
140         int minute = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[4]);
141         int second = IccUtils.gsmBcdByteToInt(mWarningSecurityInformation[5]);
142 
143         // For the timezone, the most significant bit of the
144         // least significant nibble is the sign byte
145         // (meaning the max range of this field is 79 quarter-hours,
146         // which is more than enough)
147 
148         byte tzByte = mWarningSecurityInformation[6];
149 
150         // Mask out sign bit.
151         int timezoneOffset = IccUtils.gsmBcdByteToInt((byte) (tzByte & (~0x08)));
152 
153         timezoneOffset = ((tzByte & 0x08) == 0) ? timezoneOffset : -timezoneOffset;
154 
155         Time time = new Time(Time.TIMEZONE_UTC);
156 
157         // We only need to support years above 2000.
158         time.year = year + 2000;
159         time.month = month - 1;
160         time.monthDay = day;
161         time.hour = hour;
162         time.minute = minute;
163         time.second = second;
164 
165         // Timezone offset is in quarter hours.
166         return time.toMillis(true) - timezoneOffset * 15 * 60 * 1000;
167     }
168 
169     /**
170      * Returns the digital signature (GSM primary notifications only). As of Release 10,
171      * 3GPP TS 23.041 states that the UE shall ignore this value if received.
172      * @return a byte array containing a copy of the primary notification digital signature
173      */
getPrimaryNotificationSignature()174     public byte[] getPrimaryNotificationSignature() {
175         if (mWarningSecurityInformation == null || mWarningSecurityInformation.length < 50) {
176             return null;
177         }
178         return Arrays.copyOfRange(mWarningSecurityInformation, 7, 50);
179     }
180 
181     @Override
toString()182     public String toString() {
183         return "SmsCbEtwsInfo{warningType=" + mWarningType + ", emergencyUserAlert="
184                 + mEmergencyUserAlert + ", activatePopup=" + mActivatePopup + '}';
185     }
186 
187     /**
188      * Describe the kinds of special objects contained in the marshalled representation.
189      * @return a bitmask indicating this Parcelable contains no special objects
190      */
191     @Override
describeContents()192     public int describeContents() {
193         return 0;
194     }
195 
196     /** Creator for unparcelling objects. */
197     public static final Creator<SmsCbEtwsInfo> CREATOR = new Creator<SmsCbEtwsInfo>() {
198         @Override
199         public SmsCbEtwsInfo createFromParcel(Parcel in) {
200             return new SmsCbEtwsInfo(in);
201         }
202 
203         @Override
204         public SmsCbEtwsInfo[] newArray(int size) {
205             return new SmsCbEtwsInfo[size];
206         }
207     };
208 }
209