Home | History | Annotate | Download | only in map
      1 /*
      2 * Copyright (C) 2013 Samsung System LSI
      3 * Licensed under the Apache License, Version 2.0 (the "License");
      4 * you may not use this file except in compliance with the License.
      5 * You may obtain a copy of the License at
      6 *
      7 *      http://www.apache.org/licenses/LICENSE-2.0
      8 *
      9 * Unless required by applicable law or agreed to in writing, software
     10 * distributed under the License is distributed on an "AS IS" BASIS,
     11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     12 * See the License for the specific language governing permissions and
     13 * limitations under the License.
     14 */
     15 package com.android.bluetooth.map;
     16 
     17 import android.database.Cursor;
     18 import android.util.Base64;
     19 import android.util.Log;
     20 
     21 import com.android.bluetooth.mapapi.BluetoothMapContract;
     22 
     23 import java.io.ByteArrayOutputStream;
     24 import java.io.UnsupportedEncodingException;
     25 import java.nio.charset.Charset;
     26 import java.nio.charset.IllegalCharsetNameException;
     27 import java.text.SimpleDateFormat;
     28 import java.util.Arrays;
     29 import java.util.BitSet;
     30 import java.util.Date;
     31 import java.util.regex.Matcher;
     32 import java.util.regex.Pattern;
     33 
     34 
     35 /**
     36  * Various utility methods and generic defines that can be used throughout MAPS
     37  */
     38 public class BluetoothMapUtils {
     39 
     40     private static final String TAG = "BluetoothMapUtils";
     41     private static final boolean D = BluetoothMapService.DEBUG;
     42     private static final boolean V = BluetoothMapService.VERBOSE;
     43     /* We use the upper 4 bits for the type mask.
     44      * TODO: When more types are needed, consider just using a number
     45      *       in stead of a bit to indicate the message type. Then 4
     46      *       bit can be use for 16 different message types.
     47      */
     48     private static final long HANDLE_TYPE_MASK            = (((long)0xff)<<56);
     49     private static final long HANDLE_TYPE_MMS_MASK        = (((long)0x01)<<56);
     50     private static final long HANDLE_TYPE_EMAIL_MASK      = (((long)0x02)<<56);
     51     private static final long HANDLE_TYPE_SMS_GSM_MASK    = (((long)0x04)<<56);
     52     private static final long HANDLE_TYPE_SMS_CDMA_MASK   = (((long)0x08)<<56);
     53     private static final long HANDLE_TYPE_IM_MASK         = (((long)0x10)<<56);
     54 
     55     public static final long CONVO_ID_TYPE_SMS_MMS = 1;
     56     public static final long CONVO_ID_TYPE_EMAIL_IM= 2;
     57 
     58     // MAP supported feature bit - included from MAP Spec 1.2
     59     static final int MAP_FEATURE_DEFAULT_BITMASK                    = 0x0000001F;
     60 
     61     static final int MAP_FEATURE_NOTIFICATION_REGISTRATION_BIT      = 1 << 0;
     62     static final int MAP_FEATURE_NOTIFICATION_BIT                   = 1 << 1;
     63     static final int MAP_FEATURE_BROWSING_BIT                       = 1 << 2;
     64     static final int MAP_FEATURE_UPLOADING_BIT                      = 1 << 3;
     65     static final int MAP_FEATURE_DELETE_BIT                         = 1 << 4;
     66     static final int MAP_FEATURE_INSTANCE_INFORMATION_BIT           = 1 << 5;
     67     static final int MAP_FEATURE_EXTENDED_EVENT_REPORT_11_BIT       = 1 << 6;
     68     static final int MAP_FEATURE_EVENT_REPORT_V12_BIT               = 1 << 7;
     69     static final int MAP_FEATURE_MESSAGE_FORMAT_V11_BIT             = 1 << 8;
     70     static final int MAP_FEATURE_MESSAGE_LISTING_FORMAT_V11_BIT     = 1 << 9;
     71     static final int MAP_FEATURE_PERSISTENT_MESSAGE_HANDLE_BIT      = 1 << 10;
     72     static final int MAP_FEATURE_DATABASE_INDENTIFIER_BIT           = 1 << 11;
     73     static final int MAP_FEATURE_FOLDER_VERSION_COUNTER_BIT         = 1 << 12;
     74     static final int MAP_FEATURE_CONVERSATION_VERSION_COUNTER_BIT   = 1 << 13;
     75     static final int MAP_FEATURE_PARTICIPANT_PRESENCE_CHANGE_BIT    = 1 << 14;
     76     static final int MAP_FEATURE_PARTICIPANT_CHAT_STATE_CHANGE_BIT  = 1 << 15;
     77 
     78     static final int MAP_FEATURE_PBAP_CONTACT_CROSS_REFERENCE_BIT   = 1 << 16;
     79     static final int MAP_FEATURE_NOTIFICATION_FILTERING_BIT         = 1 << 17;
     80     static final int MAP_FEATURE_DEFINED_TIMESTAMP_FORMAT_BIT       = 1 << 18;
     81 
     82     static final String MAP_V10_STR = "1.0";
     83     static final String MAP_V11_STR = "1.1";
     84     static final String MAP_V12_STR = "1.2";
     85 
     86     // Event Report versions
     87     static final int MAP_EVENT_REPORT_V10           = 10; // MAP spec 1.1
     88     static final int MAP_EVENT_REPORT_V11           = 11; // MAP spec 1.2
     89     static final int MAP_EVENT_REPORT_V12           = 12; // MAP spec 1.3 'to be' incl. IM
     90 
     91     // Message Format versions
     92     static final int MAP_MESSAGE_FORMAT_V10         = 10; // MAP spec below 1.3
     93     static final int MAP_MESSAGE_FORMAT_V11         = 11; // MAP spec 1.3
     94 
     95     // Message Listing Format versions
     96     static final int MAP_MESSAGE_LISTING_FORMAT_V10 = 10; // MAP spec below 1.3
     97     static final int MAP_MESSAGE_LISTING_FORMAT_V11 = 11; // MAP spec 1.3
     98 
     99     /**
    100      * This enum is used to convert from the bMessage type property to a type safe
    101      * type. Hence do not change the names of the enum values.
    102      */
    103     public enum TYPE{
    104         NONE,
    105         EMAIL,
    106         SMS_GSM,
    107         SMS_CDMA,
    108         MMS,
    109         IM;
    110         private static TYPE[] allValues = values();
    111         public static TYPE fromOrdinal(int n) {
    112             if(n < allValues.length)
    113                return allValues[n];
    114             return NONE;
    115         }
    116     }
    117 
    118     static public String getDateTimeString(long timestamp) {
    119         SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd'T'HHmmss");
    120         Date date = new Date(timestamp);
    121         return format.format(date); // Format to YYYYMMDDTHHMMSS local time
    122     }
    123 
    124 
    125     public static void printCursor(Cursor c) {
    126         if (D) {
    127             StringBuilder sb = new StringBuilder();
    128             sb.append("\nprintCursor:\n");
    129             for(int i = 0; i < c.getColumnCount(); i++) {
    130                 if(c.getColumnName(i).equals(BluetoothMapContract.MessageColumns.DATE) ||
    131                    c.getColumnName(i).equals(
    132                            BluetoothMapContract.ConversationColumns.LAST_THREAD_ACTIVITY) ||
    133                    c.getColumnName(i).equals(BluetoothMapContract.ChatStatusColumns.LAST_ACTIVE) ||
    134                    c.getColumnName(i).equals(BluetoothMapContract.PresenceColumns.LAST_ONLINE) ){
    135                     sb.append("  ").append(c.getColumnName(i)).append(" : ").append(
    136                             getDateTimeString(c.getLong(i))).append("\n");
    137                 } else {
    138                     sb.append("  ").append(c.getColumnName(i)).append(" : ").append(
    139                             c.getString(i)).append("\n");
    140                 }
    141             }
    142             Log.d(TAG, sb.toString());
    143         }
    144     }
    145 
    146     public static String getLongAsString(long v) {
    147         char[] result = new char[16];
    148         int v1 = (int) (v & 0xffffffff);
    149         int v2 = (int) ((v>>32) & 0xffffffff);
    150         int c;
    151         for (int i = 0; i < 8; i++) {
    152             c = v2 & 0x0f;
    153             c += (c < 10) ? '0' : ('A'-10);
    154             result[7 - i] = (char) c;
    155             v2 >>= 4;
    156             c = v1 & 0x0f;
    157             c += (c < 10) ? '0' : ('A'-10);
    158             result[15 - i] = (char)c;
    159             v1 >>= 4;
    160         }
    161         return new String(result);
    162     }
    163 
    164     /**
    165      * Converts a hex-string to a long - please mind that Java has no unsigned data types, hence
    166      * any value passed to this function, which has the upper bit set, will return a negative value.
    167      * The bitwise content of the variable will however be the same.
    168      * Will ignore any white-space characters as well as '-' seperators
    169      * @param valueStr a hexstring - NOTE: shall not contain any "0x" prefix.
    170      * @return
    171      * @throws UnsupportedEncodingException if "US-ASCII" charset is not supported,
    172      * NullPointerException if a null pointer is passed to the function,
    173      * NumberFormatException if the string contains invalid characters.
    174      *
    175      */
    176     public static long getLongFromString(String valueStr) throws UnsupportedEncodingException {
    177         if(valueStr == null) throw new NullPointerException();
    178         if(V) Log.i(TAG, "getLongFromString(): converting: " + valueStr);
    179         byte[] nibbles;
    180         nibbles = valueStr.getBytes("US-ASCII");
    181         if(V) Log.i(TAG, "  byte values: " + Arrays.toString(nibbles));
    182         byte c;
    183         int count = 0;
    184         int length = nibbles.length;
    185         long value = 0;
    186         for(int i = 0; i != length; i++) {
    187             c = nibbles[i];
    188             if(c >= '0' && c <= '9') {
    189                 c -= '0';
    190             } else if(c >= 'A' && c <= 'F') {
    191                 c -= ('A'-10);
    192             } else if(c >= 'a' && c <= 'f') {
    193                 c -= ('a'-10);
    194             } else if(c <= ' ' || c == '-') {
    195                 if(V)Log.v(TAG, "Skipping c = '" + new String(new byte[]{ (byte)c }, "US-ASCII")
    196                         + "'");
    197                 continue; // Skip any whitespace and '-' (which is used for UUIDs)
    198             } else {
    199                 throw new NumberFormatException("Invalid character:" + c);
    200             }
    201             value = value << 4; // The last nibble shall not be shifted
    202             value += c;
    203             count++;
    204             if(count > 16) throw new NullPointerException("String to large - count: " + count);
    205         }
    206         if(V) Log.i(TAG, "  length: " + count);
    207         return value;
    208     }
    209     private static final int LONG_LONG_LENGTH = 32;
    210     public static String getLongLongAsString(long vLow, long vHigh) {
    211         char[] result = new char[LONG_LONG_LENGTH];
    212         int v1 = (int) (vLow & 0xffffffff);
    213         int v2 = (int) ((vLow>>32) & 0xffffffff);
    214         int v3 = (int) (vHigh & 0xffffffff);
    215         int v4 = (int) ((vHigh>>32) & 0xffffffff);
    216         int c,d,i;
    217         // Handle the lower bytes
    218         for (i = 0; i < 8; i++) {
    219             c = v2 & 0x0f;
    220             c += (c < 10) ? '0' : ('A'-10);
    221             d = v4 & 0x0f;
    222             d += (d < 10) ? '0' : ('A'-10);
    223             result[23 - i] = (char) c;
    224             result[7 - i] = (char) d;
    225             v2 >>= 4;
    226             v4 >>= 4;
    227             c = v1 & 0x0f;
    228             c += (c < 10) ? '0' : ('A'-10);
    229             d = v3 & 0x0f;
    230             d += (d < 10) ? '0' : ('A'-10);
    231             result[31 - i] = (char)c;
    232             result[15 - i] = (char)d;
    233             v1 >>= 4;
    234             v3 >>= 4;
    235         }
    236         // Remove any leading 0's
    237         for(i = 0; i < LONG_LONG_LENGTH; i++) {
    238             if(result[i] != '0') {
    239                 break;
    240             }
    241         }
    242         return new String(result, i, LONG_LONG_LENGTH-i);
    243     }
    244 
    245 
    246     /**
    247      * Convert a Content Provider handle and a Messagetype into a unique handle
    248      * @param cpHandle content provider handle
    249      * @param messageType message type (TYPE_MMS/TYPE_SMS_GSM/TYPE_SMS_CDMA/TYPE_EMAIL)
    250      * @return String Formatted Map Handle
    251      */
    252     public static String getMapHandle(long cpHandle, TYPE messageType){
    253         String mapHandle = "-1";
    254         /* Avoid NPE for possible "null" value of messageType */
    255         if(messageType != null) {
    256             switch(messageType)
    257             {
    258                 case MMS:
    259                     mapHandle = getLongAsString(cpHandle | HANDLE_TYPE_MMS_MASK);
    260                     break;
    261                 case SMS_GSM:
    262                     mapHandle = getLongAsString(cpHandle | HANDLE_TYPE_SMS_GSM_MASK);
    263                     break;
    264                 case SMS_CDMA:
    265                     mapHandle = getLongAsString(cpHandle | HANDLE_TYPE_SMS_CDMA_MASK);
    266                     break;
    267                 case EMAIL:
    268                     mapHandle = getLongAsString(cpHandle | HANDLE_TYPE_EMAIL_MASK);
    269                     break;
    270                 case IM:
    271                     mapHandle = getLongAsString(cpHandle | HANDLE_TYPE_IM_MASK);
    272                     break;
    273                 case NONE:
    274                     break;
    275                 default:
    276                     throw new IllegalArgumentException("Message type not supported");
    277             }
    278         } else {
    279             if(D)Log.e(TAG," Invalid messageType input");
    280         }
    281         return mapHandle;
    282 
    283     }
    284 
    285     /**
    286      * Convert a Content Provider handle and a Messagetype into a unique handle
    287      * @param cpHandle content provider handle
    288      * @param messageType message type (TYPE_MMS/TYPE_SMS_GSM/TYPE_SMS_CDMA/TYPE_EMAIL)
    289      * @return String Formatted Map Handle
    290      */
    291     public static String getMapConvoHandle(long cpHandle, TYPE messageType){
    292         String mapHandle = "-1";
    293         switch(messageType)
    294         {
    295             case MMS:
    296             case SMS_GSM:
    297             case SMS_CDMA:
    298                 mapHandle = getLongLongAsString(cpHandle, CONVO_ID_TYPE_SMS_MMS);
    299                 break;
    300             case EMAIL:
    301             case IM:
    302                 mapHandle = getLongLongAsString(cpHandle, CONVO_ID_TYPE_EMAIL_IM);
    303                 break;
    304             default:
    305                 throw new IllegalArgumentException("Message type not supported");
    306         }
    307         return mapHandle;
    308 
    309     }
    310 
    311     /**
    312      * Convert a handle string the the raw long representation, including the type bit.
    313      * @param mapHandle the handle string
    314      * @return the handle value
    315      */
    316     static public long getMsgHandleAsLong(String mapHandle){
    317         return Long.parseLong(mapHandle, 16);
    318     }
    319     /**
    320      * Convert a Map Handle into a content provider Handle
    321      * @param mapHandle handle to convert from
    322      * @return content provider handle without message type mask
    323      */
    324     static public long getCpHandle(String mapHandle)
    325     {
    326         long cpHandle = getMsgHandleAsLong(mapHandle);
    327         if(D)Log.d(TAG,"-> MAP handle:"+mapHandle);
    328         /* remove masks as the call should already know what type of message this handle is for */
    329         cpHandle &= ~HANDLE_TYPE_MASK;
    330         if(D)Log.d(TAG,"->CP handle:"+cpHandle);
    331 
    332         return cpHandle;
    333     }
    334 
    335     /**
    336      * Extract the message type from the handle.
    337      * @param mapHandle
    338      * @return
    339      */
    340     static public TYPE getMsgTypeFromHandle(String mapHandle) {
    341         long cpHandle = getMsgHandleAsLong(mapHandle);
    342 
    343         if((cpHandle & HANDLE_TYPE_MMS_MASK) != 0)
    344             return TYPE.MMS;
    345         if((cpHandle & HANDLE_TYPE_EMAIL_MASK) != 0)
    346             return TYPE.EMAIL;
    347         if((cpHandle & HANDLE_TYPE_SMS_GSM_MASK) != 0)
    348             return TYPE.SMS_GSM;
    349         if((cpHandle & HANDLE_TYPE_SMS_CDMA_MASK) != 0)
    350             return TYPE.SMS_CDMA;
    351         if((cpHandle & HANDLE_TYPE_IM_MASK) != 0)
    352             return TYPE.IM;
    353 
    354         throw new IllegalArgumentException("Message type not found in handle string.");
    355     }
    356 
    357     /**
    358      * TODO: Is this still needed after changing to another XML encoder? It should escape illegal
    359      *       characters.
    360      * Strip away any illegal XML characters, that would otherwise cause the
    361      * xml serializer to throw an exception.
    362      * Examples of such characters are the emojis used on Android.
    363      * @param text The string to validate
    364      * @return the same string if valid, otherwise a new String stripped for
    365      * any illegal characters. If a null pointer is passed an empty string will be returned.
    366      */
    367     static public String stripInvalidChars(String text) {
    368         if(text == null) {
    369             return "";
    370         }
    371         char out[] = new char[text.length()];
    372         int i, o, l;
    373         for(i=0, o=0, l=text.length(); i<l; i++){
    374             char c = text.charAt(i);
    375             if((c >= 0x20 && c <= 0xd7ff) || (c >= 0xe000 && c <= 0xfffd)) {
    376                 out[o++] = c;
    377             } // Else we skip the character
    378         }
    379 
    380         if(i==o) {
    381             return text;
    382         } else { // We removed some characters, create the new string
    383             return new String(out,0,o);
    384         }
    385     }
    386 
    387     /**
    388      * Truncate UTF-8 string encoded byte array to desired length
    389      * @param utf8String String to convert to bytes array h
    390      * @param length Max length of byte array returned including null termination
    391      * @return byte array containing valid utf8 characters with max length
    392      * @throws UnsupportedEncodingException
    393      */
    394     static public byte[] truncateUtf8StringToBytearray(String utf8String, int maxLength)
    395             throws UnsupportedEncodingException {
    396 
    397         byte[] utf8Bytes = new byte[utf8String.length() + 1];
    398         try {
    399             System.arraycopy(utf8String.getBytes("UTF-8"), 0,
    400                              utf8Bytes, 0, utf8String.length());
    401         } catch (UnsupportedEncodingException e) {
    402             Log.e(TAG,"truncateUtf8StringToBytearray: getBytes exception ", e);
    403             throw e;
    404         }
    405 
    406         if (utf8Bytes.length > maxLength) {
    407             /* if 'continuation' byte is in place 200,
    408              * then strip previous bytes until utf-8 start byte is found */
    409             if ( (utf8Bytes[maxLength - 1] & 0xC0) == 0x80 ) {
    410                 for (int i = maxLength - 2; i >= 0; i--) {
    411                     if ((utf8Bytes[i] & 0xC0) == 0xC0) {
    412                         /* first byte in utf-8 character found,
    413                          * now copy i - 1 bytes to outBytes and add null termination */
    414                         utf8Bytes = Arrays.copyOf(utf8Bytes, i+1);
    415                         utf8Bytes[i] = 0;
    416                         break;
    417                     }
    418                 }
    419             } else {
    420                 /* copy bytes to outBytes and null terminate */
    421                 utf8Bytes = Arrays.copyOf(utf8Bytes, maxLength);
    422                 utf8Bytes[maxLength - 1] = 0;
    423             }
    424         }
    425         return utf8Bytes;
    426     }
    427     private static Pattern p = Pattern.compile("=\\?(.+?)\\?(.)\\?(.+?(?=\\?=))\\?=");
    428 
    429     /**
    430      * Method for converting quoted printable og base64 encoded string from headers.
    431      * @param in the string with encoding
    432      * @return decoded string if success - else the same string as was as input.
    433      */
    434     static public String stripEncoding(String in){
    435         String str = null;
    436         if(in.contains("=?") && in.contains("?=")){
    437             String encoding;
    438             String charset;
    439             String encodedText;
    440             String match;
    441             Matcher m = p.matcher(in);
    442             while(m.find()){
    443                 match = m.group(0);
    444                 charset = m.group(1);
    445                 encoding = m.group(2);
    446                 encodedText = m.group(3);
    447                 Log.v(TAG, "Matching:" + match +"\nCharset: "+charset +"\nEncoding : " +encoding
    448                         + "\nText: " + encodedText);
    449                 if(encoding.equalsIgnoreCase("Q")){
    450                     //quoted printable
    451                     Log.d(TAG,"StripEncoding: Quoted Printable string : " + encodedText);
    452                     str = new String(quotedPrintableToUtf8(encodedText,charset));
    453                     in = in.replace(match, str);
    454                 }else if(encoding.equalsIgnoreCase("B")){
    455                     // base64
    456                     try{
    457 
    458                         Log.d(TAG,"StripEncoding: base64 string : " + encodedText);
    459                         str = new String(Base64.decode(encodedText.getBytes(charset),
    460                                 Base64.DEFAULT), charset);
    461                         Log.d(TAG,"StripEncoding: decoded string : " + str);
    462                         in = in.replace(match, str);
    463                     }catch(UnsupportedEncodingException e){
    464                         Log.e(TAG, "stripEncoding: Unsupported charset: " + charset);
    465                     }catch (IllegalArgumentException e){
    466                         Log.e(TAG,"stripEncoding: string not encoded as base64: " +encodedText);
    467                     }
    468                 }else{
    469                     Log.e(TAG, "stripEncoding: Hit unknown encoding: "+encoding);
    470                 }
    471             }
    472         }
    473         return in;
    474     }
    475 
    476 
    477     /**
    478      * Convert a quoted-printable encoded string to a UTF-8 string:
    479      *  - Remove any soft line breaks: "=<CRLF>"
    480      *  - Convert all "=xx" to the corresponding byte
    481      * @param text quoted-printable encoded UTF-8 text
    482      * @return decoded UTF-8 string
    483      */
    484     public static byte[] quotedPrintableToUtf8(String text, String charset) {
    485         byte[] output = new byte[text.length()]; // We allocate for the worst case memory need
    486         byte[] input = null;
    487         try {
    488             input = text.getBytes("US-ASCII");
    489         } catch (UnsupportedEncodingException e) {
    490             /* This cannot happen as "US-ASCII" is supported for all Java implementations */ }
    491 
    492         if(input == null){
    493             return "".getBytes();
    494         }
    495 
    496         int in, out, stopCnt = input.length-2; // Leave room for peaking the next two bytes
    497 
    498         /* Algorithm:
    499          *  - Search for token, copying all non token chars
    500          * */
    501         for(in=0, out=0; in < stopCnt; in++){
    502             byte b0 = input[in];
    503             if(b0 == '=') {
    504                 byte b1 = input[++in];
    505                 byte b2 = input[++in];
    506                 if(b1 == '\r' && b2 == '\n') {
    507                     continue; // soft line break, remove all tree;
    508                 }
    509                 if(((b1 >= '0' && b1 <= '9') || (b1 >= 'A' && b1 <= 'F')
    510                         || (b1 >= 'a' && b1 <= 'f'))
    511                         && ((b2 >= '0' && b2 <= '9') || (b2 >= 'A' && b2 <= 'F')
    512                         || (b2 >= 'a' && b2 <= 'f'))) {
    513                     if(V)Log.v(TAG, "Found hex number: " + String.format("%c%c", b1, b2));
    514                     if(b1 <= '9')       b1 = (byte) (b1 - '0');
    515                     else if (b1 <= 'F') b1 = (byte) (b1 - 'A' + 10);
    516                     else if (b1 <= 'f') b1 = (byte) (b1 - 'a' + 10);
    517 
    518                     if(b2 <= '9')       b2 = (byte) (b2 - '0');
    519                     else if (b2 <= 'F') b2 = (byte) (b2 - 'A' + 10);
    520                     else if (b2 <= 'f') b2 = (byte) (b2 - 'a' + 10);
    521 
    522                     if(V)Log.v(TAG, "Resulting nibble values: " +
    523                             String.format("b1=%x b2=%x", b1, b2));
    524 
    525                     output[out++] = (byte)(b1<<4 | b2); // valid hex char, append
    526                     if(V)Log.v(TAG, "Resulting value: "  + String.format("0x%2x", output[out-1]));
    527                     continue;
    528                 }
    529                 Log.w(TAG, "Received wrongly quoted printable encoded text. " +
    530                         "Continuing at best effort...");
    531                 /* If we get a '=' without either a hex value or CRLF following, just add it and
    532                  * rewind the in counter. */
    533                 output[out++] = b0;
    534                 in -= 2;
    535                 continue;
    536             } else {
    537                 output[out++] = b0;
    538                 continue;
    539             }
    540         }
    541 
    542         // Just add any remaining characters. If they contain any encoding, it is invalid,
    543         // and best effort would be just to display the characters.
    544         while (in < input.length) {
    545             output[out++] = input[in++];
    546         }
    547 
    548         String result = null;
    549         // Figure out if we support the charset, else fall back to UTF-8, as this is what
    550         // the MAP specification suggest to use, and is compatible with US-ASCII.
    551         if(charset == null){
    552             charset = "UTF-8";
    553         } else {
    554             charset = charset.toUpperCase();
    555             try {
    556                 if(Charset.isSupported(charset) == false) {
    557                     charset = "UTF-8";
    558                 }
    559             } catch (IllegalCharsetNameException e) {
    560                 Log.w(TAG, "Received unknown charset: " + charset + " - using UTF-8.");
    561                 charset = "UTF-8";
    562             }
    563         }
    564         try{
    565             result = new String(output, 0, out, charset);
    566         } catch (UnsupportedEncodingException e) {
    567             /* This cannot happen unless Charset.isSupported() is out of sync with String */
    568             try{
    569                 result = new String(output, 0, out, "UTF-8");
    570             } catch (UnsupportedEncodingException e2) {/* This cannot happen */}
    571         }
    572         return result.getBytes(); /* return the result as "UTF-8" bytes */
    573     }
    574 
    575     /**
    576      * Encodes an array of bytes into an array of quoted-printable 7-bit characters.
    577      * Unsafe characters are escaped.
    578      * Simplified version of encoder from QuetedPrintableCodec.java (Apache external)
    579      *
    580      * @param bytes
    581      *                  array of bytes to be encoded
    582      * @return UTF-8 string containing quoted-printable characters
    583      */
    584 
    585     private static byte ESCAPE_CHAR = '=';
    586     private static byte TAB = 9;
    587     private static byte SPACE = 32;
    588 
    589     public static final String encodeQuotedPrintable(byte[] bytes) {
    590         if (bytes == null) {
    591             return null;
    592         }
    593 
    594         BitSet printable = new BitSet(256);
    595         // alpha characters
    596         for (int i = 33; i <= 60; i++) {
    597             printable.set(i);
    598         }
    599         for (int i = 62; i <= 126; i++) {
    600             printable.set(i);
    601         }
    602         printable.set(TAB);
    603         printable.set(SPACE);
    604         ByteArrayOutputStream buffer = new ByteArrayOutputStream();
    605         for (int i = 0; i < bytes.length; i++) {
    606             int b = bytes[i];
    607             if (b < 0) {
    608                 b = 256 + b;
    609             }
    610             if (printable.get(b)) {
    611                 buffer.write(b);
    612             } else {
    613                 buffer.write(ESCAPE_CHAR);
    614                 char hex1 = Character.toUpperCase(Character.forDigit((b >> 4) & 0xF, 16));
    615                 char hex2 = Character.toUpperCase(Character.forDigit(b & 0xF, 16));
    616                 buffer.write(hex1);
    617                 buffer.write(hex2);
    618             }
    619         }
    620         try {
    621             return buffer.toString("UTF-8");
    622         } catch (UnsupportedEncodingException e) {
    623             //cannot happen
    624             return "";
    625         }
    626     }
    627 
    628 }
    629 
    630