blob: 4e20b1a5e1de8f29952b2b53d1d2c970b4b95020 [file] [log] [blame]
Mike Lockwood16864ba2010-05-11 17:16:59 -04001/*
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
Mike Lockwoodb14e5882010-06-29 18:11:52 -040017#define LOG_TAG "MtpDataPacket"
18
Mike Lockwood16864ba2010-05-11 17:16:59 -040019#include <stdio.h>
20#include <sys/types.h>
21#include <fcntl.h>
22
Mike Lockwood0cf89f22010-07-26 20:40:45 -040023#include <usbhost/usbhost.h>
24
Mike Lockwood16864ba2010-05-11 17:16:59 -040025#include "MtpDataPacket.h"
26#include "MtpStringBuffer.h"
27
Mike Lockwood44c19082011-12-01 18:36:06 -050028#define MTP_BUFFER_SIZE 16384
29
Mike Lockwood7850ef92010-05-14 10:10:36 -040030namespace android {
31
Mike Lockwood16864ba2010-05-11 17:16:59 -040032MtpDataPacket::MtpDataPacket()
Mike Lockwood44c19082011-12-01 18:36:06 -050033 : MtpPacket(MTP_BUFFER_SIZE), // MAX_USBFS_BUFFER_SIZE
Mike Lockwood16864ba2010-05-11 17:16:59 -040034 mOffset(MTP_CONTAINER_HEADER_SIZE)
35{
36}
37
38MtpDataPacket::~MtpDataPacket() {
39}
40
41void MtpDataPacket::reset() {
42 MtpPacket::reset();
43 mOffset = MTP_CONTAINER_HEADER_SIZE;
44}
45
46void MtpDataPacket::setOperationCode(MtpOperationCode code) {
47 MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code);
48}
49
50void MtpDataPacket::setTransactionID(MtpTransactionID id) {
51 MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id);
52}
53
Mike Lockwoodab063842014-11-12 14:20:06 -080054bool MtpDataPacket::getUInt8(uint8_t& value) {
55 if (mPacketSize - mOffset < sizeof(value))
56 return false;
57 value = mBuffer[mOffset++];
58 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040059}
60
Mike Lockwoodab063842014-11-12 14:20:06 -080061bool MtpDataPacket::getUInt16(uint16_t& value) {
62 if (mPacketSize - mOffset < sizeof(value))
63 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -040064 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -080065 value = (uint16_t)mBuffer[offset] | ((uint16_t)mBuffer[offset + 1] << 8);
66 mOffset += sizeof(value);
67 return true;
68}
69
70bool MtpDataPacket::getUInt32(uint32_t& value) {
71 if (mPacketSize - mOffset < sizeof(value))
72 return false;
73 int offset = mOffset;
74 value = (uint32_t)mBuffer[offset] | ((uint32_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -040075 ((uint32_t)mBuffer[offset + 2] << 16) | ((uint32_t)mBuffer[offset + 3] << 24);
Mike Lockwoodab063842014-11-12 14:20:06 -080076 mOffset += sizeof(value);
77 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040078}
79
Mike Lockwoodab063842014-11-12 14:20:06 -080080bool MtpDataPacket::getUInt64(uint64_t& value) {
81 if (mPacketSize - mOffset < sizeof(value))
82 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -040083 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -080084 value = (uint64_t)mBuffer[offset] | ((uint64_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -040085 ((uint64_t)mBuffer[offset + 2] << 16) | ((uint64_t)mBuffer[offset + 3] << 24) |
86 ((uint64_t)mBuffer[offset + 4] << 32) | ((uint64_t)mBuffer[offset + 5] << 40) |
87 ((uint64_t)mBuffer[offset + 6] << 48) | ((uint64_t)mBuffer[offset + 7] << 56);
Mike Lockwoodab063842014-11-12 14:20:06 -080088 mOffset += sizeof(value);
89 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040090}
91
Mike Lockwoodab063842014-11-12 14:20:06 -080092bool MtpDataPacket::getUInt128(uint128_t& value) {
93 return getUInt32(value[0]) && getUInt32(value[1]) && getUInt32(value[2]) && getUInt32(value[3]);
Mike Lockwooda6c490b2010-06-05 22:45:01 -040094}
95
Mike Lockwoodab063842014-11-12 14:20:06 -080096bool MtpDataPacket::getString(MtpStringBuffer& string)
Mike Lockwood16864ba2010-05-11 17:16:59 -040097{
Mike Lockwoodab063842014-11-12 14:20:06 -080098 return string.readFromPacket(this);
Mike Lockwood16864ba2010-05-11 17:16:59 -040099}
100
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400101Int8List* MtpDataPacket::getAInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800102 uint32_t count;
103 if (!getUInt32(count))
104 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400105 Int8List* result = new Int8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800106 for (uint32_t i = 0; i < count; i++) {
107 int8_t value;
108 if (!getInt8(value)) {
109 delete result;
110 return NULL;
111 }
112 result->push(value);
113 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400114 return result;
115}
116
117UInt8List* MtpDataPacket::getAUInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800118 uint32_t count;
119 if (!getUInt32(count))
120 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400121 UInt8List* result = new UInt8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800122 for (uint32_t i = 0; i < count; i++) {
123 uint8_t value;
124 if (!getUInt8(value)) {
125 delete result;
126 return NULL;
127 }
128 result->push(value);
129 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400130 return result;
131}
132
133Int16List* MtpDataPacket::getAInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800134 uint32_t count;
135 if (!getUInt32(count))
136 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400137 Int16List* result = new Int16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800138 for (uint32_t i = 0; i < count; i++) {
139 int16_t value;
140 if (!getInt16(value)) {
141 delete result;
142 return NULL;
143 }
144 result->push(value);
145 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400146 return result;
147}
148
149UInt16List* MtpDataPacket::getAUInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800150 uint32_t count;
151 if (!getUInt32(count))
152 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400153 UInt16List* result = new UInt16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800154 for (uint32_t i = 0; i < count; i++) {
155 uint16_t value;
156 if (!getUInt16(value)) {
157 delete result;
158 return NULL;
159 }
160 result->push(value);
161 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400162 return result;
163}
164
165Int32List* MtpDataPacket::getAInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800166 uint32_t count;
167 if (!getUInt32(count))
168 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400169 Int32List* result = new Int32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800170 for (uint32_t i = 0; i < count; i++) {
171 int32_t value;
172 if (!getInt32(value)) {
173 delete result;
174 return NULL;
175 }
176 result->push(value);
177 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400178 return result;
179}
180
181UInt32List* MtpDataPacket::getAUInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800182 uint32_t count;
183 if (!getUInt32(count))
184 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400185 UInt32List* result = new UInt32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800186 for (uint32_t i = 0; i < count; i++) {
187 uint32_t value;
188 if (!getUInt32(value)) {
189 delete result;
190 return NULL;
191 }
192 result->push(value);
193 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400194 return result;
195}
196
197Int64List* MtpDataPacket::getAInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800198 uint32_t count;
199 if (!getUInt32(count))
200 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400201 Int64List* result = new Int64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800202 for (uint32_t i = 0; i < count; i++) {
203 int64_t value;
204 if (!getInt64(value)) {
205 delete result;
206 return NULL;
207 }
208 result->push(value);
209 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400210 return result;
211}
212
213UInt64List* MtpDataPacket::getAUInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800214 uint32_t count;
215 if (!getUInt32(count))
216 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400217 UInt64List* result = new UInt64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800218 for (uint32_t i = 0; i < count; i++) {
219 uint64_t value;
220 if (!getUInt64(value)) {
221 delete result;
222 return NULL;
223 }
224 result->push(value);
225 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400226 return result;
227}
228
Mike Lockwood16864ba2010-05-11 17:16:59 -0400229void MtpDataPacket::putInt8(int8_t value) {
230 allocate(mOffset + 1);
231 mBuffer[mOffset++] = (uint8_t)value;
232 if (mPacketSize < mOffset)
233 mPacketSize = mOffset;
234}
235
236void MtpDataPacket::putUInt8(uint8_t value) {
237 allocate(mOffset + 1);
238 mBuffer[mOffset++] = (uint8_t)value;
239 if (mPacketSize < mOffset)
240 mPacketSize = mOffset;
241}
242
243void MtpDataPacket::putInt16(int16_t value) {
244 allocate(mOffset + 2);
245 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
246 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
247 if (mPacketSize < mOffset)
248 mPacketSize = mOffset;
249}
250
251void MtpDataPacket::putUInt16(uint16_t value) {
252 allocate(mOffset + 2);
253 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
254 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
255 if (mPacketSize < mOffset)
256 mPacketSize = mOffset;
257}
258
259void MtpDataPacket::putInt32(int32_t value) {
260 allocate(mOffset + 4);
261 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
262 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
263 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
264 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
265 if (mPacketSize < mOffset)
266 mPacketSize = mOffset;
267}
268
269void MtpDataPacket::putUInt32(uint32_t value) {
270 allocate(mOffset + 4);
271 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
272 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
273 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
274 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
275 if (mPacketSize < mOffset)
276 mPacketSize = mOffset;
277}
278
279void MtpDataPacket::putInt64(int64_t value) {
280 allocate(mOffset + 8);
281 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
282 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
283 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
284 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
285 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
286 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
287 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
288 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
289 if (mPacketSize < mOffset)
290 mPacketSize = mOffset;
291}
292
293void MtpDataPacket::putUInt64(uint64_t value) {
294 allocate(mOffset + 8);
295 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
296 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
297 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
298 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
299 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
300 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
301 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
302 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
303 if (mPacketSize < mOffset)
304 mPacketSize = mOffset;
305}
306
Mike Lockwooda6c490b2010-06-05 22:45:01 -0400307void MtpDataPacket::putInt128(const int128_t& value) {
308 putInt32(value[0]);
309 putInt32(value[1]);
310 putInt32(value[2]);
311 putInt32(value[3]);
312}
313
314void MtpDataPacket::putUInt128(const uint128_t& value) {
315 putUInt32(value[0]);
316 putUInt32(value[1]);
317 putUInt32(value[2]);
318 putUInt32(value[3]);
319}
320
Mike Lockwood8277cec2010-08-10 15:20:35 -0400321void MtpDataPacket::putInt128(int64_t value) {
322 putInt64(value);
Mike Lockwood59599882010-08-25 19:10:24 -0400323 putInt64(value < 0 ? -1 : 0);
Mike Lockwood8277cec2010-08-10 15:20:35 -0400324}
325
326void MtpDataPacket::putUInt128(uint64_t value) {
327 putUInt64(value);
328 putUInt64(0);
329}
330
Mike Lockwood16864ba2010-05-11 17:16:59 -0400331void MtpDataPacket::putAInt8(const int8_t* values, int count) {
332 putUInt32(count);
333 for (int i = 0; i < count; i++)
334 putInt8(*values++);
335}
336
337void MtpDataPacket::putAUInt8(const uint8_t* values, int count) {
338 putUInt32(count);
339 for (int i = 0; i < count; i++)
340 putUInt8(*values++);
341}
342
343void MtpDataPacket::putAInt16(const int16_t* values, int count) {
344 putUInt32(count);
345 for (int i = 0; i < count; i++)
346 putInt16(*values++);
347}
348
349void MtpDataPacket::putAUInt16(const uint16_t* values, int count) {
350 putUInt32(count);
351 for (int i = 0; i < count; i++)
352 putUInt16(*values++);
353}
354
Mike Lockwood782aef12010-08-10 07:37:50 -0400355void MtpDataPacket::putAUInt16(const UInt16List* values) {
356 size_t count = (values ? values->size() : 0);
357 putUInt32(count);
358 for (size_t i = 0; i < count; i++)
359 putUInt16((*values)[i]);
360}
361
Mike Lockwood16864ba2010-05-11 17:16:59 -0400362void MtpDataPacket::putAInt32(const int32_t* values, int count) {
363 putUInt32(count);
364 for (int i = 0; i < count; i++)
365 putInt32(*values++);
366}
367
368void MtpDataPacket::putAUInt32(const uint32_t* values, int count) {
369 putUInt32(count);
370 for (int i = 0; i < count; i++)
371 putUInt32(*values++);
372}
373
374void MtpDataPacket::putAUInt32(const UInt32List* list) {
375 if (!list) {
376 putEmptyArray();
377 } else {
378 size_t size = list->size();
379 putUInt32(size);
380 for (size_t i = 0; i < size; i++)
381 putUInt32((*list)[i]);
382 }
383}
384
385void MtpDataPacket::putAInt64(const int64_t* values, int count) {
386 putUInt32(count);
387 for (int i = 0; i < count; i++)
388 putInt64(*values++);
389}
390
391void MtpDataPacket::putAUInt64(const uint64_t* values, int count) {
392 putUInt32(count);
393 for (int i = 0; i < count; i++)
394 putUInt64(*values++);
395}
396
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400397void MtpDataPacket::putString(const MtpStringBuffer& string) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400398 string.writeToPacket(this);
399}
400
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400401void MtpDataPacket::putString(const char* s) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400402 MtpStringBuffer string(s);
403 string.writeToPacket(this);
404}
405
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400406void MtpDataPacket::putString(const uint16_t* string) {
407 int count = 0;
Yin Liu014897f2012-12-04 09:19:53 +0100408 for (int i = 0; i <= MTP_STRING_MAX_CHARACTER_NUMBER; i++) {
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400409 if (string[i])
410 count++;
411 else
412 break;
413 }
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400414 putUInt8(count > 0 ? count + 1 : 0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400415 for (int i = 0; i < count; i++)
416 putUInt16(string[i]);
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400417 // only terminate with zero if string is not empty
418 if (count > 0)
419 putUInt16(0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400420}
421
Mike Lockwood16864ba2010-05-11 17:16:59 -0400422#ifdef MTP_DEVICE
423int MtpDataPacket::read(int fd) {
Mike Lockwood44c19082011-12-01 18:36:06 -0500424 int ret = ::read(fd, mBuffer, MTP_BUFFER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400425 if (ret < MTP_CONTAINER_HEADER_SIZE)
Mike Lockwood16864ba2010-05-11 17:16:59 -0400426 return -1;
Mike Lockwoodef441d92011-07-14 21:00:02 -0400427 mPacketSize = ret;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400428 mOffset = MTP_CONTAINER_HEADER_SIZE;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400429 return ret;
430}
431
432int MtpDataPacket::write(int fd) {
433 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
434 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400435 int ret = ::write(fd, mBuffer, mPacketSize);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400436 return (ret < 0 ? ret : 0);
437}
Mike Lockwood64000782011-04-24 18:40:17 -0700438
439int MtpDataPacket::writeData(int fd, void* data, uint32_t length) {
Marco Nelissenebb10812014-06-23 08:15:13 -0700440 allocate(length + MTP_CONTAINER_HEADER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400441 memcpy(mBuffer + MTP_CONTAINER_HEADER_SIZE, data, length);
442 length += MTP_CONTAINER_HEADER_SIZE;
443 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700444 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400445 int ret = ::write(fd, mBuffer, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700446 return (ret < 0 ? ret : 0);
447}
448
Mike Lockwood16864ba2010-05-11 17:16:59 -0400449#endif // MTP_DEVICE
450
451#ifdef MTP_HOST
Mike Lockwood42d0b792011-01-04 14:48:57 -0500452int MtpDataPacket::read(struct usb_request *request) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400453 // first read the header
Mike Lockwood42d0b792011-01-04 14:48:57 -0500454 request->buffer = mBuffer;
455 request->buffer_length = mBufferSize;
456 int length = transfer(request);
Mike Lockwood437e9452010-07-20 12:01:36 -0400457 if (length >= MTP_CONTAINER_HEADER_SIZE) {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400458 // look at the length field to see if the data spans multiple packets
459 uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET);
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500460 allocate(totalLength);
Mike Lockwood3e072b32010-06-10 16:34:20 -0400461 while (totalLength > length) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500462 request->buffer = mBuffer + length;
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500463 request->buffer_length = totalLength - length;
Mike Lockwood42d0b792011-01-04 14:48:57 -0500464 int ret = transfer(request);
Mike Lockwood3e072b32010-06-10 16:34:20 -0400465 if (ret >= 0)
466 length += ret;
467 else {
468 length = ret;
469 break;
470 }
471 }
472 }
473 if (length >= 0)
474 mPacketSize = length;
475 return length;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400476}
477
Mike Lockwood42d0b792011-01-04 14:48:57 -0500478int MtpDataPacket::readData(struct usb_request *request, void* buffer, int length) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400479 int read = 0;
480 while (read < length) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500481 request->buffer = (char *)buffer + read;
482 request->buffer_length = length - read;
483 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400484 if (ret < 0) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400485 return ret;
486 }
487 read += ret;
488 }
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400489 return read;
490}
491
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500492// Queue a read request. Call readDataWait to wait for result
Mike Lockwood42d0b792011-01-04 14:48:57 -0500493int MtpDataPacket::readDataAsync(struct usb_request *req) {
494 if (usb_request_queue(req)) {
Steve Block29357bc2012-01-06 19:20:56 +0000495 ALOGE("usb_endpoint_queue failed, errno: %d", errno);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500496 return -1;
497 }
498 return 0;
499}
500
501// Wait for result of readDataAsync
Mike Lockwood42d0b792011-01-04 14:48:57 -0500502int MtpDataPacket::readDataWait(struct usb_device *device) {
503 struct usb_request *req = usb_request_wait(device);
504 return (req ? req->actual_length : -1);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500505}
506
Mike Lockwood42d0b792011-01-04 14:48:57 -0500507int MtpDataPacket::readDataHeader(struct usb_request *request) {
508 request->buffer = mBuffer;
509 request->buffer_length = request->max_packet_size;
510 int length = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400511 if (length >= 0)
512 mPacketSize = length;
513 return length;
514}
515
Mike Lockwood42d0b792011-01-04 14:48:57 -0500516int MtpDataPacket::writeDataHeader(struct usb_request *request, uint32_t length) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400517 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
518 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwood42d0b792011-01-04 14:48:57 -0500519 request->buffer = mBuffer;
520 request->buffer_length = MTP_CONTAINER_HEADER_SIZE;
521 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400522 return (ret < 0 ? ret : 0);
523}
524
Mike Lockwood42d0b792011-01-04 14:48:57 -0500525int MtpDataPacket::write(struct usb_request *request) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400526 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
527 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwood42d0b792011-01-04 14:48:57 -0500528 request->buffer = mBuffer;
Tomasz Mikolajewski64c948b2015-08-13 15:31:02 +0900529 request->buffer_length = mPacketSize;
Mike Lockwood42d0b792011-01-04 14:48:57 -0500530 int ret = transfer(request);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400531 return (ret < 0 ? ret : 0);
532}
533
Mike Lockwood42d0b792011-01-04 14:48:57 -0500534int MtpDataPacket::write(struct usb_request *request, void* buffer, uint32_t length) {
535 request->buffer = buffer;
536 request->buffer_length = length;
537 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400538 return (ret < 0 ? ret : 0);
539}
540
Mike Lockwood16864ba2010-05-11 17:16:59 -0400541#endif // MTP_HOST
Mike Lockwood7850ef92010-05-14 10:10:36 -0400542
Mike Lockwood3e072b32010-06-10 16:34:20 -0400543void* MtpDataPacket::getData(int& outLength) const {
544 int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE;
545 if (length > 0) {
546 void* result = malloc(length);
547 if (result) {
548 memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length);
549 outLength = length;
550 return result;
551 }
552 }
553 outLength = 0;
554 return NULL;
555}
556
Mike Lockwood7850ef92010-05-14 10:10:36 -0400557} // namespace android