blob: 0ddee8608080ea6c0db67c68f76b187fd55027ca [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"
Jerry Zhang487be612016-10-24 12:10:41 -070027#include "IMtpHandle.h"
Mike Lockwood16864ba2010-05-11 17:16:59 -040028
Mike Lockwood7850ef92010-05-14 10:10:36 -040029namespace android {
30
Mike Lockwood16864ba2010-05-11 17:16:59 -040031MtpDataPacket::MtpDataPacket()
Mike Lockwood44c19082011-12-01 18:36:06 -050032 : MtpPacket(MTP_BUFFER_SIZE), // MAX_USBFS_BUFFER_SIZE
Mike Lockwood16864ba2010-05-11 17:16:59 -040033 mOffset(MTP_CONTAINER_HEADER_SIZE)
34{
35}
36
37MtpDataPacket::~MtpDataPacket() {
38}
39
40void MtpDataPacket::reset() {
41 MtpPacket::reset();
42 mOffset = MTP_CONTAINER_HEADER_SIZE;
43}
44
45void MtpDataPacket::setOperationCode(MtpOperationCode code) {
46 MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code);
47}
48
49void MtpDataPacket::setTransactionID(MtpTransactionID id) {
50 MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id);
51}
52
Mike Lockwoodab063842014-11-12 14:20:06 -080053bool MtpDataPacket::getUInt8(uint8_t& value) {
54 if (mPacketSize - mOffset < sizeof(value))
55 return false;
56 value = mBuffer[mOffset++];
57 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040058}
59
Mike Lockwoodab063842014-11-12 14:20:06 -080060bool MtpDataPacket::getUInt16(uint16_t& value) {
61 if (mPacketSize - mOffset < sizeof(value))
62 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -040063 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -080064 value = (uint16_t)mBuffer[offset] | ((uint16_t)mBuffer[offset + 1] << 8);
65 mOffset += sizeof(value);
66 return true;
67}
68
69bool MtpDataPacket::getUInt32(uint32_t& value) {
70 if (mPacketSize - mOffset < sizeof(value))
71 return false;
72 int offset = mOffset;
73 value = (uint32_t)mBuffer[offset] | ((uint32_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -040074 ((uint32_t)mBuffer[offset + 2] << 16) | ((uint32_t)mBuffer[offset + 3] << 24);
Mike Lockwoodab063842014-11-12 14:20:06 -080075 mOffset += sizeof(value);
76 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040077}
78
Mike Lockwoodab063842014-11-12 14:20:06 -080079bool MtpDataPacket::getUInt64(uint64_t& value) {
80 if (mPacketSize - mOffset < sizeof(value))
81 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -040082 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -080083 value = (uint64_t)mBuffer[offset] | ((uint64_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -040084 ((uint64_t)mBuffer[offset + 2] << 16) | ((uint64_t)mBuffer[offset + 3] << 24) |
85 ((uint64_t)mBuffer[offset + 4] << 32) | ((uint64_t)mBuffer[offset + 5] << 40) |
86 ((uint64_t)mBuffer[offset + 6] << 48) | ((uint64_t)mBuffer[offset + 7] << 56);
Mike Lockwoodab063842014-11-12 14:20:06 -080087 mOffset += sizeof(value);
88 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040089}
90
Mike Lockwoodab063842014-11-12 14:20:06 -080091bool MtpDataPacket::getUInt128(uint128_t& value) {
92 return getUInt32(value[0]) && getUInt32(value[1]) && getUInt32(value[2]) && getUInt32(value[3]);
Mike Lockwooda6c490b2010-06-05 22:45:01 -040093}
94
Mike Lockwoodab063842014-11-12 14:20:06 -080095bool MtpDataPacket::getString(MtpStringBuffer& string)
Mike Lockwood16864ba2010-05-11 17:16:59 -040096{
Mike Lockwoodab063842014-11-12 14:20:06 -080097 return string.readFromPacket(this);
Mike Lockwood16864ba2010-05-11 17:16:59 -040098}
99
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400100Int8List* MtpDataPacket::getAInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800101 uint32_t count;
102 if (!getUInt32(count))
103 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400104 Int8List* result = new Int8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800105 for (uint32_t i = 0; i < count; i++) {
106 int8_t value;
107 if (!getInt8(value)) {
108 delete result;
109 return NULL;
110 }
111 result->push(value);
112 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400113 return result;
114}
115
116UInt8List* MtpDataPacket::getAUInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800117 uint32_t count;
118 if (!getUInt32(count))
119 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400120 UInt8List* result = new UInt8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800121 for (uint32_t i = 0; i < count; i++) {
122 uint8_t value;
123 if (!getUInt8(value)) {
124 delete result;
125 return NULL;
126 }
127 result->push(value);
128 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400129 return result;
130}
131
132Int16List* MtpDataPacket::getAInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800133 uint32_t count;
134 if (!getUInt32(count))
135 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400136 Int16List* result = new Int16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800137 for (uint32_t i = 0; i < count; i++) {
138 int16_t value;
139 if (!getInt16(value)) {
140 delete result;
141 return NULL;
142 }
143 result->push(value);
144 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400145 return result;
146}
147
148UInt16List* MtpDataPacket::getAUInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800149 uint32_t count;
150 if (!getUInt32(count))
151 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400152 UInt16List* result = new UInt16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800153 for (uint32_t i = 0; i < count; i++) {
154 uint16_t value;
155 if (!getUInt16(value)) {
156 delete result;
157 return NULL;
158 }
159 result->push(value);
160 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400161 return result;
162}
163
164Int32List* MtpDataPacket::getAInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800165 uint32_t count;
166 if (!getUInt32(count))
167 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400168 Int32List* result = new Int32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800169 for (uint32_t i = 0; i < count; i++) {
170 int32_t value;
171 if (!getInt32(value)) {
172 delete result;
173 return NULL;
174 }
175 result->push(value);
176 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400177 return result;
178}
179
180UInt32List* MtpDataPacket::getAUInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800181 uint32_t count;
182 if (!getUInt32(count))
183 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400184 UInt32List* result = new UInt32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800185 for (uint32_t i = 0; i < count; i++) {
186 uint32_t value;
187 if (!getUInt32(value)) {
188 delete result;
189 return NULL;
190 }
191 result->push(value);
192 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400193 return result;
194}
195
196Int64List* MtpDataPacket::getAInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800197 uint32_t count;
198 if (!getUInt32(count))
199 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400200 Int64List* result = new Int64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800201 for (uint32_t i = 0; i < count; i++) {
202 int64_t value;
203 if (!getInt64(value)) {
204 delete result;
205 return NULL;
206 }
207 result->push(value);
208 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400209 return result;
210}
211
212UInt64List* MtpDataPacket::getAUInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800213 uint32_t count;
214 if (!getUInt32(count))
215 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400216 UInt64List* result = new UInt64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800217 for (uint32_t i = 0; i < count; i++) {
218 uint64_t value;
219 if (!getUInt64(value)) {
220 delete result;
221 return NULL;
222 }
223 result->push(value);
224 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400225 return result;
226}
227
Mike Lockwood16864ba2010-05-11 17:16:59 -0400228void MtpDataPacket::putInt8(int8_t value) {
229 allocate(mOffset + 1);
230 mBuffer[mOffset++] = (uint8_t)value;
231 if (mPacketSize < mOffset)
232 mPacketSize = mOffset;
233}
234
235void MtpDataPacket::putUInt8(uint8_t value) {
236 allocate(mOffset + 1);
237 mBuffer[mOffset++] = (uint8_t)value;
238 if (mPacketSize < mOffset)
239 mPacketSize = mOffset;
240}
241
242void MtpDataPacket::putInt16(int16_t value) {
243 allocate(mOffset + 2);
244 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
245 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
246 if (mPacketSize < mOffset)
247 mPacketSize = mOffset;
248}
249
250void MtpDataPacket::putUInt16(uint16_t value) {
251 allocate(mOffset + 2);
252 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
253 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
254 if (mPacketSize < mOffset)
255 mPacketSize = mOffset;
256}
257
258void MtpDataPacket::putInt32(int32_t value) {
259 allocate(mOffset + 4);
260 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
261 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
262 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
263 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
264 if (mPacketSize < mOffset)
265 mPacketSize = mOffset;
266}
267
268void MtpDataPacket::putUInt32(uint32_t value) {
269 allocate(mOffset + 4);
270 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
271 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
272 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
273 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
274 if (mPacketSize < mOffset)
275 mPacketSize = mOffset;
276}
277
278void MtpDataPacket::putInt64(int64_t value) {
279 allocate(mOffset + 8);
280 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
281 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
282 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
283 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
284 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
285 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
286 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
287 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
288 if (mPacketSize < mOffset)
289 mPacketSize = mOffset;
290}
291
292void MtpDataPacket::putUInt64(uint64_t value) {
293 allocate(mOffset + 8);
294 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
295 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
296 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
297 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
298 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
299 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
300 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
301 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
302 if (mPacketSize < mOffset)
303 mPacketSize = mOffset;
304}
305
Mike Lockwooda6c490b2010-06-05 22:45:01 -0400306void MtpDataPacket::putInt128(const int128_t& value) {
307 putInt32(value[0]);
308 putInt32(value[1]);
309 putInt32(value[2]);
310 putInt32(value[3]);
311}
312
313void MtpDataPacket::putUInt128(const uint128_t& value) {
314 putUInt32(value[0]);
315 putUInt32(value[1]);
316 putUInt32(value[2]);
317 putUInt32(value[3]);
318}
319
Mike Lockwood8277cec2010-08-10 15:20:35 -0400320void MtpDataPacket::putInt128(int64_t value) {
321 putInt64(value);
Mike Lockwood59599882010-08-25 19:10:24 -0400322 putInt64(value < 0 ? -1 : 0);
Mike Lockwood8277cec2010-08-10 15:20:35 -0400323}
324
325void MtpDataPacket::putUInt128(uint64_t value) {
326 putUInt64(value);
327 putUInt64(0);
328}
329
Mike Lockwood16864ba2010-05-11 17:16:59 -0400330void MtpDataPacket::putAInt8(const int8_t* values, int count) {
331 putUInt32(count);
332 for (int i = 0; i < count; i++)
333 putInt8(*values++);
334}
335
336void MtpDataPacket::putAUInt8(const uint8_t* values, int count) {
337 putUInt32(count);
338 for (int i = 0; i < count; i++)
339 putUInt8(*values++);
340}
341
342void MtpDataPacket::putAInt16(const int16_t* values, int count) {
343 putUInt32(count);
344 for (int i = 0; i < count; i++)
345 putInt16(*values++);
346}
347
348void MtpDataPacket::putAUInt16(const uint16_t* values, int count) {
349 putUInt32(count);
350 for (int i = 0; i < count; i++)
351 putUInt16(*values++);
352}
353
Mike Lockwood782aef12010-08-10 07:37:50 -0400354void MtpDataPacket::putAUInt16(const UInt16List* values) {
355 size_t count = (values ? values->size() : 0);
356 putUInt32(count);
357 for (size_t i = 0; i < count; i++)
358 putUInt16((*values)[i]);
359}
360
Mike Lockwood16864ba2010-05-11 17:16:59 -0400361void MtpDataPacket::putAInt32(const int32_t* values, int count) {
362 putUInt32(count);
363 for (int i = 0; i < count; i++)
364 putInt32(*values++);
365}
366
367void MtpDataPacket::putAUInt32(const uint32_t* values, int count) {
368 putUInt32(count);
369 for (int i = 0; i < count; i++)
370 putUInt32(*values++);
371}
372
373void MtpDataPacket::putAUInt32(const UInt32List* list) {
374 if (!list) {
375 putEmptyArray();
376 } else {
377 size_t size = list->size();
378 putUInt32(size);
379 for (size_t i = 0; i < size; i++)
380 putUInt32((*list)[i]);
381 }
382}
383
384void MtpDataPacket::putAInt64(const int64_t* values, int count) {
385 putUInt32(count);
386 for (int i = 0; i < count; i++)
387 putInt64(*values++);
388}
389
390void MtpDataPacket::putAUInt64(const uint64_t* values, int count) {
391 putUInt32(count);
392 for (int i = 0; i < count; i++)
393 putUInt64(*values++);
394}
395
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400396void MtpDataPacket::putString(const MtpStringBuffer& string) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400397 string.writeToPacket(this);
398}
399
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400400void MtpDataPacket::putString(const char* s) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400401 MtpStringBuffer string(s);
402 string.writeToPacket(this);
403}
404
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400405void MtpDataPacket::putString(const uint16_t* string) {
406 int count = 0;
Yin Liu014897f2012-12-04 09:19:53 +0100407 for (int i = 0; i <= MTP_STRING_MAX_CHARACTER_NUMBER; i++) {
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400408 if (string[i])
409 count++;
410 else
411 break;
412 }
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400413 putUInt8(count > 0 ? count + 1 : 0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400414 for (int i = 0; i < count; i++)
415 putUInt16(string[i]);
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400416 // only terminate with zero if string is not empty
417 if (count > 0)
418 putUInt16(0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400419}
420
Jerry Zhang487be612016-10-24 12:10:41 -0700421#ifdef MTP_DEVICE
422int MtpDataPacket::read(IMtpHandle *h) {
423 int ret = h->read(mBuffer, MTP_BUFFER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400424 if (ret < MTP_CONTAINER_HEADER_SIZE)
Mike Lockwood16864ba2010-05-11 17:16:59 -0400425 return -1;
Mike Lockwoodef441d92011-07-14 21:00:02 -0400426 mPacketSize = ret;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400427 mOffset = MTP_CONTAINER_HEADER_SIZE;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400428 return ret;
429}
430
Jerry Zhang487be612016-10-24 12:10:41 -0700431int MtpDataPacket::write(IMtpHandle *h) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400432 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
433 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Jerry Zhang487be612016-10-24 12:10:41 -0700434 int ret = h->write(mBuffer, mPacketSize);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400435 return (ret < 0 ? ret : 0);
436}
Mike Lockwood64000782011-04-24 18:40:17 -0700437
Jerry Zhang487be612016-10-24 12:10:41 -0700438int MtpDataPacket::writeData(IMtpHandle *h, void* data, uint32_t length) {
Marco Nelissenebb10812014-06-23 08:15:13 -0700439 allocate(length + MTP_CONTAINER_HEADER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400440 memcpy(mBuffer + MTP_CONTAINER_HEADER_SIZE, data, length);
441 length += MTP_CONTAINER_HEADER_SIZE;
442 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700443 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Jerry Zhang487be612016-10-24 12:10:41 -0700444 int ret = h->write(mBuffer, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700445 return (ret < 0 ? ret : 0);
446}
447
Mike Lockwood16864ba2010-05-11 17:16:59 -0400448#endif // MTP_DEVICE
449
450#ifdef MTP_HOST
Mike Lockwood42d0b792011-01-04 14:48:57 -0500451int MtpDataPacket::read(struct usb_request *request) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400452 // first read the header
Mike Lockwood42d0b792011-01-04 14:48:57 -0500453 request->buffer = mBuffer;
454 request->buffer_length = mBufferSize;
455 int length = transfer(request);
Mike Lockwood437e9452010-07-20 12:01:36 -0400456 if (length >= MTP_CONTAINER_HEADER_SIZE) {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400457 // look at the length field to see if the data spans multiple packets
458 uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET);
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500459 allocate(totalLength);
Daichi Hironob3be0062016-02-25 12:42:58 +0900460 while (totalLength > static_cast<uint32_t>(length)) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500461 request->buffer = mBuffer + length;
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500462 request->buffer_length = totalLength - length;
Mike Lockwood42d0b792011-01-04 14:48:57 -0500463 int ret = transfer(request);
Mike Lockwood3e072b32010-06-10 16:34:20 -0400464 if (ret >= 0)
465 length += ret;
466 else {
467 length = ret;
468 break;
469 }
470 }
471 }
472 if (length >= 0)
473 mPacketSize = length;
474 return length;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400475}
476
Mike Lockwood42d0b792011-01-04 14:48:57 -0500477int MtpDataPacket::readData(struct usb_request *request, void* buffer, int length) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400478 int read = 0;
479 while (read < length) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500480 request->buffer = (char *)buffer + read;
481 request->buffer_length = length - read;
482 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400483 if (ret < 0) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400484 return ret;
485 }
486 read += ret;
487 }
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400488 return read;
489}
490
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500491// Queue a read request. Call readDataWait to wait for result
Mike Lockwood42d0b792011-01-04 14:48:57 -0500492int MtpDataPacket::readDataAsync(struct usb_request *req) {
493 if (usb_request_queue(req)) {
Steve Block29357bc2012-01-06 19:20:56 +0000494 ALOGE("usb_endpoint_queue failed, errno: %d", errno);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500495 return -1;
496 }
497 return 0;
498}
499
500// Wait for result of readDataAsync
Mike Lockwood42d0b792011-01-04 14:48:57 -0500501int MtpDataPacket::readDataWait(struct usb_device *device) {
502 struct usb_request *req = usb_request_wait(device);
503 return (req ? req->actual_length : -1);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500504}
505
Mike Lockwood42d0b792011-01-04 14:48:57 -0500506int MtpDataPacket::readDataHeader(struct usb_request *request) {
507 request->buffer = mBuffer;
508 request->buffer_length = request->max_packet_size;
509 int length = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400510 if (length >= 0)
511 mPacketSize = length;
512 return length;
513}
514
Mike Lockwood42d0b792011-01-04 14:48:57 -0500515int MtpDataPacket::writeDataHeader(struct usb_request *request, uint32_t length) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400516 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
517 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwood42d0b792011-01-04 14:48:57 -0500518 request->buffer = mBuffer;
519 request->buffer_length = MTP_CONTAINER_HEADER_SIZE;
520 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400521 return (ret < 0 ? ret : 0);
522}
523
Mike Lockwood42d0b792011-01-04 14:48:57 -0500524int MtpDataPacket::write(struct usb_request *request) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400525 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
526 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Mike Lockwood42d0b792011-01-04 14:48:57 -0500527 request->buffer = mBuffer;
Tomasz Mikolajewski64c948b2015-08-13 15:31:02 +0900528 request->buffer_length = mPacketSize;
Mike Lockwood42d0b792011-01-04 14:48:57 -0500529 int ret = transfer(request);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400530 return (ret < 0 ? ret : 0);
531}
532
Mike Lockwood42d0b792011-01-04 14:48:57 -0500533int MtpDataPacket::write(struct usb_request *request, void* buffer, uint32_t length) {
534 request->buffer = buffer;
535 request->buffer_length = length;
536 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400537 return (ret < 0 ? ret : 0);
538}
539
Mike Lockwood16864ba2010-05-11 17:16:59 -0400540#endif // MTP_HOST
Mike Lockwood7850ef92010-05-14 10:10:36 -0400541
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900542void* MtpDataPacket::getData(int* outLength) const {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400543 int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE;
544 if (length > 0) {
545 void* result = malloc(length);
546 if (result) {
547 memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length);
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900548 *outLength = length;
Mike Lockwood3e072b32010-06-10 16:34:20 -0400549 return result;
550 }
551 }
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900552 *outLength = 0;
Mike Lockwood3e072b32010-06-10 16:34:20 -0400553 return NULL;
554}
555
Mike Lockwood7850ef92010-05-14 10:10:36 -0400556} // namespace android