1 /* 2 * Copyright (C) 2010 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #define LOG_TAG "MtpDataPacket" 18 19 #include <stdio.h> 20 #include <sys/types.h> 21 #include <fcntl.h> 22 23 #include <usbhost/usbhost.h> 24 25 #include "MtpDataPacket.h" 26 #include "MtpStringBuffer.h" 27 28 namespace android { 29 30 MtpDataPacket::MtpDataPacket() 31 : MtpPacket(16384), // MAX_USBFS_BUFFER_SIZE 32 mOffset(MTP_CONTAINER_HEADER_SIZE) 33 { 34 } 35 36 MtpDataPacket::~MtpDataPacket() { 37 } 38 39 void MtpDataPacket::reset() { 40 MtpPacket::reset(); 41 mOffset = MTP_CONTAINER_HEADER_SIZE; 42 } 43 44 void MtpDataPacket::setOperationCode(MtpOperationCode code) { 45 MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code); 46 } 47 48 void MtpDataPacket::setTransactionID(MtpTransactionID id) { 49 MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id); 50 } 51 52 uint16_t MtpDataPacket::getUInt16() { 53 int offset = mOffset; 54 uint16_t result = (uint16_t)mBuffer[offset] | ((uint16_t)mBuffer[offset + 1] << 8); 55 mOffset += 2; 56 return result; 57 } 58 59 uint32_t MtpDataPacket::getUInt32() { 60 int offset = mOffset; 61 uint32_t result = (uint32_t)mBuffer[offset] | ((uint32_t)mBuffer[offset + 1] << 8) | 62 ((uint32_t)mBuffer[offset + 2] << 16) | ((uint32_t)mBuffer[offset + 3] << 24); 63 mOffset += 4; 64 return result; 65 } 66 67 uint64_t MtpDataPacket::getUInt64() { 68 int offset = mOffset; 69 uint64_t result = (uint64_t)mBuffer[offset] | ((uint64_t)mBuffer[offset + 1] << 8) | 70 ((uint64_t)mBuffer[offset + 2] << 16) | ((uint64_t)mBuffer[offset + 3] << 24) | 71 ((uint64_t)mBuffer[offset + 4] << 32) | ((uint64_t)mBuffer[offset + 5] << 40) | 72 ((uint64_t)mBuffer[offset + 6] << 48) | ((uint64_t)mBuffer[offset + 7] << 56); 73 mOffset += 8; 74 return result; 75 } 76 77 void MtpDataPacket::getUInt128(uint128_t& value) { 78 value[0] = getUInt32(); 79 value[1] = getUInt32(); 80 value[2] = getUInt32(); 81 value[3] = getUInt32(); 82 } 83 84 void MtpDataPacket::getString(MtpStringBuffer& string) 85 { 86 string.readFromPacket(this); 87 } 88 89 Int8List* MtpDataPacket::getAInt8() { 90 Int8List* result = new Int8List; 91 int count = getUInt32(); 92 for (int i = 0; i < count; i++) 93 result->push(getInt8()); 94 return result; 95 } 96 97 UInt8List* MtpDataPacket::getAUInt8() { 98 UInt8List* result = new UInt8List; 99 int count = getUInt32(); 100 for (int i = 0; i < count; i++) 101 result->push(getUInt8()); 102 return result; 103 } 104 105 Int16List* MtpDataPacket::getAInt16() { 106 Int16List* result = new Int16List; 107 int count = getUInt32(); 108 for (int i = 0; i < count; i++) 109 result->push(getInt16()); 110 return result; 111 } 112 113 UInt16List* MtpDataPacket::getAUInt16() { 114 UInt16List* result = new UInt16List; 115 int count = getUInt32(); 116 for (int i = 0; i < count; i++) 117 result->push(getUInt16()); 118 return result; 119 } 120 121 Int32List* MtpDataPacket::getAInt32() { 122 Int32List* result = new Int32List; 123 int count = getUInt32(); 124 for (int i = 0; i < count; i++) 125 result->push(getInt32()); 126 return result; 127 } 128 129 UInt32List* MtpDataPacket::getAUInt32() { 130 UInt32List* result = new UInt32List; 131 int count = getUInt32(); 132 for (int i = 0; i < count; i++) 133 result->push(getUInt32()); 134 return result; 135 } 136 137 Int64List* MtpDataPacket::getAInt64() { 138 Int64List* result = new Int64List; 139 int count = getUInt32(); 140 for (int i = 0; i < count; i++) 141 result->push(getInt64()); 142 return result; 143 } 144 145 UInt64List* MtpDataPacket::getAUInt64() { 146 UInt64List* result = new UInt64List; 147 int count = getUInt32(); 148 for (int i = 0; i < count; i++) 149 result->push(getUInt64()); 150 return result; 151 } 152 153 void MtpDataPacket::putInt8(int8_t value) { 154 allocate(mOffset + 1); 155 mBuffer[mOffset++] = (uint8_t)value; 156 if (mPacketSize < mOffset) 157 mPacketSize = mOffset; 158 } 159 160 void MtpDataPacket::putUInt8(uint8_t value) { 161 allocate(mOffset + 1); 162 mBuffer[mOffset++] = (uint8_t)value; 163 if (mPacketSize < mOffset) 164 mPacketSize = mOffset; 165 } 166 167 void MtpDataPacket::putInt16(int16_t value) { 168 allocate(mOffset + 2); 169 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 170 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 171 if (mPacketSize < mOffset) 172 mPacketSize = mOffset; 173 } 174 175 void MtpDataPacket::putUInt16(uint16_t value) { 176 allocate(mOffset + 2); 177 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 178 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 179 if (mPacketSize < mOffset) 180 mPacketSize = mOffset; 181 } 182 183 void MtpDataPacket::putInt32(int32_t value) { 184 allocate(mOffset + 4); 185 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 186 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 187 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); 188 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); 189 if (mPacketSize < mOffset) 190 mPacketSize = mOffset; 191 } 192 193 void MtpDataPacket::putUInt32(uint32_t value) { 194 allocate(mOffset + 4); 195 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 196 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 197 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); 198 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); 199 if (mPacketSize < mOffset) 200 mPacketSize = mOffset; 201 } 202 203 void MtpDataPacket::putInt64(int64_t value) { 204 allocate(mOffset + 8); 205 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 206 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 207 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); 208 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); 209 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF); 210 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF); 211 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF); 212 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF); 213 if (mPacketSize < mOffset) 214 mPacketSize = mOffset; 215 } 216 217 void MtpDataPacket::putUInt64(uint64_t value) { 218 allocate(mOffset + 8); 219 mBuffer[mOffset++] = (uint8_t)(value & 0xFF); 220 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF); 221 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF); 222 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF); 223 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF); 224 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF); 225 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF); 226 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF); 227 if (mPacketSize < mOffset) 228 mPacketSize = mOffset; 229 } 230 231 void MtpDataPacket::putInt128(const int128_t& value) { 232 putInt32(value[0]); 233 putInt32(value[1]); 234 putInt32(value[2]); 235 putInt32(value[3]); 236 } 237 238 void MtpDataPacket::putUInt128(const uint128_t& value) { 239 putUInt32(value[0]); 240 putUInt32(value[1]); 241 putUInt32(value[2]); 242 putUInt32(value[3]); 243 } 244 245 void MtpDataPacket::putInt128(int64_t value) { 246 putInt64(value); 247 putInt64(value < 0 ? -1 : 0); 248 } 249 250 void MtpDataPacket::putUInt128(uint64_t value) { 251 putUInt64(value); 252 putUInt64(0); 253 } 254 255 void MtpDataPacket::putAInt8(const int8_t* values, int count) { 256 putUInt32(count); 257 for (int i = 0; i < count; i++) 258 putInt8(*values++); 259 } 260 261 void MtpDataPacket::putAUInt8(const uint8_t* values, int count) { 262 putUInt32(count); 263 for (int i = 0; i < count; i++) 264 putUInt8(*values++); 265 } 266 267 void MtpDataPacket::putAInt16(const int16_t* values, int count) { 268 putUInt32(count); 269 for (int i = 0; i < count; i++) 270 putInt16(*values++); 271 } 272 273 void MtpDataPacket::putAUInt16(const uint16_t* values, int count) { 274 putUInt32(count); 275 for (int i = 0; i < count; i++) 276 putUInt16(*values++); 277 } 278 279 void MtpDataPacket::putAUInt16(const UInt16List* values) { 280 size_t count = (values ? values->size() : 0); 281 putUInt32(count); 282 for (size_t i = 0; i < count; i++) 283 putUInt16((*values)[i]); 284 } 285 286 void MtpDataPacket::putAInt32(const int32_t* values, int count) { 287 putUInt32(count); 288 for (int i = 0; i < count; i++) 289 putInt32(*values++); 290 } 291 292 void MtpDataPacket::putAUInt32(const uint32_t* values, int count) { 293 putUInt32(count); 294 for (int i = 0; i < count; i++) 295 putUInt32(*values++); 296 } 297 298 void MtpDataPacket::putAUInt32(const UInt32List* list) { 299 if (!list) { 300 putEmptyArray(); 301 } else { 302 size_t size = list->size(); 303 putUInt32(size); 304 for (size_t i = 0; i < size; i++) 305 putUInt32((*list)[i]); 306 } 307 } 308 309 void MtpDataPacket::putAInt64(const int64_t* values, int count) { 310 putUInt32(count); 311 for (int i = 0; i < count; i++) 312 putInt64(*values++); 313 } 314 315 void MtpDataPacket::putAUInt64(const uint64_t* values, int count) { 316 putUInt32(count); 317 for (int i = 0; i < count; i++) 318 putUInt64(*values++); 319 } 320 321 void MtpDataPacket::putString(const MtpStringBuffer& string) { 322 string.writeToPacket(this); 323 } 324 325 void MtpDataPacket::putString(const char* s) { 326 MtpStringBuffer string(s); 327 string.writeToPacket(this); 328 } 329 330 void MtpDataPacket::putString(const uint16_t* string) { 331 int count = 0; 332 for (int i = 0; i < 256; i++) { 333 if (string[i]) 334 count++; 335 else 336 break; 337 } 338 putUInt8(count > 0 ? count + 1 : 0); 339 for (int i = 0; i < count; i++) 340 putUInt16(string[i]); 341 // only terminate with zero if string is not empty 342 if (count > 0) 343 putUInt16(0); 344 } 345 346 #ifdef MTP_DEVICE 347 int MtpDataPacket::read(int fd) { 348 int ret = ::read(fd, mBuffer, mBufferSize); 349 if (ret < MTP_CONTAINER_HEADER_SIZE) 350 return -1; 351 mPacketSize = ret; 352 mOffset = MTP_CONTAINER_HEADER_SIZE; 353 return ret; 354 } 355 356 int MtpDataPacket::write(int fd) { 357 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize); 358 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); 359 int ret = ::write(fd, mBuffer, mPacketSize); 360 return (ret < 0 ? ret : 0); 361 } 362 363 int MtpDataPacket::writeData(int fd, void* data, uint32_t length) { 364 allocate(length); 365 memcpy(mBuffer + MTP_CONTAINER_HEADER_SIZE, data, length); 366 length += MTP_CONTAINER_HEADER_SIZE; 367 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length); 368 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); 369 int ret = ::write(fd, mBuffer, length); 370 return (ret < 0 ? ret : 0); 371 } 372 373 #endif // MTP_DEVICE 374 375 #ifdef MTP_HOST 376 int MtpDataPacket::read(struct usb_request *request) { 377 // first read the header 378 request->buffer = mBuffer; 379 request->buffer_length = mBufferSize; 380 int length = transfer(request); 381 if (length >= MTP_CONTAINER_HEADER_SIZE) { 382 // look at the length field to see if the data spans multiple packets 383 uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET); 384 allocate(totalLength); 385 while (totalLength > length) { 386 request->buffer = mBuffer + length; 387 request->buffer_length = totalLength - length; 388 int ret = transfer(request); 389 if (ret >= 0) 390 length += ret; 391 else { 392 length = ret; 393 break; 394 } 395 } 396 } 397 if (length >= 0) 398 mPacketSize = length; 399 return length; 400 } 401 402 int MtpDataPacket::readData(struct usb_request *request, void* buffer, int length) { 403 int read = 0; 404 while (read < length) { 405 request->buffer = (char *)buffer + read; 406 request->buffer_length = length - read; 407 int ret = transfer(request); 408 if (ret < 0) { 409 return ret; 410 } 411 read += ret; 412 } 413 return read; 414 } 415 416 // Queue a read request. Call readDataWait to wait for result 417 int MtpDataPacket::readDataAsync(struct usb_request *req) { 418 if (usb_request_queue(req)) { 419 LOGE("usb_endpoint_queue failed, errno: %d", errno); 420 return -1; 421 } 422 return 0; 423 } 424 425 // Wait for result of readDataAsync 426 int MtpDataPacket::readDataWait(struct usb_device *device) { 427 struct usb_request *req = usb_request_wait(device); 428 return (req ? req->actual_length : -1); 429 } 430 431 int MtpDataPacket::readDataHeader(struct usb_request *request) { 432 request->buffer = mBuffer; 433 request->buffer_length = request->max_packet_size; 434 int length = transfer(request); 435 if (length >= 0) 436 mPacketSize = length; 437 return length; 438 } 439 440 int MtpDataPacket::writeDataHeader(struct usb_request *request, uint32_t length) { 441 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length); 442 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); 443 request->buffer = mBuffer; 444 request->buffer_length = MTP_CONTAINER_HEADER_SIZE; 445 int ret = transfer(request); 446 return (ret < 0 ? ret : 0); 447 } 448 449 int MtpDataPacket::write(struct usb_request *request) { 450 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize); 451 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA); 452 453 // send header separately from data 454 request->buffer = mBuffer; 455 request->buffer_length = MTP_CONTAINER_HEADER_SIZE; 456 int ret = transfer(request); 457 if (ret == MTP_CONTAINER_HEADER_SIZE) { 458 request->buffer = mBuffer + MTP_CONTAINER_HEADER_SIZE; 459 request->buffer_length = mPacketSize - MTP_CONTAINER_HEADER_SIZE; 460 ret = transfer(request); 461 } 462 return (ret < 0 ? ret : 0); 463 } 464 465 int MtpDataPacket::write(struct usb_request *request, void* buffer, uint32_t length) { 466 request->buffer = buffer; 467 request->buffer_length = length; 468 int ret = transfer(request); 469 return (ret < 0 ? ret : 0); 470 } 471 472 #endif // MTP_HOST 473 474 void* MtpDataPacket::getData(int& outLength) const { 475 int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE; 476 if (length > 0) { 477 void* result = malloc(length); 478 if (result) { 479 memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length); 480 outLength = length; 481 return result; 482 } 483 } 484 outLength = 0; 485 return NULL; 486 } 487 488 } // namespace android 489