123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653 |
- /*
- * 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 <algorithm>
- #include <errno.h>
- #include <fcntl.h>
- #include <stdio.h>
- #include <sys/types.h>
- #include <usbhost/usbhost.h>
- #include "MtpStringBuffer.h"
- #include "IMtpHandle.h"
- namespace android {
- 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<int8_t*>(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<uint32_t>(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)) {
- ALOGE("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) {
- ALOGE("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) {
- ALOGE("Failed to write bytes to the device.");
- return -1;
- }
- processedBytes += result;
- }
- return processedBytes == mPacketSize ? processedBytes : -1;
- }
- int64_t 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) {
- ALOGE("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) {
- ALOGE("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<ssize_t>(bulkTransferSize)) {
- // Cannot recover writing error.
- ALOGE("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;
- }
- } // namespace android
|