/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "MtpDataPacket" #include "MtpDataPacket.h" #include #include #include #include #include #include #include "MtpStringBuffer.h" #include "IMtpHandle.h" #include "MtpDebug.h" namespace { // Reads the exact |count| bytes from |fd| to |buf|. // Returns |count| if it succeed to read the bytes. Otherwise returns -1. If it reaches EOF, the // function regards it as an error. ssize_t readExactBytes(int fd, void* buf, size_t count) { if (count > SSIZE_MAX) { return -1; } size_t read_count = 0; while (read_count < count) { int result = read(fd, static_cast(buf) + read_count, count - read_count); // Assume that EOF is error. if (result <= 0) { return -1; } read_count += result; } return read_count == count ? count : -1; } } // namespace MtpDataPacket::MtpDataPacket() : MtpPacket(MTP_BUFFER_SIZE), // MAX_USBFS_BUFFER_SIZE mOffset(MTP_CONTAINER_HEADER_SIZE) { } MtpDataPacket::~MtpDataPacket() { } void MtpDataPacket::reset() { MtpPacket::reset(); mOffset = MTP_CONTAINER_HEADER_SIZE; } void MtpDataPacket::setOperationCode(MtpOperationCode code) { MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code); } void MtpDataPacket::setTransactionID(MtpTransactionID id) { MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id); } bool MtpDataPacket::getUInt8(uint8_t& value) { if (mPacketSize - mOffset < sizeof(value)) return false; value = mBuffer[mOffset++]; return true; } bool MtpDataPacket::getUInt16(uint16_t& value) { if (mPacketSize - mOffset < sizeof(value)) return false; int offset = mOffset; value = (uint16_t)mBuffer[offset] | ((uint16_t)mBuffer[offset + 1] << 8); mOffset += sizeof(value); return true; } bool MtpDataPacket::getUInt32(uint32_t& value) { if (mPacketSize - mOffset < sizeof(value)) return false; int offset = mOffset; value = (uint32_t)mBuffer[offset] | ((uint32_t)mBuffer[offset + 1] << 8) | ((uint32_t)mBuffer[offset + 2] << 16) | ((uint32_t)mBuffer[offset + 3] << 24); mOffset += sizeof(value); return true; } bool MtpDataPacket::getUInt64(uint64_t& value) { if (mPacketSize - mOffset < sizeof(value)) return false; int offset = mOffset; value = (uint64_t)mBuffer[offset] | ((uint64_t)mBuffer[offset + 1] << 8) | ((uint64_t)mBuffer[offset + 2] << 16) | ((uint64_t)mBuffer[offset + 3] << 24) | ((uint64_t)mBuffer[offset + 4] << 32) | ((uint64_t)mBuffer[offset + 5] << 40) | ((uint64_t)mBuffer[offset + 6] << 48) | ((uint64_t)mBuffer[offset + 7] << 56); mOffset += sizeof(value); return true; } bool MtpDataPacket::getUInt128(uint128_t& value) { return getUInt32(value[0]) && getUInt32(value[1]) && getUInt32(value[2]) && getUInt32(value[3]); } bool MtpDataPacket::getString(MtpStringBuffer& string) { return string.readFromPacket(this); } Int8List* MtpDataPacket::getAInt8() { uint32_t count; if (!getUInt32(count)) return NULL; Int8List* result = new Int8List; for (uint32_t i = 0; i < count; i++) { int8_t value; if (!getInt8(value)) { delete result; return NULL; } result->push_back(value); } return result; } UInt8List* MtpDataPacket::getAUInt8() { uint32_t count; if (!getUInt32(count)) return NULL; UInt8List* result = new UInt8List; for (uint32_t i = 0; i < count; i++) { uint8_t value; if (!getUInt8(value)) { delete result; return NULL; } result->push_back(value); } return result; } Int16List* MtpDataPacket::getAInt16() { uint32_t count; if (!getUInt32(count)) return NULL; Int16List* result = new Int16List; for (uint32_t i = 0; i < count; i++) { int16_t value; if (!getInt16(value)) { delete result; return NULL; } result->push_back(value); } return result; } UInt16List* MtpDataPacket::getAUInt16() { uint32_t count; if (!getUInt32(count)) return NULL; UInt16List* result = new UInt16List; for (uint32_t i = 0; i < count; i++) { uint16_t value; if (!getUInt16(value)) { delete result; return NULL; } result->push_back(value); } return result; } Int32List* MtpDataPacket::getAInt32() { uint32_t count; if (!getUInt32(count)) return NULL; Int32List* result = new Int32List; for (uint32_t i = 0; i < count; i++) { int32_t value; if (!getInt32(value)) { delete result; return NULL; } result->push_back(value); } return result; } UInt32List* MtpDataPacket::getAUInt32() { uint32_t count; if (!getUInt32(count)) return NULL; UInt32List* result = new UInt32List; for (uint32_t i = 0; i < count; i++) { uint32_t value; if (!getUInt32(value)) { delete result; return NULL; } result->push_back(value); } return result; } Int64List* MtpDataPacket::getAInt64() { uint32_t count; if (!getUInt32(count)) return NULL; Int64List* result = new Int64List; for (uint32_t i = 0; i < count; i++) { int64_t value; if (!getInt64(value)) { delete result; return NULL; } result->push_back(value); } return result; } UInt64List* MtpDataPacket::getAUInt64() { uint32_t count; if (!getUInt32(count)) return NULL; UInt64List* result = new UInt64List; for (uint32_t i = 0; i < count; i++) { uint64_t value; if (!getUInt64(value)) { delete result; return NULL; } result->push_back(value); } return result; } void MtpDataPacket::putInt8(int8_t value) { allocate(mOffset + 1); mBuffer[mOffset++] = (uint8_t)value; if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putUInt8(uint8_t value) { allocate(mOffset + 1); mBuffer[mOffset++] = (uint8_t)value; if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putInt16(int16_t value) { allocate(mOffset + 2); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putUInt16(uint16_t value) { allocate(mOffset + 2); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putInt32(int32_t value) { allocate(mOffset + 4); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putUInt32(uint32_t value) { allocate(mOffset + 4); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putInt64(int64_t value) { allocate(mOffset + 8); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putUInt64(uint64_t value) { allocate(mOffset + 8); mBuffer[mOffset++] = (uint8_t)(value & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF); mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF); if (mPacketSize < mOffset) mPacketSize = mOffset; } void MtpDataPacket::putInt128(const int128_t& value) { putInt32(value[0]); putInt32(value[1]); putInt32(value[2]); putInt32(value[3]); } void MtpDataPacket::putUInt128(const uint128_t& value) { putUInt32(value[0]); putUInt32(value[1]); putUInt32(value[2]); putUInt32(value[3]); } void MtpDataPacket::putInt128(int64_t value) { putInt64(value); putInt64(value < 0 ? -1 : 0); } void MtpDataPacket::putUInt128(uint64_t value) { putUInt64(value); putUInt64(0); } void MtpDataPacket::putAInt8(const int8_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putInt8(*values++); } void MtpDataPacket::putAUInt8(const uint8_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putUInt8(*values++); } void MtpDataPacket::putAInt16(const int16_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putInt16(*values++); } void MtpDataPacket::putAUInt16(const uint16_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putUInt16(*values++); } void MtpDataPacket::putAUInt16(const UInt16List* values) { size_t count = (values ? values->size() : 0); putUInt32(count); for (size_t i = 0; i < count; i++) putUInt16((*values)[i]); } void MtpDataPacket::putAInt32(const int32_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putInt32(*values++); } void MtpDataPacket::putAUInt32(const uint32_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putUInt32(*values++); } void MtpDataPacket::putAUInt32(const UInt32List* list) { if (!list) { putEmptyArray(); } else { size_t size = list->size(); putUInt32(size); for (size_t i = 0; i < size; i++) putUInt32((*list)[i]); } } void MtpDataPacket::putAInt64(const int64_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putInt64(*values++); } void MtpDataPacket::putAUInt64(const uint64_t* values, int count) { putUInt32(count); for (int i = 0; i < count; i++) putUInt64(*values++); } void MtpDataPacket::putString(const MtpStringBuffer& string) { string.writeToPacket(this); } void MtpDataPacket::putString(const char* s) { MtpStringBuffer string(s); string.writeToPacket(this); } void MtpDataPacket::putString(const uint16_t* string) { int count = 0; for (int i = 0; i <= MTP_STRING_MAX_CHARACTER_NUMBER; i++) { if (string[i]) count++; else break; } putUInt8(count > 0 ? count + 1 : 0); for (int i = 0; i < count; i++) putUInt16(string[i]); // only terminate with zero if string is not empty if (count > 0) putUInt16(0); } #ifdef MTP_DEVICE int MtpDataPacket::read(IMtpHandle *h) { int ret = h->read(mBuffer, MTP_BUFFER_SIZE); if (ret < MTP_CONTAINER_HEADER_SIZE) return -1; mPacketSize = ret; mOffset = MTP_CONTAINER_HEADER_SIZE; return ret; } int MtpDataPacket::write(IMtpHandle *h) { MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize); MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); int ret = h->write(mBuffer, mPacketSize); return (ret < 0 ? ret : 0); } int MtpDataPacket::writeData(IMtpHandle *h, void* data, uint32_t length) { allocate(length + MTP_CONTAINER_HEADER_SIZE); memcpy(mBuffer + MTP_CONTAINER_HEADER_SIZE, data, length); length += MTP_CONTAINER_HEADER_SIZE; MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length); MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); int ret = h->write(mBuffer, length); return (ret < 0 ? ret : 0); } #endif // MTP_DEVICE #ifdef MTP_HOST int MtpDataPacket::read(struct usb_request *request) { // first read the header request->buffer = mBuffer; request->buffer_length = mBufferSize; int length = transfer(request); if (length >= MTP_CONTAINER_HEADER_SIZE) { // look at the length field to see if the data spans multiple packets uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET); allocate(totalLength); while (totalLength > static_cast(length)) { request->buffer = mBuffer + length; request->buffer_length = totalLength - length; int ret = transfer(request); if (ret >= 0) length += ret; else { length = ret; break; } } } if (length >= 0) mPacketSize = length; return length; } int MtpDataPacket::readData(struct usb_request *request, void* buffer, int length) { int read = 0; while (read < length) { request->buffer = (char *)buffer + read; request->buffer_length = length - read; int ret = transfer(request); if (ret < 0) { return ret; } read += ret; } return read; } // Queue a read request. Call readDataWait to wait for result int MtpDataPacket::readDataAsync(struct usb_request *req) { if (usb_request_queue(req)) { MTPE("usb_endpoint_queue failed, errno: %d", errno); return -1; } return 0; } // Wait for result of readDataAsync int MtpDataPacket::readDataWait(struct usb_device *device) { struct usb_request *req = usb_request_wait(device, -1); return (req ? req->actual_length : -1); } int MtpDataPacket::readDataHeader(struct usb_request *request) { request->buffer = mBuffer; request->buffer_length = request->max_packet_size; int length = transfer(request); if (length >= 0) mPacketSize = length; return length; } int MtpDataPacket::write(struct usb_request *request, UrbPacketDivisionMode divisionMode) { if (mPacketSize < MTP_CONTAINER_HEADER_SIZE || mPacketSize > MTP_BUFFER_SIZE) { MTPE("Illegal packet size."); return -1; } MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize); MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); size_t processedBytes = 0; while (processedBytes < mPacketSize) { const size_t write_size = processedBytes == 0 && divisionMode == FIRST_PACKET_ONLY_HEADER ? MTP_CONTAINER_HEADER_SIZE : mPacketSize - processedBytes; request->buffer = mBuffer + processedBytes; request->buffer_length = write_size; const int result = transfer(request); if (result < 0) { MTPE("Failed to write bytes to the device."); return -1; } processedBytes += result; } return processedBytes == mPacketSize ? processedBytes : -1; } int MtpDataPacket::write(struct usb_request *request, UrbPacketDivisionMode divisionMode, int fd, size_t payloadSize) { // Obtain the greatest multiple of minimum packet size that is not greater than // MTP_BUFFER_SIZE. if (request->max_packet_size <= 0) { MTPE("Cannot determine bulk transfer size due to illegal max packet size %d.", request->max_packet_size); return -1; } const size_t maxBulkTransferSize = MTP_BUFFER_SIZE - (MTP_BUFFER_SIZE % request->max_packet_size); const size_t containerLength = payloadSize + MTP_CONTAINER_HEADER_SIZE; size_t processedBytes = 0; bool readError = false; // Bind the packet with given request. request->buffer = mBuffer; allocate(maxBulkTransferSize); while (processedBytes < containerLength) { size_t bulkTransferSize = 0; // prepare header. const bool headerSent = processedBytes != 0; if (!headerSent) { MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, containerLength); MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); bulkTransferSize += MTP_CONTAINER_HEADER_SIZE; } // Prepare payload. if (headerSent || divisionMode == FIRST_PACKET_HAS_PAYLOAD) { const size_t processedPayloadBytes = headerSent ? processedBytes - MTP_CONTAINER_HEADER_SIZE : 0; const size_t maxRead = payloadSize - processedPayloadBytes; const size_t maxWrite = maxBulkTransferSize - bulkTransferSize; const size_t bulkTransferPayloadSize = std::min(maxRead, maxWrite); // prepare payload. if (!readError) { const ssize_t result = readExactBytes( fd, mBuffer + bulkTransferSize, bulkTransferPayloadSize); if (result < 0) { MTPE("Found an error while reading data from FD. Send 0 data instead."); readError = true; } } if (readError) { memset(mBuffer + bulkTransferSize, 0, bulkTransferPayloadSize); } bulkTransferSize += bulkTransferPayloadSize; } // Bulk transfer. mPacketSize = bulkTransferSize; request->buffer_length = bulkTransferSize; const int result = transfer(request); if (result != static_cast(bulkTransferSize)) { // Cannot recover writing error. MTPE("Found an error while write data to MtpDevice."); return -1; } // Update variables. processedBytes += bulkTransferSize; } return readError ? -1 : processedBytes; } #endif // MTP_HOST void* MtpDataPacket::getData(int* outLength) const { int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE; if (length > 0) { void* result = malloc(length); if (result) { memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length); *outLength = length; return result; } } *outLength = 0; return NULL; }