summaryrefslogblamecommitdiffstats
path: root/mtp/ffs/MtpDataPacket.cpp
blob: 08f57c5e4040bb5723932e9eeea23138b583f745 (plain) (tree)
1
2
3
4
5
6
7
8






                                                                   
                                                          





                                                                           

   






                               

                      
                            
                            
                       

                     



















                                                                                                  

                              
                                                                      



















                                                                      









                                                  
                             


                                                                                 

 


                                                  
                             
                                                                                  
                                                                                                  

                                 

 


                                                  
                             
                                                                                  


                                                                                                  

                                 

 

                                                                                                        

 
                                                      
 
                                           


                                     


                              
                                        







                                              



                                       


                              
                                          







                                              



                                       


                              
                                          







                                              



                                         


                              
                                            







                                              



                                       


                              
                                          







                                              



                                         


                              
                                            







                                              



                                       


                              
                                          







                                              



                                         


                              
                                            







                                              





















































































































































































                                                                   
                                                                    












                                                          


                                                    






                                            
                                         

                                                                                 
                                                 


                                   

                                                                          



                                                                                 
                                            









                                                      
                                       



                                                                                         
                                                                     








































                                                                                    
                                                               











                                                                





                                                                                           


                                                                                 












                                                                                                         
         

                                                                   

 






































































                                                                                                                


                  

                                                    




                                                                                    
                                            


                                      
                       

                    
/*
 * 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"
#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<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)) {
		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<ssize_t>(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;
}