1/*
2 * Copyright (C) 2008 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
17package com.android.internal.telephony.cdma.sms;
18
19import android.util.SparseIntArray;
20
21import com.android.internal.telephony.SmsHeader;
22import com.android.internal.util.HexDump;
23
24public class UserData {
25
26    /**
27     * User data encoding types.
28     * (See 3GPP2 C.R1001-F, v1.0, table 9.1-1)
29     */
30    public static final int ENCODING_OCTET                      = 0x00;
31    public static final int ENCODING_IS91_EXTENDED_PROTOCOL     = 0x01;
32    public static final int ENCODING_7BIT_ASCII                 = 0x02;
33    public static final int ENCODING_IA5                        = 0x03;
34    public static final int ENCODING_UNICODE_16                 = 0x04;
35    public static final int ENCODING_SHIFT_JIS                  = 0x05;
36    public static final int ENCODING_KOREAN                     = 0x06;
37    public static final int ENCODING_LATIN_HEBREW               = 0x07;
38    public static final int ENCODING_LATIN                      = 0x08;
39    public static final int ENCODING_GSM_7BIT_ALPHABET          = 0x09;
40    public static final int ENCODING_GSM_DCS                    = 0x0A;
41
42    /**
43     * User data message type encoding types.
44     * (See 3GPP2 C.S0015-B, 4.5.2 and 3GPP 23.038, Section 4)
45     */
46    public static final int ENCODING_GSM_DCS_7BIT               = 0x00;
47    public static final int ENCODING_GSM_DCS_8BIT               = 0x01;
48    public static final int ENCODING_GSM_DCS_16BIT              = 0x02;
49
50    /**
51     * IS-91 message types.
52     * (See TIA/EIS/IS-91-A-ENGL 1999, table 3.7.1.1-3)
53     */
54    public static final int IS91_MSG_TYPE_VOICEMAIL_STATUS   = 0x82;
55    public static final int IS91_MSG_TYPE_SHORT_MESSAGE_FULL = 0x83;
56    public static final int IS91_MSG_TYPE_CLI                = 0x84;
57    public static final int IS91_MSG_TYPE_SHORT_MESSAGE      = 0x85;
58
59    /**
60     * US ASCII character mapping table.
61     *
62     * This table contains only the printable ASCII characters, with a
63     * 0x20 offset, meaning that the ASCII SPACE character is at index
64     * 0, with the resulting code of 0x20.
65     *
66     * Note this mapping is also equivalent to that used by both the
67     * IA5 and the IS-91 encodings.  For the former this is defined
68     * using CCITT Rec. T.50 Tables 1 and 3.  For the latter IS 637 B,
69     * Table 4.3.1.4.1-1 -- and note the encoding uses only 6 bits,
70     * and hence only maps entries up to the '_' character.
71     *
72     */
73    public static final char[] ASCII_MAP = {
74        ' ', '!', '"', '#', '$', '%', '&', '\'', '(', ')', '*', '+', ',', '-', '.', '/',
75        '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', ':', ';', '<', '=', '>', '?',
76        '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
77        'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '[', '\\', ']', '^', '_',
78        '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',
79        'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '{', '|', '}', '~'};
80
81    /**
82     * Character to use when forced to encode otherwise unencodable
83     * characters, meaning those not in the respective ASCII or GSM
84     * 7-bit encoding tables.  Current choice is SPACE, which is 0x20
85     * in both the GSM-7bit and ASCII-7bit encodings.
86     */
87    static final byte UNENCODABLE_7_BIT_CHAR = 0x20;
88
89    /**
90     * Only elements between these indices in the ASCII table are printable.
91     */
92    public static final int PRINTABLE_ASCII_MIN_INDEX = 0x20;
93    public static final int ASCII_NL_INDEX = 0x0A;
94    public static final int ASCII_CR_INDEX = 0x0D;
95    public static final SparseIntArray charToAscii = new SparseIntArray();
96    static {
97        for (int i = 0; i < ASCII_MAP.length; i++) {
98            charToAscii.put(ASCII_MAP[i], PRINTABLE_ASCII_MIN_INDEX + i);
99        }
100        charToAscii.put('\n', ASCII_NL_INDEX);
101        charToAscii.put('\r', ASCII_CR_INDEX);
102    }
103
104    /*
105     * TODO(cleanup): Move this very generic functionality somewhere
106     * more general.
107     */
108    /**
109     * Given a string generate a corresponding ASCII-encoded byte
110     * array, but limited to printable characters.  If the input
111     * contains unprintable characters, return null.
112     */
113    public static byte[] stringToAscii(String str) {
114        int len = str.length();
115        byte[] result = new byte[len];
116        for (int i = 0; i < len; i++) {
117            int charCode = charToAscii.get(str.charAt(i), -1);
118            if (charCode == -1) return null;
119            result[i] = (byte)charCode;
120        }
121        return result;
122    }
123
124    /**
125     * Mapping for ASCII values less than 32 are flow control signals
126     * and not used here.
127     */
128    public static final int ASCII_MAP_BASE_INDEX = 0x20;
129    public static final int ASCII_MAP_MAX_INDEX = ASCII_MAP_BASE_INDEX + ASCII_MAP.length - 1;
130
131    /**
132     * Contains the data header of the user data
133     */
134    public SmsHeader userDataHeader;
135
136    /**
137     * Contains the data encoding type for the SMS message
138     */
139    public int msgEncoding;
140    public boolean msgEncodingSet = false;
141
142    public int msgType;
143
144    /**
145     * Number of invalid bits in the last byte of data.
146     */
147    public int paddingBits;
148
149    public int numFields;
150
151    /**
152     * Contains the user data of a SMS message
153     * (See 3GPP2 C.S0015-B, v2, 4.5.2)
154     */
155    public byte[] payload;
156    public String payloadStr;
157
158    @Override
159    public String toString() {
160        StringBuilder builder = new StringBuilder();
161        builder.append("UserData ");
162        builder.append("{ msgEncoding=" + (msgEncodingSet ? msgEncoding : "unset"));
163        builder.append(", msgType=" + msgType);
164        builder.append(", paddingBits=" + paddingBits);
165        builder.append(", numFields=" + numFields);
166        builder.append(", userDataHeader=" + userDataHeader);
167        builder.append(", payload='" + HexDump.toHexString(payload) + "'");
168        builder.append(", payloadStr='" + payloadStr + "'");
169        builder.append(" }");
170        return builder.toString();
171    }
172
173}
174