1*ec779b8eSAndroid Build Coastguard Worker /* 2*ec779b8eSAndroid Build Coastguard Worker * Copyright (C) 2010 The Android Open Source Project 3*ec779b8eSAndroid Build Coastguard Worker * 4*ec779b8eSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*ec779b8eSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*ec779b8eSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*ec779b8eSAndroid Build Coastguard Worker * 8*ec779b8eSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*ec779b8eSAndroid Build Coastguard Worker * 10*ec779b8eSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*ec779b8eSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*ec779b8eSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*ec779b8eSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*ec779b8eSAndroid Build Coastguard Worker * limitations under the License. 15*ec779b8eSAndroid Build Coastguard Worker */ 16*ec779b8eSAndroid Build Coastguard Worker 17*ec779b8eSAndroid Build Coastguard Worker #define LOG_TAG "MtpStringBuffer" 18*ec779b8eSAndroid Build Coastguard Worker 19*ec779b8eSAndroid Build Coastguard Worker #include <codecvt> 20*ec779b8eSAndroid Build Coastguard Worker #include <locale> 21*ec779b8eSAndroid Build Coastguard Worker #include <string> 22*ec779b8eSAndroid Build Coastguard Worker #include <vector> 23*ec779b8eSAndroid Build Coastguard Worker 24*ec779b8eSAndroid Build Coastguard Worker #include "MtpDataPacket.h" 25*ec779b8eSAndroid Build Coastguard Worker #include "MtpStringBuffer.h" 26*ec779b8eSAndroid Build Coastguard Worker 27*ec779b8eSAndroid Build Coastguard Worker namespace { 28*ec779b8eSAndroid Build Coastguard Worker 29*ec779b8eSAndroid Build Coastguard Worker const char * utf16_cerror = "__CONVERSION_ERROR__"; 30*ec779b8eSAndroid Build Coastguard Worker const char16_t * utf8_cerror = u"__CONVERSION_ERROR__"; 31*ec779b8eSAndroid Build Coastguard Worker 32*ec779b8eSAndroid Build Coastguard Worker std::wstring_convert<std::codecvt_utf8_utf16<char16_t>,char16_t> gConvert(utf16_cerror, utf8_cerror); 33*ec779b8eSAndroid Build Coastguard Worker utf16ToUtf8(std::u16string input_str)34*ec779b8eSAndroid Build Coastguard Workerstatic std::string utf16ToUtf8(std::u16string input_str) { 35*ec779b8eSAndroid Build Coastguard Worker std::string conversion = gConvert.to_bytes(input_str); 36*ec779b8eSAndroid Build Coastguard Worker 37*ec779b8eSAndroid Build Coastguard Worker if (conversion == utf16_cerror) { 38*ec779b8eSAndroid Build Coastguard Worker ALOGE("Unable to convert UTF-16 string to UTF-8"); 39*ec779b8eSAndroid Build Coastguard Worker return ""; 40*ec779b8eSAndroid Build Coastguard Worker } else { 41*ec779b8eSAndroid Build Coastguard Worker return conversion; 42*ec779b8eSAndroid Build Coastguard Worker } 43*ec779b8eSAndroid Build Coastguard Worker } 44*ec779b8eSAndroid Build Coastguard Worker utf8ToUtf16(std::string input_str)45*ec779b8eSAndroid Build Coastguard Workerstatic std::u16string utf8ToUtf16(std::string input_str) { 46*ec779b8eSAndroid Build Coastguard Worker std::u16string conversion = gConvert.from_bytes(input_str); 47*ec779b8eSAndroid Build Coastguard Worker 48*ec779b8eSAndroid Build Coastguard Worker if (conversion == utf8_cerror) { 49*ec779b8eSAndroid Build Coastguard Worker ALOGE("Unable to convert UTF-8 string to UTF-16"); 50*ec779b8eSAndroid Build Coastguard Worker return u""; 51*ec779b8eSAndroid Build Coastguard Worker } else { 52*ec779b8eSAndroid Build Coastguard Worker return conversion; 53*ec779b8eSAndroid Build Coastguard Worker } 54*ec779b8eSAndroid Build Coastguard Worker } 55*ec779b8eSAndroid Build Coastguard Worker 56*ec779b8eSAndroid Build Coastguard Worker } // namespace 57*ec779b8eSAndroid Build Coastguard Worker 58*ec779b8eSAndroid Build Coastguard Worker namespace android { 59*ec779b8eSAndroid Build Coastguard Worker MtpStringBuffer(const char * src)60*ec779b8eSAndroid Build Coastguard WorkerMtpStringBuffer::MtpStringBuffer(const char* src) 61*ec779b8eSAndroid Build Coastguard Worker { 62*ec779b8eSAndroid Build Coastguard Worker set(src); 63*ec779b8eSAndroid Build Coastguard Worker } 64*ec779b8eSAndroid Build Coastguard Worker MtpStringBuffer(const uint16_t * src)65*ec779b8eSAndroid Build Coastguard WorkerMtpStringBuffer::MtpStringBuffer(const uint16_t* src) 66*ec779b8eSAndroid Build Coastguard Worker { 67*ec779b8eSAndroid Build Coastguard Worker set(src); 68*ec779b8eSAndroid Build Coastguard Worker } 69*ec779b8eSAndroid Build Coastguard Worker MtpStringBuffer(const MtpStringBuffer & src)70*ec779b8eSAndroid Build Coastguard WorkerMtpStringBuffer::MtpStringBuffer(const MtpStringBuffer& src) 71*ec779b8eSAndroid Build Coastguard Worker { 72*ec779b8eSAndroid Build Coastguard Worker mString = src.mString; 73*ec779b8eSAndroid Build Coastguard Worker } 74*ec779b8eSAndroid Build Coastguard Worker set(const char * src)75*ec779b8eSAndroid Build Coastguard Workervoid MtpStringBuffer::set(const char* src) { 76*ec779b8eSAndroid Build Coastguard Worker mString = std::string(src); 77*ec779b8eSAndroid Build Coastguard Worker } 78*ec779b8eSAndroid Build Coastguard Worker set(const uint16_t * src)79*ec779b8eSAndroid Build Coastguard Workervoid MtpStringBuffer::set(const uint16_t* src) { 80*ec779b8eSAndroid Build Coastguard Worker mString = utf16ToUtf8(std::u16string((const char16_t*)src)); 81*ec779b8eSAndroid Build Coastguard Worker } 82*ec779b8eSAndroid Build Coastguard Worker readFromPacket(MtpDataPacket * packet)83*ec779b8eSAndroid Build Coastguard Workerbool MtpStringBuffer::readFromPacket(MtpDataPacket* packet) { 84*ec779b8eSAndroid Build Coastguard Worker uint8_t count; 85*ec779b8eSAndroid Build Coastguard Worker if (!packet->getUInt8(count)) 86*ec779b8eSAndroid Build Coastguard Worker return false; 87*ec779b8eSAndroid Build Coastguard Worker if (count == 0) 88*ec779b8eSAndroid Build Coastguard Worker return true; 89*ec779b8eSAndroid Build Coastguard Worker 90*ec779b8eSAndroid Build Coastguard Worker std::vector<char16_t> buffer(count); 91*ec779b8eSAndroid Build Coastguard Worker for (int i = 0; i < count; i++) { 92*ec779b8eSAndroid Build Coastguard Worker uint16_t ch; 93*ec779b8eSAndroid Build Coastguard Worker if (!packet->getUInt16(ch)) 94*ec779b8eSAndroid Build Coastguard Worker return false; 95*ec779b8eSAndroid Build Coastguard Worker buffer[i] = ch; 96*ec779b8eSAndroid Build Coastguard Worker } 97*ec779b8eSAndroid Build Coastguard Worker if (buffer[count-1] != '\0') { 98*ec779b8eSAndroid Build Coastguard Worker ALOGE("Mtp string not null terminated\n"); 99*ec779b8eSAndroid Build Coastguard Worker return false; 100*ec779b8eSAndroid Build Coastguard Worker } 101*ec779b8eSAndroid Build Coastguard Worker mString = utf16ToUtf8(std::u16string(buffer.data())); 102*ec779b8eSAndroid Build Coastguard Worker return true; 103*ec779b8eSAndroid Build Coastguard Worker } 104*ec779b8eSAndroid Build Coastguard Worker writeToPacket(MtpDataPacket * packet) const105*ec779b8eSAndroid Build Coastguard Workervoid MtpStringBuffer::writeToPacket(MtpDataPacket* packet) const { 106*ec779b8eSAndroid Build Coastguard Worker std::u16string src16 = utf8ToUtf16(mString); 107*ec779b8eSAndroid Build Coastguard Worker int count = src16.length(); 108*ec779b8eSAndroid Build Coastguard Worker 109*ec779b8eSAndroid Build Coastguard Worker if (count == 0) { 110*ec779b8eSAndroid Build Coastguard Worker packet->putUInt8(0); 111*ec779b8eSAndroid Build Coastguard Worker return; 112*ec779b8eSAndroid Build Coastguard Worker } 113*ec779b8eSAndroid Build Coastguard Worker packet->putUInt8(std::min(count + 1, MTP_STRING_MAX_CHARACTER_NUMBER)); 114*ec779b8eSAndroid Build Coastguard Worker 115*ec779b8eSAndroid Build Coastguard Worker int i = 0; 116*ec779b8eSAndroid Build Coastguard Worker for (char16_t &c : src16) { 117*ec779b8eSAndroid Build Coastguard Worker if (i == MTP_STRING_MAX_CHARACTER_NUMBER - 1) { 118*ec779b8eSAndroid Build Coastguard Worker // Leave a slot for null termination. 119*ec779b8eSAndroid Build Coastguard Worker ALOGI("Mtp truncating long string\n"); 120*ec779b8eSAndroid Build Coastguard Worker break; 121*ec779b8eSAndroid Build Coastguard Worker } 122*ec779b8eSAndroid Build Coastguard Worker packet->putUInt16(c); 123*ec779b8eSAndroid Build Coastguard Worker i++; 124*ec779b8eSAndroid Build Coastguard Worker } 125*ec779b8eSAndroid Build Coastguard Worker // only terminate with zero if string is not empty 126*ec779b8eSAndroid Build Coastguard Worker packet->putUInt16(0); 127*ec779b8eSAndroid Build Coastguard Worker } 128*ec779b8eSAndroid Build Coastguard Worker 129*ec779b8eSAndroid Build Coastguard Worker } // namespace android 130