/* Copyright (C) 2007-2008 The Android Open Source Project ** ** This software is licensed under the terms of the GNU General Public ** License version 2, as published by the Free Software Foundation, and ** may be copied, distributed, and modified under those terms. ** ** This program is distributed in the hope that it will be useful, ** but WITHOUT ANY WARRANTY; without even the implied warranty of ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ** GNU General Public License for more details. */ #include "sms.h" #include "gsm.h" #include <memory.h> #include <stdlib.h> #include <string.h> #include <assert.h> #define DEBUG 1 #if 1 # include "android/utils/debug.h" # define D_ACTIVE VERBOSE_CHECK(modem) #else # define D_ACTIVE DEBUG #endif #if DEBUG # define D(...) VERBOSE_PRINT(modem,__VA_ARGS__) #else # define D(...) ((void)0) #endif /* maximum number of data bytes in a SMS data message */ #define MAX_USER_DATA_BYTES 140 /* maximum number of 7-bit septets in a SMS text message */ #define MAX_USER_DATA_SEPTETS 160 /* size of the user data header in bytes */ #define USER_DATA_HEADER_SIZE 6 /** MESSAGE TEXT **/ int sms_utf8_from_message_str( const char* str, int strlen, unsigned char* utf8, int utf8len ) { cbytes_t p = (cbytes_t)str; cbytes_t end = p + strlen; int count = 0; int escaped = 0; while (p < end) { int c = p[0]; /* read the value from the string */ p += 1; if (c >= 128) { if ((c & 0xe0) == 0xc0) c &= 0x1f; else if ((c & 0xf0) == 0xe0) c &= 0x0f; else c &= 0x07; p++; while (p < end && (p[0] & 0xc0) == 0x80) { c = (c << 6) | (p[0] & 0x3f); p++; } } if (escaped) { switch (c) { case '\\': break; case 'n': /* \n is line feed */ c = 10; break; case 'x': /* \xNN, where NN is a 2-digit hexadecimal value */ if (p+2 > end) return -1; c = gsm_hex2_to_byte( (const char*)p ); if (c < 0) return -1; p += 2; break; case 'u': /* \uNNNN where NNNN is a 4-digiti hexadecimal value */ if (p + 4 > end) return -1; c = gsm_hex4_to_short( (const char*)p ); if (c < 0) return -1; p += 4; break; default: /* invalid escape, return -1 */ return -1; } escaped = 0; } else if (c == '\\') { escaped = 1; continue; } /* now, try to write it to the destination */ if (c < 128) { if (count < utf8len) utf8[count] = (byte_t) c; count += 1; } else if (c < 0x800) { if (count < utf8len) utf8[count] = (byte_t)(0xc0 | ((c >> 6) & 0x1f)); if (count+1 < utf8len) utf8[count+1] = (byte_t)(0x80 | (c & 0x3f)); count += 2; } else { if (count < utf8len) utf8[count] = (byte_t)(0xc0 | ((c >> 12) & 0xf)); if (count+1 < utf8len) utf8[count+1] = (byte_t)(0x80 | ((c >> 6) & 0x3f)); if (count+2 < utf8len) utf8[count+2] = (byte_t)(0x80 | (c & 0x3f)); count += 3; } } if (escaped) /* bad final escape */ return -1; return count; } /* to convert utf-8 to a message string, we only need to deal with control characters * and that's it */ int sms_utf8_to_message_str( const unsigned char* utf8, int utf8len, char* str, int strlen ) { cbytes_t p = utf8; cbytes_t end = p + utf8len; int count = 0; while (p < end) { int c = p[0]; int escape = 0; /* read the value from the string */ p += 1; if (c >= 128) { if ((c & 0xe0) == 0xc0) c &= 0x1f; else if ((c & 0xf0) == 0xe0) c &= 0x0f; else c &= 0x07; p++; while (p < end && (p[0] & 0xc0) == 0x80) { c = (c << 6) | (p[0] & 0x3f); p++; } } if (c < ' ') { escape = 1; if (c == '\n') { c = 'n'; escape = 2; } } else if (c == '\\') escape = 2; switch (escape) { case 0: if (c < 128) { if (count < strlen) str[count] = (char) c; count += 1; } else if (c < 0x800) { if (count < strlen) str[count] = (byte_t)(0xc0 | ((c >> 6) & 0x1f)); if (count+1 < strlen) str[count+1] = (byte_t)(0x80 | (c & 0x3f)); count += 2; } else { if (count < strlen) str[count] = (byte_t)(0xc0 | ((c >> 12) & 0xf)); if (count+1 < strlen) str[count+1] = (byte_t)(0x80 | ((c >> 6) & 0x3f)); if (count+2 < strlen) str[count+2] = (byte_t)(0x80 | (c & 0x3f)); count += 3; } break; case 1: if (count+3 < strlen) { str[count+0] = '\\'; str[count+1] = 'x'; gsm_hex_from_byte(str + count + 2, c); } count += 4; break; default: if (count+2 < strlen) { str[count+0] = '\\'; str[count+1] = (char) c; } count += 2; } } return count; } /** TIMESTAMPS **/ void sms_timestamp_now( SmsTimeStamp stamp ) { time_t now_time = time(NULL); struct tm gm = *(gmtime(&now_time)); struct tm local = *(localtime(&now_time)); int tzdiff = 0; stamp->data[0] = gsm_int_to_bcdi( local.tm_year % 100 ); stamp->data[1] = gsm_int_to_bcdi( local.tm_mon+1 ); stamp->data[2] = gsm_int_to_bcdi( local.tm_mday ); stamp->data[3] = gsm_int_to_bcdi( local.tm_hour ); stamp->data[4] = gsm_int_to_bcdi( local.tm_min ); stamp->data[5] = gsm_int_to_bcdi( local.tm_sec ); tzdiff = (local.tm_hour*4 + local.tm_min/15) - (gm.tm_hour*4 + gm.tm_min/15); if (local.tm_yday > gm.tm_yday) tzdiff += 24*4; else if (local.tm_yday < gm.tm_yday) tzdiff -= 24*4; stamp->data[6] = gsm_int_to_bcdi( tzdiff >= 0 ? tzdiff : -tzdiff ); if (tzdiff < 0) stamp->data[6] |= 0x08; } int sms_timestamp_to_tm( SmsTimeStamp stamp, struct tm* tm ) { int tzdiff; tm->tm_year = gsm_int_from_bcdi( stamp->data[0] ); if (tm->tm_year < 50) tm->tm_year += 100; tm->tm_mon = gsm_int_from_bcdi( stamp->data[1] ) -1; tm->tm_mday = gsm_int_from_bcdi( stamp->data[2] ); tm->tm_hour = gsm_int_from_bcdi( stamp->data[3] ); tm->tm_min = gsm_int_from_bcdi( stamp->data[4] ); tm->tm_sec = gsm_int_from_bcdi( stamp->data[5] ); tm->tm_isdst = -1; tzdiff = gsm_int_from_bcdi( stamp->data[6] & 0xf7 ); if (stamp->data[6] & 0x8) tzdiff = -tzdiff; return tzdiff; } static void gsm_rope_add_timestamp( GsmRope rope, const SmsTimeStampRec* ts ) { gsm_rope_add( rope, ts->data, 7 ); } /** SMS ADDRESSES **/ int sms_address_from_str( SmsAddress address, const char* src, int srclen ) { const char* end = src + srclen; int shift = 0, len = 0; bytes_t data = address->data; address->len = 0; address->toa = 0x81; if (src >= end) return -1; if ( src[0] == '+' ) { address->toa = 0x91; if (++src == end) goto Fail; } memset( address->data, 0, sizeof(address->data) ); shift = 0; while (src < end) { int c = *src++ - '0'; if ( (unsigned)c >= 10 || data >= address->data + sizeof(address->data) ) goto Fail; data[0] |= c << shift; len += 1; shift += 4; if (shift == 8) { shift = 0; data += 1; } } if (shift != 0) data[0] |= 0xf0; address->len = len; return 0; Fail: return -1; } int sms_address_to_str( SmsAddress address, char* str, int strlen ) { static const char dialdigits[16] = "0123456789*#,N%"; int n, count = 0; if (address->toa == 0x91) { if (count < strlen) str[count] = '+'; count++; } for (n = 0; n < address->len; n += 2) { int c = address->data[n/2]; if (count < strlen) str[count] = dialdigits[c & 0xf]; count += 1; if (n+1 > address->len) break; if (count < strlen) str[count] = dialdigits[(c >> 4) & 0xf]; if (str[count]) count += 1; } return count; } int sms_address_from_bytes( SmsAddress address, const unsigned char* buf, int buflen ) { int len = sizeof(address->data), num_digits; if (buflen < 2) return -1; address->len = num_digits = buf[0]; address->toa = buf[1]; len = (num_digits+1)/2; if ( len > sizeof(address->data) ) return -1; memcpy( address->data, buf+2, len ); return 0; } int sms_address_to_bytes( SmsAddress address, unsigned char* buf, int bufsize ) { int len = (address->len + 1)/2 + 2; if (buf == NULL) bufsize = 0; if (bufsize < 1) goto Exit; buf[0] = address->len; if (bufsize < 2) goto Exit; buf[1] = address->toa; buf += 2; bufsize -= 2; if (bufsize > len-2) bufsize = len - 2; memcpy( buf, address->data, bufsize ); Exit: return len; } int sms_address_from_hex ( SmsAddress address, const char* hex, int hexlen ) { const char* hexend = hex + hexlen; int nn, len, num_digits; if (hexlen < 4) return -1; address->len = num_digits = gsm_hex2_to_byte( hex ); address->toa = gsm_hex2_to_byte( hex+2 ); hex += 4; len = (num_digits + 1)/2; if (hex + len*2 > hexend) return -1; for ( nn = 0; nn < len; nn++ ) address->data[nn] = gsm_hex2_to_byte( hex + nn*2 ); return 0; } int sms_address_to_hex ( SmsAddress address, char* hex, int hexlen ) { int len = (address->len + 1)/2 + 2; int nn; if (hex == NULL) hexlen = 0; if (hexlen < 2) goto Exit; gsm_hex_from_byte( hex, address->len ); if (hexlen < 4) goto Exit; gsm_hex_from_byte( hex+2, address->toa ); hex += 4; hexlen -= 4; if ( hexlen > 2*(len - 2) ) hexlen = (len - 2)/2; for ( nn = 0; nn < hexlen; nn += 2 ) gsm_hex_from_byte( hex+nn, address->data[nn/2] ); Exit: return len*2; } static void gsm_rope_add_address( GsmRope rope, const SmsAddressRec* addr ) { gsm_rope_add_c( rope, addr->len ); gsm_rope_add_c( rope, addr->toa ); gsm_rope_add( rope, addr->data, (addr->len+1)/2 ); if (addr->len & 1) { if (!rope->error && rope->data != NULL) rope->data[ rope->pos-1 ] |= 0xf0; } } static int sms_address_eq( const SmsAddressRec* addr1, const SmsAddressRec* addr2 ) { if ( addr1->toa != addr2->toa || addr1->len != addr2->len ) return 0; return ( !memcmp( addr1->data, addr2->data, addr1->len ) ); } /** SMS PARSER **/ static int sms_get_byte( cbytes_t *pcur, cbytes_t end ) { cbytes_t cur = *pcur; int result = -1; if (cur < end) { result = cur[0]; *pcur = cur + 1; } return result; } /* parse a service center address, returns -1 in case of error */ static int sms_get_sc_address( cbytes_t *pcur, cbytes_t end, SmsAddress address ) { cbytes_t cur = *pcur; int result = -1; if (cur < end) { int len = cur[0]; int dlen, adjust = 0; cur += 1; if (len == 0) { /* empty address */ address->len = 0; address->toa = 0x00; result = 0; goto Exit; } if (cur + len > end) { goto Exit; } address->toa = *cur++; len -= 1; result = 0; for (dlen = 0; dlen < len; dlen+=1) { int c = cur[dlen]; int v; adjust = 0; if (dlen >= sizeof(address->data)) { result = -1; break; } v = (c & 0xf); if (v >= 0xe) break; adjust = 1; address->data[dlen] = (byte_t) c; v = (c >> 4) & 0xf; if (v >= 0xe) { break; } } address->len = 2*dlen + adjust; } Exit: if (!result) *pcur = cur; return result; } static int sms_skip_sc_address( cbytes_t *pcur, cbytes_t end ) { cbytes_t cur = *pcur; int result = -1; int len; if (cur >= end) goto Exit; len = cur[0]; cur += 1 + len; if (cur > end) goto Exit; *pcur = cur; result = 0; Exit: return result; } /* parse a sender/receiver address, returns -1 in case of error */ static int sms_get_address( cbytes_t *pcur, cbytes_t end, SmsAddress address ) { cbytes_t cur = *pcur; int result = -1; int len, dlen; if (cur >= end) goto Exit; dlen = *cur++; if (dlen == 0) { address->len = 0; address->toa = 0; result = 0; goto Exit; } if (cur + 1 + (dlen+1)/2 > end) goto Exit; address->len = dlen; address->toa = *cur++; len = (dlen + 1)/2; if (len > sizeof(address->data)) goto Exit; memcpy( address->data, cur, len ); cur += len; result = 0; Exit: if (!result) *pcur = cur; return result; } static int sms_skip_address( cbytes_t *pcur, cbytes_t end ) { cbytes_t cur = *pcur; int result = -1; int dlen; if (cur + 2 > end) goto Exit; dlen = cur[0]; cur += 2 + (dlen + 1)/2; if (cur > end) goto Exit; result = 0; Exit: return result; } /* parse a service center timestamp */ static int sms_get_timestamp( cbytes_t *pcur, cbytes_t end, SmsTimeStamp ts ) { cbytes_t cur = *pcur; if (cur + 7 > end) return -1; memcpy( ts->data, cur, 7 ); *pcur = cur + 7; return 0; } static int sms_skip_timestamp( cbytes_t *pcur, cbytes_t end ) { cbytes_t cur = *pcur; if (cur + 7 > end) return -1; *pcur = cur + 7; return 0; } static int sms_skip_validity_period( cbytes_t *pcur, cbytes_t end, int mtiByte ) { cbytes_t cur = *pcur; switch ((mtiByte >> 3) & 3) { case 1: /* relative format */ cur += 1; break; case 2: /* enhanced format */ case 3: /* absolute format */ cur += 7; } if (cur > end) return -1; *pcur = cur; return 0; } /** SMS PDU **/ typedef struct SmsPDURec { bytes_t base; bytes_t end; bytes_t tpdu; } SmsPDURec; void smspdu_free( SmsPDU pdu ) { if (pdu) { free( pdu->base ); pdu->base = NULL; pdu->end = NULL; pdu->tpdu = NULL; } } SmsPduType smspdu_get_type( SmsPDU pdu ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte(&data, end); switch (mtiByte & 3) { case 0: return SMS_PDU_DELIVER; case 1: return SMS_PDU_SUBMIT; case 2: return SMS_PDU_STATUS_REPORT; default: return SMS_PDU_INVALID; } } int smspdu_get_sender_address( SmsPDU pdu, SmsAddress address ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte(&data, end); switch (mtiByte & 3) { case 0: /* SMS_PDU_DELIVER; */ return sms_get_sc_address( &data, end, address ); default: return -1; } } int smspdu_get_sc_timestamp( SmsPDU pdu, SmsTimeStamp ts ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte( &data, end ); switch (mtiByte & 3) { case 0: /* SMS_PDU_DELIVER */ { SmsAddressRec address; if ( sms_get_sc_address( &data, end, &address ) < 0 ) return -1; data += 2; /* skip protocol identifer + coding scheme */ return sms_get_timestamp( &data, end, ts ); } default: return -1; } } int smspdu_get_receiver_address( SmsPDU pdu, SmsAddress address ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte( &data, end ); switch (mtiByte & 3) { case 1: /* SMS_PDU_SUBMIT */ { data += 1; /* skip message reference */ return sms_get_address( &data, end, address ); } default: return -1; } } typedef enum { SMS_CODING_SCHEME_UNKNOWN = 0, SMS_CODING_SCHEME_GSM7, SMS_CODING_SCHEME_UCS2 } SmsCodingScheme; /* see TS 23.038 Section 5 for details */ static SmsCodingScheme sms_get_coding_scheme( cbytes_t *pcur, cbytes_t end ) { cbytes_t cur = *pcur; int dataCoding; if (cur >= end) return SMS_CODING_SCHEME_UNKNOWN; dataCoding = *cur++; *pcur = cur; switch (dataCoding >> 4) { case 0x00: case 0x02: case 0x03: return SMS_CODING_SCHEME_GSM7; case 0x01: if (dataCoding == 0x10) return SMS_CODING_SCHEME_GSM7; if (dataCoding == 0x11) return SMS_CODING_SCHEME_UCS2; break; case 0x04: case 0x05: case 0x06: case 0x07: if (dataCoding & 0x20) return SMS_CODING_SCHEME_UNKNOWN; /* compressed 7-bits */ if (((dataCoding >> 2) & 3) == 0) return SMS_CODING_SCHEME_GSM7; if (((dataCoding >> 2) & 3) == 2) return SMS_CODING_SCHEME_UCS2; break; case 0xF: if (!(dataCoding & 4)) return SMS_CODING_SCHEME_GSM7; break; } return SMS_CODING_SCHEME_UNKNOWN; } /* see TS 23.040 section 9.2.3.24 for details */ static int sms_get_text_utf8( cbytes_t *pcur, cbytes_t end, int hasUDH, SmsCodingScheme coding, GsmRope rope ) { cbytes_t cur = *pcur; int result = -1; int len; if (cur >= end) goto Exit; len = *cur++; /* skip user data header if any */ if ( hasUDH ) { int hlen; if (cur >= end) goto Exit; hlen = *cur++; if (cur + hlen > end) goto Exit; cur += hlen; if (coding == SMS_CODING_SCHEME_GSM7) len -= 2*(hlen+1); else len -= hlen+1; if (len < 0) goto Exit; } /* switch the user data header if any */ if (coding == SMS_CODING_SCHEME_GSM7) { int count = utf8_from_gsm7( cur, 0, len, NULL ); if (rope != NULL) { bytes_t dst = gsm_rope_reserve( rope, count ); if (dst != NULL) utf8_from_gsm7( cur, 0, len, dst ); } cur += (len+1)/2; } else if (coding == SMS_CODING_SCHEME_UCS2) { int count = ucs2_to_utf8( cur, len/2, NULL ); if (rope != NULL) { bytes_t dst = gsm_rope_reserve( rope, count ); if (dst != NULL) ucs2_to_utf8( cur, len/2, dst ); } cur += len; } result = 0; Exit: if (!result) *pcur = cur; return result; } /* get the message embedded in a SMS PDU as a utf8 byte array, returns the length of the message in bytes */ /* or -1 in case of error */ int smspdu_get_text_message( SmsPDU pdu, unsigned char* utf8, int utf8len ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte( &data, end ); switch (mtiByte & 3) { case 0: /* SMS_PDU_DELIVER */ { SmsAddressRec address; SmsTimeStampRec timestamp; SmsCodingScheme coding; GsmRopeRec rope[1]; int result; if ( sms_get_sc_address( &data, end, &address ) < 0 ) goto Fail; data += 1; /* skip protocol identifier */ coding = sms_get_coding_scheme( &data, end ); if (coding == SMS_CODING_SCHEME_UNKNOWN) goto Fail; if ( sms_get_timestamp( &data, end, ×tamp ) < 0 ) goto Fail; if ( sms_get_text_utf8( &data, end, (mtiByte & 0x40), coding, rope ) < 0 ) goto Fail; result = rope->pos; if (utf8len > result) utf8len = result; if (utf8len > 0) memcpy( utf8, rope->data, utf8len ); gsm_rope_done( rope ); return result; } case 1: /* SMS_PDU_SUBMIT */ { SmsAddressRec address; SmsCodingScheme coding; GsmRopeRec rope[1]; int result; data += 1; /* message reference */ if ( sms_get_address( &data, end, &address ) < 0 ) goto Fail; data += 1; /* skip protocol identifier */ coding = sms_get_coding_scheme( &data, end ); if (coding == SMS_CODING_SCHEME_UNKNOWN) goto Fail; gsm_rope_init_alloc( rope, 0 ); if ( sms_get_text_utf8( &data, end, (mtiByte & 0x40), coding, rope ) < 0 ) { gsm_rope_done( rope ); goto Fail; } result = rope->pos; if (utf8len > result) utf8len = result; if (utf8len > 0) memcpy( utf8, rope->data, utf8len ); gsm_rope_done( rope ); return result; } } Fail: return -1; } static cbytes_t smspdu_get_user_data_ref( SmsPDU pdu ) { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte( &data, end ); int len; /* if there is no user-data-header, there is no message reference here */ if ((mtiByte & 0x40) == 0) goto Fail; switch (mtiByte & 3) { case 0: /* SMS_PDU_DELIVER */ if ( sms_skip_address( &data, end ) < 0 ) goto Fail; data += 2; /* skip protocol identifier + coding scheme */ if ( sms_skip_timestamp( &data, end ) < 0 ) goto Fail; break; case 1: /* SMS_PDU_SUBMIT */ data += 1; /* skip message reference */ if ( sms_skip_address( &data, end ) < 0 ) goto Fail; data += 2; /* protocol identifier + oding schene */ if ( sms_skip_validity_period( &data, end, mtiByte ) < 0 ) goto Fail; break; default: goto Fail; } /* skip user-data length */ if (data+1 >= end) goto Fail; len = data[1]; data += 2; while (len >= 2 && data + 2 <= end) { int htype = data[0]; int hlen = data[1]; if (htype == 00 && hlen == 3 && data + 5 <= end) { return data + 2; } data += hlen; len -= hlen - 2; } Fail: return NULL; } int smspdu_get_ref( SmsPDU pdu ) { cbytes_t user_ref = smspdu_get_user_data_ref( pdu ); if (user_ref != NULL) { return user_ref[0]; } else { cbytes_t data = pdu->tpdu; cbytes_t end = pdu->end; int mtiByte = sms_get_byte( &data, end ); if ((mtiByte & 3) == 1) { /* try to extract directly the reference for a SMS-SUBMIT */ if (data < end) return data[0]; } } return -1; } int smspdu_get_max_index( SmsPDU pdu ) { cbytes_t user_ref = smspdu_get_user_data_ref( pdu ); if (user_ref != NULL) { return user_ref[1]; } else { return 1; } } int smspdu_get_cur_index( SmsPDU pdu ) { cbytes_t user_ref = smspdu_get_user_data_ref( pdu ); if (user_ref != NULL) { return user_ref[2] - 1; } else { return 0; } } static void gsm_rope_add_sms_user_header( GsmRope rope, int ref_number, int pdu_count, int pdu_index ) { gsm_rope_add_c( rope, 0x05 ); /* total header length == 5 bytes */ gsm_rope_add_c( rope, 0x00 ); /* element id: concatenated message reference number */ gsm_rope_add_c( rope, 0x03 ); /* element len: 3 bytes */ gsm_rope_add_c( rope, (byte_t)ref_number ); /* reference number */ gsm_rope_add_c( rope, (byte_t)pdu_count ); /* max pdu index */ gsm_rope_add_c( rope, (byte_t)pdu_index+1 ); /* current pdu index */ } /* write a SMS-DELIVER PDU into a rope */ static void gsm_rope_add_sms_deliver_pdu( GsmRope rope, cbytes_t utf8, int utf8len, int use_gsm7, const SmsAddressRec* sender_address, const SmsTimeStampRec* timestamp, int ref_num, int pdu_count, int pdu_index) { int coding; int mtiByte = 0x20; /* message type - SMS DELIVER */ if (pdu_count > 1) mtiByte |= 0x40; /* user data header indicator */ gsm_rope_add_c( rope, 0 ); /* no SC Address */ gsm_rope_add_c( rope, mtiByte ); /* message type - SMS-DELIVER */ gsm_rope_add_address( rope, sender_address ); gsm_rope_add_c( rope, 0 ); /* protocol identifier */ /* data coding scheme - GSM 7 bits / no class - or - 16-bit UCS2 / class 1 */ coding = (use_gsm7 ? 0x00 : 0x09); gsm_rope_add_c( rope, coding ); /* data coding scheme */ gsm_rope_add_timestamp( rope, timestamp ); /* service center timestamp */ if (use_gsm7) { bytes_t dst; int count = utf8_to_gsm7( utf8, utf8len, NULL, 0 ); int pad = 0; assert( count <= MAX_USER_DATA_SEPTETS - USER_DATA_HEADER_SIZE ); if (pdu_count > 1) { int headerBits = 6*8; /* 6 is size of header in bytes */ int headerSeptets = headerBits / 7; if (headerBits % 7 > 0) headerSeptets += 1; pad = headerSeptets*7 - headerBits; gsm_rope_add_c( rope, count + headerSeptets ); gsm_rope_add_sms_user_header(rope, ref_num, pdu_count, pdu_index); } else gsm_rope_add_c( rope, count ); count = (count*7+pad+7)/8; /* convert to byte count */ dst = gsm_rope_reserve( rope, count ); if (dst != NULL) { utf8_to_gsm7( utf8, utf8len, dst, pad ); } } else { bytes_t dst; int count = utf8_to_ucs2( utf8, utf8len, NULL ); assert( count*2 <= MAX_USER_DATA_BYTES - USER_DATA_HEADER_SIZE ); if (pdu_count > 1) { gsm_rope_add_c( rope, count*2 + 6 ); gsm_rope_add_sms_user_header( rope, ref_num, pdu_count, pdu_index ); } else gsm_rope_add_c( rope, count*2 ); gsm_rope_add_c( rope, count*2 ); dst = gsm_rope_reserve( rope, count*2 ); if (dst != NULL) { utf8_to_ucs2( utf8, utf8len, dst ); } } } static SmsPDU smspdu_create_deliver( cbytes_t utf8, int utf8len, int use_gsm7, const SmsAddressRec* sender_address, const SmsTimeStampRec* timestamp, int ref_num, int pdu_count, int pdu_index ) { SmsPDU p; GsmRopeRec rope[1]; int size; p = calloc( sizeof(*p), 1 ); if (!p) goto Exit; gsm_rope_init( rope ); gsm_rope_add_sms_deliver_pdu( rope, utf8, utf8len, use_gsm7, sender_address, timestamp, ref_num, pdu_count, pdu_index); if (rope->error) goto Fail; gsm_rope_init_alloc( rope, rope->pos ); gsm_rope_add_sms_deliver_pdu( rope, utf8, utf8len, use_gsm7, sender_address, timestamp, ref_num, pdu_count, pdu_index ); p->base = gsm_rope_done_acquire( rope, &size ); if (p->base == NULL) goto Fail; p->end = p->base + size; p->tpdu = p->base + 1; Exit: return p; Fail: free(p); return NULL; } void smspdu_free_list( SmsPDU* pdus ) { if (pdus) { int nn; for (nn = 0; pdus[nn] != NULL; nn++) smspdu_free( pdus[nn] ); free( pdus ); } } SmsPDU* smspdu_create_deliver_utf8( const unsigned char* utf8, int utf8len, const SmsAddressRec* sender_address, const SmsTimeStampRec* timestamp ) { SmsTimeStampRec ts0; int use_gsm7; int count, block; int num_pdus = 0; int leftover = 0; SmsPDU* list = NULL; static unsigned char ref_num = 0; if (timestamp == NULL) { sms_timestamp_now( &ts0 ); timestamp = &ts0; } /* can we encode the message with the GSM 7-bit alphabet ? */ use_gsm7 = utf8_check_gsm7( utf8, utf8len ); /* count the number of SMS PDUs we'll need */ block = MAX_USER_DATA_SEPTETS - USER_DATA_HEADER_SIZE; if (use_gsm7) { count = utf8_to_gsm7( utf8, utf8len, NULL, 0 ); } else { count = utf8_to_ucs2( utf8, utf8len, NULL ); block = MAX_USER_DATA_BYTES - USER_DATA_HEADER_SIZE; } num_pdus = count / block; leftover = count - num_pdus*block; if (leftover > 0) num_pdus += 1; list = calloc( sizeof(SmsPDU*), num_pdus + 1 ); if (list == NULL) return NULL; /* now create each SMS PDU */ { cbytes_t src = utf8; cbytes_t src_end = utf8 + utf8len; int nn; for (nn = 0; nn < num_pdus; nn++) { int skip = block; cbytes_t src_next; if (leftover > 0 && nn == num_pdus-1) skip = leftover; src_next = utf8_skip_gsm7( src, src_end, skip ); list[nn] = smspdu_create_deliver( src, src_next - src, use_gsm7, sender_address, timestamp, ref_num, num_pdus, nn ); if (list[nn] == NULL) goto Fail; src = src_next; } } ref_num++; return list; Fail: smspdu_free_list(list); return NULL; } SmsPDU smspdu_create_from_hex( const char* hex, int hexlen ) { SmsPDU p; cbytes_t data; p = calloc( sizeof(*p), 1 ); if (!p) goto Exit; p->base = malloc( (hexlen+1)/2 ); if (p->base == NULL) { free(p); p = NULL; goto Exit; } if ( gsm_hex_to_bytes( (cbytes_t)hex, hexlen, p->base ) < 0 ) goto Fail; p->end = p->base + (hexlen+1)/2; data = p->base; if ( sms_skip_sc_address( &data, p->end ) < 0 ) goto Fail; p->tpdu = (bytes_t) data; Exit: return p; Fail: free(p->base); free(p); return NULL; } int smspdu_to_hex( SmsPDU pdu, char* hex, int hexlen ) { int result = (pdu->end - pdu->base)*2; int nn; if (hexlen > result) hexlen = result; for (nn = 0; nn*2 < hexlen; nn++) { gsm_hex_from_byte( &hex[nn*2], pdu->base[nn] ); } return result; } /** SMS SUBMIT RECEIVER ** collects one or more SMS-SUBMIT PDUs to generate a single message to deliver **/ typedef struct SmsFragmentRec { struct SmsFragmentRec* next; SmsAddressRec from[1]; byte_t ref; byte_t max; byte_t count; int index; SmsPDU* pdus; } SmsFragmentRec, *SmsFragment; typedef struct SmsReceiverRec { int last; SmsFragment fragments; } SmsReceiverRec; static void sms_fragment_free( SmsFragment frag ) { int nn; for (nn = 0; nn < frag->max; nn++) { if (frag->pdus[nn] != NULL) { smspdu_free( frag->pdus[nn] ); frag->pdus[nn] = NULL; } } frag->pdus = NULL; frag->count = 0; frag->max = 0; frag->index = 0; free( frag ); } static SmsFragment sms_fragment_alloc( SmsReceiver rec, const SmsAddressRec* from, int ref, int max ) { SmsFragment frag = calloc(sizeof(*frag) + max*sizeof(SmsPDU), 1 ); if (frag != NULL) { frag->from[0] = from[0]; frag->ref = ref; frag->max = max; frag->pdus = (SmsPDU*)(frag + 1); frag->index = ++rec->last; } return frag; } SmsReceiver sms_receiver_create( void ) { SmsReceiver rec = calloc(sizeof(*rec),1); return rec; } void sms_receiver_destroy( SmsReceiver rec ) { while (rec->fragments) { SmsFragment frag = rec->fragments; rec->fragments = frag->next; sms_fragment_free(frag); } } static SmsFragment* sms_receiver_find_p( SmsReceiver rec, const SmsAddressRec* from, int ref ) { SmsFragment* pnode = &rec->fragments; SmsFragment node; for (;;) { node = *pnode; if (node == NULL) break; if (node->ref == ref && sms_address_eq( node->from, from )) break; pnode = &node->next; } return pnode; } static SmsFragment* sms_receiver_find_index_p( SmsReceiver rec, int index ) { SmsFragment* pnode = &rec->fragments; SmsFragment node; for (;;) { node = *pnode; if (node == NULL) break; if (node->index == index) break; pnode = &node->next; } return pnode; } int sms_receiver_add_submit_pdu( SmsReceiver rec, SmsPDU submit_pdu ) { SmsAddressRec from[1]; int ref, max, cur; SmsFragment* pnode; SmsFragment frag; if ( smspdu_get_receiver_address( submit_pdu, from ) < 0 ) { D( "%s: could not extract receiver address\n", __FUNCTION__ ); return -1; } ref = smspdu_get_ref( submit_pdu ); if (ref < 0) { D( "%s: could not extract message reference from pdu\n", __FUNCTION__ ); return -1; } max = smspdu_get_max_index( submit_pdu ); if (max < 0) { D( "%s: invalid max fragment value: %d should be >= 1\n", __FUNCTION__, max ); return -1; } pnode = sms_receiver_find_p( rec, from, ref ); frag = *pnode; if (frag == NULL) { frag = sms_fragment_alloc( rec, from, ref, max ); if (frag == NULL) { D("%s: not enough memory to allocate new fragment\n", __FUNCTION__ ); return -1; } if (D_ACTIVE) { char tmp[32]; int len; len = sms_address_to_str( from, tmp, sizeof(tmp) ); if (len < 0) { strcpy( tmp, "<unknown>" ); len = strlen(tmp); } D("%s: created SMS index %d, from %.*s, ref %d, max %d\n", __FUNCTION__, frag->index, len, tmp, frag->ref, frag->max); } *pnode = frag; } cur = smspdu_get_cur_index( submit_pdu ); if (cur < 0) { D("%s: SMS fragment index is too small: %d should be >= 1\n", __FUNCTION__, cur+1 ); return -1; } if (cur >= max) { D("%s: SMS fragment index is too large (%d >= %d)\n", __FUNCTION__, cur, max); return -1; } if ( frag->pdus[cur] != NULL ) { D("%s: receiving duplicate SMS fragment for %d/%d, ref=%d, discarding old one\n", __FUNCTION__, cur+1, max, ref); smspdu_free( frag->pdus[cur] ); frag->count -= 1; } frag->pdus[cur] = submit_pdu; frag->count += 1; if (frag->count >= frag->max) { /* yes, we received all fragments for this SMS */ D( "%s: SMS index %d, received all %d fragments\n", __FUNCTION__, frag->index, frag->count ); return frag->index; } else { /* still waiting for more */ D( "%s: SMS index %d, received %d/%d, waiting for %d more\n", __FUNCTION__, frag->index, cur+1, max, frag->max - frag->count ); return 0; } } int sms_receiver_get_text_message( SmsReceiver rec, int index, bytes_t utf8, int utf8len ) { SmsFragment* pnode = sms_receiver_find_index_p( rec, index ); SmsFragment frag = *pnode; int nn, total; if (frag == NULL) { D( "%s: invalid SMS index %d\n", __FUNCTION__, index ); return -1; } if (frag->count != frag->max) { D( "%s: SMS index %d still needs %d fragments\n", __FUNCTION__, frag->index, frag->max - frag->count ); return -1; } /* get the size of all combined text */ total = 0; for ( nn = 0; nn < frag->count; nn++ ) { int partial; if (utf8 && utf8len > 0) { partial = smspdu_get_text_message( frag->pdus[nn], utf8, utf8len ); utf8 += partial; utf8len -= partial; } else { partial = smspdu_get_text_message( frag->pdus[nn], NULL, 0 ); } total += partial; } return total; } static void sms_receiver_remove( SmsReceiver rec, int index ) { SmsFragment* pnode = sms_receiver_find_index_p( rec, index ); SmsFragment frag = *pnode; if (frag != NULL) { *pnode = frag->next; sms_fragment_free(frag); } } SmsPDU* sms_receiver_create_deliver( SmsReceiver rec, int index, const SmsAddressRec* from ) { SmsPDU* result = NULL; SmsFragment* pnode = sms_receiver_find_index_p( rec, index ); SmsFragment frag = *pnode; SmsTimeStampRec now[1]; int nn, total; bytes_t utf8; int utf8len; if (frag == NULL) { D( "%s: invalid SMS index %d\n", __FUNCTION__, index ); return NULL; } if (frag->count != frag->max) { D( "%s: SMS index %d still needs %d fragments\n", __FUNCTION__, frag->index, frag->max - frag->count ); return NULL; } /* get the combined text message */ utf8len = sms_receiver_get_text_message( rec, index, NULL, 0 ); if (utf8len < 0) goto Exit; utf8 = malloc( utf8len + 1 ); if (utf8 == NULL) { D( "%s: not enough memory to allocate %d bytes\n", __FUNCTION__, utf8len+1 ); goto Exit; } total = 0; for ( nn = 0; nn < frag->count; nn++ ) { total += smspdu_get_text_message( frag->pdus[nn], utf8 + total, utf8len - total ); } sms_timestamp_now( now ); result = smspdu_create_deliver_utf8( utf8, utf8len, from, now ); free(utf8); Exit: sms_receiver_remove( rec, index ); return result; }