• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2021 Huawei Device Co., Ltd.
3  * Copyright (C) 2014 Samsung Electronics Co., Ltd. All rights reserved
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 #include "sms_common_utils.h"
18 
19 #include <ctime>
20 #include "securec.h"
21 #include "telephony_log_wrapper.h"
22 
23 namespace OHOS {
24 namespace Telephony {
Pack7bitChar(const unsigned char * userData,int dataLen,int fillBits,unsigned char * packData)25 int SmsCommonUtils::Pack7bitChar(const unsigned char *userData, int dataLen, int fillBits, unsigned char *packData)
26 {
27     int srcIdx = 0;
28     int dstIdx = 0;
29     auto shift = static_cast<unsigned int>(fillBits);
30 
31     if (userData == nullptr || packData == nullptr) {
32         return dstIdx;
33     }
34 
35     if (shift > 0) {
36         dstIdx = 1;
37     }
38     while (srcIdx < dataLen) {
39         if (shift == 0) {
40             packData[dstIdx] = userData[srcIdx];
41             shift = SmsCommonUtils::SMS_ENCODE_GSM_BIT;
42             srcIdx++;
43             dstIdx++;
44             if (srcIdx >= dataLen) {
45                 break;
46             }
47         }
48         if (shift > 1) {
49             packData[dstIdx - 1] |= userData[srcIdx] << shift;
50             packData[dstIdx] = userData[srcIdx] >> (SmsCommonUtils::SMS_BYTE_BIT - shift);
51             shift--;
52             srcIdx++;
53             dstIdx++;
54         } else if (shift == 1) {
55             packData[dstIdx - 1] |= userData[srcIdx] << shift;
56             srcIdx++;
57             shift--;
58         }
59     }
60     return dstIdx;
61 }
62 
Unpack7bitChar(const unsigned char * tpdu,unsigned char dataLen,int fillBits,unsigned char * unpackData,unsigned int unpackDataLen)63 int SmsCommonUtils::Unpack7bitChar(const unsigned char *tpdu, unsigned char dataLen, int fillBits,
64     unsigned char *unpackData, unsigned int unpackDataLen)
65 {
66     int srcIdx = 0;
67     int dstIdx = 0;
68     auto shift = static_cast<unsigned int>(fillBits);
69     if (unpackData == nullptr || tpdu == nullptr || dataLen > unpackDataLen) {
70         return dstIdx;
71     }
72     if (shift > 0) {
73         srcIdx = 1;
74     }
75     for (; dstIdx < dataLen; dstIdx++) {
76         if (shift == 0) {
77             unpackData[dstIdx] = tpdu[srcIdx] & 0x7F;
78             shift = SmsCommonUtils::SMS_ENCODE_GSM_BIT;
79             srcIdx++;
80             dstIdx++;
81             if (dstIdx >= dataLen) {
82                 break;
83             }
84         }
85         if (shift > 0) {
86             unpackData[dstIdx] =
87                 (tpdu[srcIdx - 1] >> shift) + (tpdu[srcIdx] << (SmsCommonUtils::SMS_BYTE_BIT - shift));
88             unpackData[dstIdx] &= 0x7F;
89             shift--;
90             if (shift > 0) {
91                 srcIdx++;
92             }
93         }
94     }
95     return dstIdx;
96 }
97 
DigitToBcd(const char * digit,int digitLen,unsigned char * bcd)98 int SmsCommonUtils::DigitToBcd(const char *digit, int digitLen, unsigned char *bcd)
99 {
100     int offset = 0;
101     unsigned char temp;
102     if (digit == nullptr || bcd == nullptr) {
103         return offset;
104     }
105     for (int i = 0; i < digitLen; i++) {
106         switch (digit[i]) {
107             case '*':
108                 temp = 0x0A;
109                 break;
110             case '#':
111                 temp = 0x0B;
112                 break;
113             case 'P':
114             case 'p':
115                 temp = 0x0C;
116                 break;
117             default:
118                 temp = digit[i] - '0';
119                 break;
120         }
121         if ((i % SmsCommonUtils::SMS_HEX_BYTE_STEP) == 0) {
122             bcd[offset] = temp & 0x0F;
123         } else {
124             bcd[offset++] |= ((temp & 0x0F) << 0x04);
125         }
126     }
127 
128     if ((digitLen % SmsCommonUtils::SMS_HEX_BYTE_STEP) == 1) {
129         bcd[offset++] |= 0xF0;
130     }
131     return offset;
132 }
133 
BcdToDigit(const unsigned char * bcd,int bcdLen,char * digit)134 int SmsCommonUtils::BcdToDigit(const unsigned char *bcd, int bcdLen, char *digit)
135 {
136     int offset = 0;
137     unsigned char temp;
138     if (bcd == nullptr || digit == nullptr) {
139         return offset;
140     }
141     for (int i = 0; i < bcdLen; i++) {
142         temp = bcd[i] & 0x0F;
143         digit[offset++] = BcdToChar(temp);
144         temp = (bcd[i] & 0xF0) >> 0x04;
145         if (temp == 0x0F) {
146             digit[offset] = '\0';
147             return offset;
148         }
149         digit[offset++] = BcdToChar(temp);
150     }
151     digit[offset] = '\0';
152     return offset;
153 }
154 
BcdToDigitCdma(const unsigned char * bcd,int bcdLen,char * digit)155 int SmsCommonUtils::BcdToDigitCdma(const unsigned char *bcd, int bcdLen, char *digit)
156 {
157     int offset = 0;
158     unsigned char temp;
159     if (bcd == nullptr || digit == nullptr) {
160         return offset;
161     }
162 
163     for (int i = 0; i < bcdLen; i++) {
164         temp = (bcd[i] & 0xF0) >> 0x04;
165         digit[offset++] = BcdToChar(temp);
166         temp = bcd[i] & 0x0F;
167         if (temp == 0x0F) {
168             digit[offset] = '\0';
169             return offset;
170         }
171         digit[offset++] = BcdToChar(temp);
172     }
173     digit[offset] = '\0';
174     return offset;
175 }
176 
BcdToChar(const unsigned char c)177 char SmsCommonUtils::BcdToChar(const unsigned char c)
178 {
179     char temp = 0;
180     switch (c) {
181         case 0x0A:
182             temp = '*';
183             break;
184         case 0x0B:
185             temp = '#';
186             break;
187         case 0x0C:
188             temp = 'P';
189             break;
190         default:
191             temp = c + '0';
192             break;
193     }
194     return temp;
195 }
196 
ConvertDigitToDTMF(const char * digit,int digitLen,int startBit,unsigned char * dtmf)197 int SmsCommonUtils::ConvertDigitToDTMF(const char *digit, int digitLen, int startBit, unsigned char *dtmf)
198 {
199     int shift = startBit;
200     int offset = 0;
201     int srcIdx = 0;
202     uint8_t smsMaxShift = 4;
203     unsigned char temp;
204     if (digit == nullptr || dtmf == nullptr) {
205         TELEPHONY_LOGE("digit or dtmf is nullptr.");
206         return offset;
207     }
208     /* shift 1 and shift 2 are supported in this spec. */
209     if (shift >= smsMaxShift) {
210         TELEPHONY_LOGE("Invalid Param value shift : %{public}d", shift);
211         return offset;
212     }
213 
214     for (int i = 0; i < digitLen; i++) {
215         temp = DigitToDtmfChar(digit[srcIdx]);
216         temp &= 0x0F;
217         if (shift == 0) {
218             if (i % SmsCommonUtils::SMS_HEX_BYTE_STEP == 0x01) {
219                 dtmf[offset] |= temp;
220                 offset++;
221             } else {
222                 dtmf[offset] |= temp << 0x04;
223             }
224         } else if (shift >= 0x01 && shift < smsMaxShift) {
225             if (i % SmsCommonUtils::SMS_HEX_BYTE_STEP == 0x01) {
226                 dtmf[offset] |= (temp >> shift);
227                 dtmf[offset + 0x01] = temp << (SmsCommonUtils::SMS_BYTE_BIT - shift);
228                 offset++;
229             } else {
230                 dtmf[offset] |= (temp << (SmsCommonUtils::SMS_BYTE_BIT - shift - smsMaxShift));
231             }
232         }
233         srcIdx++;
234     }
235     return offset;
236 }
237 
DigitToDtmfChar(const unsigned char c)238 unsigned char SmsCommonUtils::DigitToDtmfChar(const unsigned char c)
239 {
240     switch (c) {
241         case '*':
242             return 0x0B;
243         case '#':
244             return 0x0C;
245         case '0':
246             return 0x0A;
247         default:
248             return (c - '0');
249     }
250 }
251 
ConvertTime(const struct SmsTimeAbs & timeAbs)252 long SmsCommonUtils::ConvertTime(const struct SmsTimeAbs &timeAbs)
253 {
254     time_t rawtime;
255     struct tm tmInfo;
256     if (memset_s(&tmInfo, sizeof(struct tm), 0x00, sizeof(tm)) != EOK) {
257         return time(nullptr);
258     }
259     tmInfo.tm_year = (timeAbs.year + BASE_GSM_YEAR);
260     tmInfo.tm_mon = (timeAbs.month - 0x01);
261     tmInfo.tm_mday = timeAbs.day;
262     tmInfo.tm_hour = timeAbs.hour;
263     tmInfo.tm_min = timeAbs.minute;
264     tmInfo.tm_sec = timeAbs.second;
265     tmInfo.tm_isdst = 0;
266     rawtime = mktime(&tmInfo);
267     DisplayTime(rawtime);
268     rawtime -= (timeAbs.timeZone * (SEC_PER_HOUR / 0x04));
269     DisplayTime(rawtime);
270 /* timezone value is tiemzone + daylight. So should not add daylight */
271 #ifdef __MSG_DAYLIGHT_APPLIED__
272     rawtime -= (timezone - daylight * SEC_PER_HOUR);
273 #else
274     rawtime -= timezone;
275 #endif
276     DisplayTime(rawtime);
277     return rawtime;
278 }
279 
DisplayTime(const time_t & rawtime)280 void SmsCommonUtils::DisplayTime(const time_t &rawtime)
281 {
282     struct tm tmInfo;
283     const uint8_t maxAbsTimeLen = 32;
284     char displayTime[maxAbsTimeLen];
285     if (memset_s(&tmInfo, sizeof(struct tm), 0x00, sizeof(tm)) != EOK) {
286         TELEPHONY_LOGE("DisplayTime memset fail.");
287         return;
288     }
289 
290     if (memset_s(displayTime, sizeof(displayTime), 0x00, sizeof(displayTime)) != EOK) {
291         TELEPHONY_LOGE("DisplayTime memset fail.");
292         return;
293     }
294 
295     localtime_r(&rawtime, &tmInfo);
296     if (strftime(displayTime, maxAbsTimeLen, "%Y-%02m-%02d %T %z", &tmInfo) <= 0) {
297         TELEPHONY_LOGE("strftime error.");
298         return;
299     }
300     TELEPHONY_LOGI("displayTime [%{public}s]", displayTime);
301 }
302 } // namespace Telephony
303 } // namespace OHOS