blob: a2b1c5e3ae64955e42df66e245cf96b96a3afd1d [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
Daichi Hironod4b42962016-10-04 17:34:43 +090019#include "MtpDataPacket.h"
20
21#include <algorithm>
Jerry Zhangbc1d4b42018-03-27 15:25:03 -070022#include <errno.h>
Daichi Hironod4b42962016-10-04 17:34:43 +090023#include <fcntl.h>
Mike Lockwood16864ba2010-05-11 17:16:59 -040024#include <stdio.h>
25#include <sys/types.h>
Mike Lockwood0cf89f22010-07-26 20:40:45 -040026#include <usbhost/usbhost.h>
Mike Lockwood16864ba2010-05-11 17:16:59 -040027#include "MtpStringBuffer.h"
Jerry Zhang487be612016-10-24 12:10:41 -070028#include "IMtpHandle.h"
Mike Lockwood16864ba2010-05-11 17:16:59 -040029
Mike Lockwood7850ef92010-05-14 10:10:36 -040030namespace android {
31
Daichi Hironod4b42962016-10-04 17:34:43 +090032namespace {
33// Reads the exact |count| bytes from |fd| to |buf|.
34// Returns |count| if it succeed to read the bytes. Otherwise returns -1. If it reaches EOF, the
35// function regards it as an error.
36ssize_t readExactBytes(int fd, void* buf, size_t count) {
37 if (count > SSIZE_MAX) {
38 return -1;
39 }
40 size_t read_count = 0;
41 while (read_count < count) {
42 int result = read(fd, static_cast<int8_t*>(buf) + read_count, count - read_count);
43 // Assume that EOF is error.
44 if (result <= 0) {
45 return -1;
46 }
47 read_count += result;
48 }
49 return read_count == count ? count : -1;
50}
51} // namespace
52
Mike Lockwood16864ba2010-05-11 17:16:59 -040053MtpDataPacket::MtpDataPacket()
Mike Lockwood44c19082011-12-01 18:36:06 -050054 : MtpPacket(MTP_BUFFER_SIZE), // MAX_USBFS_BUFFER_SIZE
Mike Lockwood16864ba2010-05-11 17:16:59 -040055 mOffset(MTP_CONTAINER_HEADER_SIZE)
56{
57}
58
59MtpDataPacket::~MtpDataPacket() {
60}
61
62void MtpDataPacket::reset() {
63 MtpPacket::reset();
64 mOffset = MTP_CONTAINER_HEADER_SIZE;
65}
66
67void MtpDataPacket::setOperationCode(MtpOperationCode code) {
68 MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code);
69}
70
71void MtpDataPacket::setTransactionID(MtpTransactionID id) {
72 MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id);
73}
74
Mike Lockwoodab063842014-11-12 14:20:06 -080075bool MtpDataPacket::getUInt8(uint8_t& value) {
Shruti Bihanid8cc2062023-07-20 09:10:41 +000076 if ((mPacketSize - mOffset < sizeof(value)) || (mOffset >= mBufferSize))
Mike Lockwoodab063842014-11-12 14:20:06 -080077 return false;
78 value = mBuffer[mOffset++];
79 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040080}
81
Mike Lockwoodab063842014-11-12 14:20:06 -080082bool MtpDataPacket::getUInt16(uint16_t& value) {
Shruti Bihanid8cc2062023-07-20 09:10:41 +000083 if ((mPacketSize - mOffset < sizeof(value)) || ((mOffset+1) >= mBufferSize))
Mike Lockwoodab063842014-11-12 14:20:06 -080084 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -040085 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -080086 value = (uint16_t)mBuffer[offset] | ((uint16_t)mBuffer[offset + 1] << 8);
87 mOffset += sizeof(value);
88 return true;
89}
90
91bool MtpDataPacket::getUInt32(uint32_t& value) {
Shruti Bihanid8cc2062023-07-20 09:10:41 +000092 if ((mPacketSize - mOffset < sizeof(value)) || ((mOffset+3) >= mBufferSize))
Mike Lockwoodab063842014-11-12 14:20:06 -080093 return false;
94 int offset = mOffset;
95 value = (uint32_t)mBuffer[offset] | ((uint32_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -040096 ((uint32_t)mBuffer[offset + 2] << 16) | ((uint32_t)mBuffer[offset + 3] << 24);
Mike Lockwoodab063842014-11-12 14:20:06 -080097 mOffset += sizeof(value);
98 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -040099}
100
Mike Lockwoodab063842014-11-12 14:20:06 -0800101bool MtpDataPacket::getUInt64(uint64_t& value) {
Shruti Bihanid8cc2062023-07-20 09:10:41 +0000102 if ((mPacketSize - mOffset < sizeof(value)) || ((mOffset+7) >= mBufferSize))
Mike Lockwoodab063842014-11-12 14:20:06 -0800103 return false;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400104 int offset = mOffset;
Mike Lockwoodab063842014-11-12 14:20:06 -0800105 value = (uint64_t)mBuffer[offset] | ((uint64_t)mBuffer[offset + 1] << 8) |
Mike Lockwood16864ba2010-05-11 17:16:59 -0400106 ((uint64_t)mBuffer[offset + 2] << 16) | ((uint64_t)mBuffer[offset + 3] << 24) |
107 ((uint64_t)mBuffer[offset + 4] << 32) | ((uint64_t)mBuffer[offset + 5] << 40) |
108 ((uint64_t)mBuffer[offset + 6] << 48) | ((uint64_t)mBuffer[offset + 7] << 56);
Mike Lockwoodab063842014-11-12 14:20:06 -0800109 mOffset += sizeof(value);
110 return true;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400111}
112
Mike Lockwoodab063842014-11-12 14:20:06 -0800113bool MtpDataPacket::getUInt128(uint128_t& value) {
114 return getUInt32(value[0]) && getUInt32(value[1]) && getUInt32(value[2]) && getUInt32(value[3]);
Mike Lockwooda6c490b2010-06-05 22:45:01 -0400115}
116
Mike Lockwoodab063842014-11-12 14:20:06 -0800117bool MtpDataPacket::getString(MtpStringBuffer& string)
Mike Lockwood16864ba2010-05-11 17:16:59 -0400118{
Mike Lockwoodab063842014-11-12 14:20:06 -0800119 return string.readFromPacket(this);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400120}
121
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400122Int8List* MtpDataPacket::getAInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800123 uint32_t count;
124 if (!getUInt32(count))
125 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400126 Int8List* result = new Int8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800127 for (uint32_t i = 0; i < count; i++) {
128 int8_t value;
129 if (!getInt8(value)) {
130 delete result;
131 return NULL;
132 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700133 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800134 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400135 return result;
136}
137
138UInt8List* MtpDataPacket::getAUInt8() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800139 uint32_t count;
140 if (!getUInt32(count))
141 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400142 UInt8List* result = new UInt8List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800143 for (uint32_t i = 0; i < count; i++) {
144 uint8_t value;
145 if (!getUInt8(value)) {
146 delete result;
147 return NULL;
148 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700149 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800150 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400151 return result;
152}
153
154Int16List* MtpDataPacket::getAInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800155 uint32_t count;
156 if (!getUInt32(count))
157 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400158 Int16List* result = new Int16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800159 for (uint32_t i = 0; i < count; i++) {
160 int16_t value;
161 if (!getInt16(value)) {
162 delete result;
163 return NULL;
164 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700165 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800166 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400167 return result;
168}
169
170UInt16List* MtpDataPacket::getAUInt16() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800171 uint32_t count;
172 if (!getUInt32(count))
173 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400174 UInt16List* result = new UInt16List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800175 for (uint32_t i = 0; i < count; i++) {
176 uint16_t value;
177 if (!getUInt16(value)) {
178 delete result;
179 return NULL;
180 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700181 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800182 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400183 return result;
184}
185
186Int32List* MtpDataPacket::getAInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800187 uint32_t count;
188 if (!getUInt32(count))
189 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400190 Int32List* result = new Int32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800191 for (uint32_t i = 0; i < count; i++) {
192 int32_t value;
193 if (!getInt32(value)) {
194 delete result;
195 return NULL;
196 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700197 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800198 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400199 return result;
200}
201
202UInt32List* MtpDataPacket::getAUInt32() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800203 uint32_t count;
204 if (!getUInt32(count))
205 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400206 UInt32List* result = new UInt32List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800207 for (uint32_t i = 0; i < count; i++) {
208 uint32_t value;
209 if (!getUInt32(value)) {
210 delete result;
211 return NULL;
212 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700213 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800214 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400215 return result;
216}
217
218Int64List* MtpDataPacket::getAInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800219 uint32_t count;
220 if (!getUInt32(count))
221 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400222 Int64List* result = new Int64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800223 for (uint32_t i = 0; i < count; i++) {
224 int64_t value;
225 if (!getInt64(value)) {
226 delete result;
227 return NULL;
228 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700229 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800230 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400231 return result;
232}
233
234UInt64List* MtpDataPacket::getAUInt64() {
Mike Lockwoodab063842014-11-12 14:20:06 -0800235 uint32_t count;
236 if (!getUInt32(count))
237 return NULL;
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400238 UInt64List* result = new UInt64List;
Mike Lockwoodab063842014-11-12 14:20:06 -0800239 for (uint32_t i = 0; i < count; i++) {
240 uint64_t value;
241 if (!getUInt64(value)) {
242 delete result;
243 return NULL;
244 }
Jerry Zhangbc1d4b42018-03-27 15:25:03 -0700245 result->push_back(value);
Mike Lockwoodab063842014-11-12 14:20:06 -0800246 }
Mike Lockwood335dd2b2010-05-19 10:33:39 -0400247 return result;
248}
249
Mike Lockwood16864ba2010-05-11 17:16:59 -0400250void MtpDataPacket::putInt8(int8_t value) {
251 allocate(mOffset + 1);
252 mBuffer[mOffset++] = (uint8_t)value;
253 if (mPacketSize < mOffset)
254 mPacketSize = mOffset;
255}
256
257void MtpDataPacket::putUInt8(uint8_t value) {
258 allocate(mOffset + 1);
259 mBuffer[mOffset++] = (uint8_t)value;
260 if (mPacketSize < mOffset)
261 mPacketSize = mOffset;
262}
263
264void MtpDataPacket::putInt16(int16_t value) {
265 allocate(mOffset + 2);
266 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
267 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
268 if (mPacketSize < mOffset)
269 mPacketSize = mOffset;
270}
271
272void MtpDataPacket::putUInt16(uint16_t value) {
273 allocate(mOffset + 2);
274 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
275 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
276 if (mPacketSize < mOffset)
277 mPacketSize = mOffset;
278}
279
280void MtpDataPacket::putInt32(int32_t value) {
281 allocate(mOffset + 4);
282 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
283 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
284 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
285 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
286 if (mPacketSize < mOffset)
287 mPacketSize = mOffset;
288}
289
290void MtpDataPacket::putUInt32(uint32_t value) {
291 allocate(mOffset + 4);
292 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
293 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
294 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
295 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
296 if (mPacketSize < mOffset)
297 mPacketSize = mOffset;
298}
299
300void MtpDataPacket::putInt64(int64_t value) {
301 allocate(mOffset + 8);
302 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
303 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
304 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
305 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
306 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
307 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
308 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
309 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
310 if (mPacketSize < mOffset)
311 mPacketSize = mOffset;
312}
313
314void MtpDataPacket::putUInt64(uint64_t value) {
315 allocate(mOffset + 8);
316 mBuffer[mOffset++] = (uint8_t)(value & 0xFF);
317 mBuffer[mOffset++] = (uint8_t)((value >> 8) & 0xFF);
318 mBuffer[mOffset++] = (uint8_t)((value >> 16) & 0xFF);
319 mBuffer[mOffset++] = (uint8_t)((value >> 24) & 0xFF);
320 mBuffer[mOffset++] = (uint8_t)((value >> 32) & 0xFF);
321 mBuffer[mOffset++] = (uint8_t)((value >> 40) & 0xFF);
322 mBuffer[mOffset++] = (uint8_t)((value >> 48) & 0xFF);
323 mBuffer[mOffset++] = (uint8_t)((value >> 56) & 0xFF);
324 if (mPacketSize < mOffset)
325 mPacketSize = mOffset;
326}
327
Mike Lockwooda6c490b2010-06-05 22:45:01 -0400328void MtpDataPacket::putInt128(const int128_t& value) {
329 putInt32(value[0]);
330 putInt32(value[1]);
331 putInt32(value[2]);
332 putInt32(value[3]);
333}
334
335void MtpDataPacket::putUInt128(const uint128_t& value) {
336 putUInt32(value[0]);
337 putUInt32(value[1]);
338 putUInt32(value[2]);
339 putUInt32(value[3]);
340}
341
Mike Lockwood8277cec2010-08-10 15:20:35 -0400342void MtpDataPacket::putInt128(int64_t value) {
343 putInt64(value);
Mike Lockwood59599882010-08-25 19:10:24 -0400344 putInt64(value < 0 ? -1 : 0);
Mike Lockwood8277cec2010-08-10 15:20:35 -0400345}
346
347void MtpDataPacket::putUInt128(uint64_t value) {
348 putUInt64(value);
349 putUInt64(0);
350}
351
Mike Lockwood16864ba2010-05-11 17:16:59 -0400352void MtpDataPacket::putAInt8(const int8_t* values, int count) {
353 putUInt32(count);
354 for (int i = 0; i < count; i++)
355 putInt8(*values++);
356}
357
358void MtpDataPacket::putAUInt8(const uint8_t* values, int count) {
359 putUInt32(count);
360 for (int i = 0; i < count; i++)
361 putUInt8(*values++);
362}
363
364void MtpDataPacket::putAInt16(const int16_t* values, int count) {
365 putUInt32(count);
366 for (int i = 0; i < count; i++)
367 putInt16(*values++);
368}
369
370void MtpDataPacket::putAUInt16(const uint16_t* values, int count) {
371 putUInt32(count);
372 for (int i = 0; i < count; i++)
373 putUInt16(*values++);
374}
375
Mike Lockwood782aef12010-08-10 07:37:50 -0400376void MtpDataPacket::putAUInt16(const UInt16List* values) {
377 size_t count = (values ? values->size() : 0);
378 putUInt32(count);
379 for (size_t i = 0; i < count; i++)
380 putUInt16((*values)[i]);
381}
382
Mike Lockwood16864ba2010-05-11 17:16:59 -0400383void MtpDataPacket::putAInt32(const int32_t* values, int count) {
384 putUInt32(count);
385 for (int i = 0; i < count; i++)
386 putInt32(*values++);
387}
388
389void MtpDataPacket::putAUInt32(const uint32_t* values, int count) {
390 putUInt32(count);
391 for (int i = 0; i < count; i++)
392 putUInt32(*values++);
393}
394
395void MtpDataPacket::putAUInt32(const UInt32List* list) {
396 if (!list) {
397 putEmptyArray();
398 } else {
399 size_t size = list->size();
400 putUInt32(size);
401 for (size_t i = 0; i < size; i++)
402 putUInt32((*list)[i]);
403 }
404}
405
406void MtpDataPacket::putAInt64(const int64_t* values, int count) {
407 putUInt32(count);
408 for (int i = 0; i < count; i++)
409 putInt64(*values++);
410}
411
412void MtpDataPacket::putAUInt64(const uint64_t* values, int count) {
413 putUInt32(count);
414 for (int i = 0; i < count; i++)
415 putUInt64(*values++);
416}
417
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400418void MtpDataPacket::putString(const MtpStringBuffer& string) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400419 string.writeToPacket(this);
420}
421
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400422void MtpDataPacket::putString(const char* s) {
Ashish Kumar Gupta4d9fccf2023-11-15 10:28:33 +0000423 if (s != NULL) {
424 MtpStringBuffer string(s);
425 string.writeToPacket(this);
426 }
Mike Lockwood16864ba2010-05-11 17:16:59 -0400427}
428
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400429void MtpDataPacket::putString(const uint16_t* string) {
Ashish Kumar Gupta79a02052023-12-12 05:03:11 +0000430 if (string == NULL) {
431 return;
432 }
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400433 int count = 0;
Yin Liu014897f2012-12-04 09:19:53 +0100434 for (int i = 0; i <= MTP_STRING_MAX_CHARACTER_NUMBER; i++) {
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400435 if (string[i])
436 count++;
437 else
438 break;
439 }
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400440 putUInt8(count > 0 ? count + 1 : 0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400441 for (int i = 0; i < count; i++)
442 putUInt16(string[i]);
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400443 // only terminate with zero if string is not empty
444 if (count > 0)
445 putUInt16(0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400446}
447
Jerry Zhang487be612016-10-24 12:10:41 -0700448#ifdef MTP_DEVICE
449int MtpDataPacket::read(IMtpHandle *h) {
450 int ret = h->read(mBuffer, MTP_BUFFER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400451 if (ret < MTP_CONTAINER_HEADER_SIZE)
Mike Lockwood16864ba2010-05-11 17:16:59 -0400452 return -1;
Mike Lockwoodef441d92011-07-14 21:00:02 -0400453 mPacketSize = ret;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400454 mOffset = MTP_CONTAINER_HEADER_SIZE;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400455 return ret;
456}
457
Jerry Zhang487be612016-10-24 12:10:41 -0700458int MtpDataPacket::write(IMtpHandle *h) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400459 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
460 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Jerry Zhang487be612016-10-24 12:10:41 -0700461 int ret = h->write(mBuffer, mPacketSize);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400462 return (ret < 0 ? ret : 0);
463}
Mike Lockwood64000782011-04-24 18:40:17 -0700464
Jerry Zhang487be612016-10-24 12:10:41 -0700465int MtpDataPacket::writeData(IMtpHandle *h, void* data, uint32_t length) {
Marco Nelissenebb10812014-06-23 08:15:13 -0700466 allocate(length + MTP_CONTAINER_HEADER_SIZE);
Mike Lockwoodef441d92011-07-14 21:00:02 -0400467 memcpy(mBuffer + MTP_CONTAINER_HEADER_SIZE, data, length);
468 length += MTP_CONTAINER_HEADER_SIZE;
469 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700470 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Jerry Zhang487be612016-10-24 12:10:41 -0700471 int ret = h->write(mBuffer, length);
Mike Lockwood64000782011-04-24 18:40:17 -0700472 return (ret < 0 ? ret : 0);
473}
474
Mike Lockwood16864ba2010-05-11 17:16:59 -0400475#endif // MTP_DEVICE
476
477#ifdef MTP_HOST
Mike Lockwood42d0b792011-01-04 14:48:57 -0500478int MtpDataPacket::read(struct usb_request *request) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400479 // first read the header
Mike Lockwood42d0b792011-01-04 14:48:57 -0500480 request->buffer = mBuffer;
481 request->buffer_length = mBufferSize;
482 int length = transfer(request);
Mike Lockwood437e9452010-07-20 12:01:36 -0400483 if (length >= MTP_CONTAINER_HEADER_SIZE) {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400484 // look at the length field to see if the data spans multiple packets
485 uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET);
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500486 allocate(totalLength);
Daichi Hironob3be0062016-02-25 12:42:58 +0900487 while (totalLength > static_cast<uint32_t>(length)) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500488 request->buffer = mBuffer + length;
Mike Lockwood33bde8d2011-03-12 14:03:23 -0500489 request->buffer_length = totalLength - length;
Mike Lockwood42d0b792011-01-04 14:48:57 -0500490 int ret = transfer(request);
Mike Lockwood3e072b32010-06-10 16:34:20 -0400491 if (ret >= 0)
492 length += ret;
493 else {
494 length = ret;
495 break;
496 }
497 }
498 }
499 if (length >= 0)
500 mPacketSize = length;
501 return length;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400502}
503
Mike Lockwood42d0b792011-01-04 14:48:57 -0500504int MtpDataPacket::readData(struct usb_request *request, void* buffer, int length) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400505 int read = 0;
506 while (read < length) {
Mike Lockwood42d0b792011-01-04 14:48:57 -0500507 request->buffer = (char *)buffer + read;
508 request->buffer_length = length - read;
509 int ret = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400510 if (ret < 0) {
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400511 return ret;
512 }
513 read += ret;
514 }
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400515 return read;
516}
517
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500518// Queue a read request. Call readDataWait to wait for result
Mike Lockwood42d0b792011-01-04 14:48:57 -0500519int MtpDataPacket::readDataAsync(struct usb_request *req) {
520 if (usb_request_queue(req)) {
Steve Block29357bc2012-01-06 19:20:56 +0000521 ALOGE("usb_endpoint_queue failed, errno: %d", errno);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500522 return -1;
523 }
524 return 0;
525}
526
527// Wait for result of readDataAsync
Mike Lockwood42d0b792011-01-04 14:48:57 -0500528int MtpDataPacket::readDataWait(struct usb_device *device) {
Philip P. Moltmanncb8d8af2016-10-19 10:11:22 -0700529 struct usb_request *req = usb_request_wait(device, -1);
Mike Lockwood42d0b792011-01-04 14:48:57 -0500530 return (req ? req->actual_length : -1);
Mike Lockwoodb9ff4442010-11-19 11:20:19 -0500531}
532
Mike Lockwood42d0b792011-01-04 14:48:57 -0500533int MtpDataPacket::readDataHeader(struct usb_request *request) {
534 request->buffer = mBuffer;
535 request->buffer_length = request->max_packet_size;
536 int length = transfer(request);
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400537 if (length >= 0)
538 mPacketSize = length;
539 return length;
540}
541
Daichi Hironod4b42962016-10-04 17:34:43 +0900542int MtpDataPacket::write(struct usb_request *request, UrbPacketDivisionMode divisionMode) {
543 if (mPacketSize < MTP_CONTAINER_HEADER_SIZE || mPacketSize > MTP_BUFFER_SIZE) {
544 ALOGE("Illegal packet size.");
545 return -1;
546 }
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400547
Mike Lockwood16864ba2010-05-11 17:16:59 -0400548 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
549 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
Daichi Hironod4b42962016-10-04 17:34:43 +0900550
551 size_t processedBytes = 0;
552 while (processedBytes < mPacketSize) {
553 const size_t write_size =
554 processedBytes == 0 && divisionMode == FIRST_PACKET_ONLY_HEADER ?
555 MTP_CONTAINER_HEADER_SIZE : mPacketSize - processedBytes;
556 request->buffer = mBuffer + processedBytes;
557 request->buffer_length = write_size;
558 const int result = transfer(request);
559 if (result < 0) {
560 ALOGE("Failed to write bytes to the device.");
561 return -1;
562 }
563 processedBytes += result;
564 }
565
566 return processedBytes == mPacketSize ? processedBytes : -1;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400567}
568
robinz1xdf4b5072018-07-09 10:37:56 +0800569int64_t MtpDataPacket::write(struct usb_request *request,
Daichi Hironod4b42962016-10-04 17:34:43 +0900570 UrbPacketDivisionMode divisionMode,
571 int fd,
572 size_t payloadSize) {
573 // Obtain the greatest multiple of minimum packet size that is not greater than
574 // MTP_BUFFER_SIZE.
575 if (request->max_packet_size <= 0) {
576 ALOGE("Cannot determine bulk transfer size due to illegal max packet size %d.",
577 request->max_packet_size);
578 return -1;
579 }
580 const size_t maxBulkTransferSize =
581 MTP_BUFFER_SIZE - (MTP_BUFFER_SIZE % request->max_packet_size);
582 const size_t containerLength = payloadSize + MTP_CONTAINER_HEADER_SIZE;
583 size_t processedBytes = 0;
584 bool readError = false;
585
586 // Bind the packet with given request.
587 request->buffer = mBuffer;
588 allocate(maxBulkTransferSize);
589
590 while (processedBytes < containerLength) {
591 size_t bulkTransferSize = 0;
592
593 // prepare header.
594 const bool headerSent = processedBytes != 0;
595 if (!headerSent) {
596 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, containerLength);
597 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
598 bulkTransferSize += MTP_CONTAINER_HEADER_SIZE;
599 }
600
601 // Prepare payload.
602 if (headerSent || divisionMode == FIRST_PACKET_HAS_PAYLOAD) {
603 const size_t processedPayloadBytes =
604 headerSent ? processedBytes - MTP_CONTAINER_HEADER_SIZE : 0;
605 const size_t maxRead = payloadSize - processedPayloadBytes;
606 const size_t maxWrite = maxBulkTransferSize - bulkTransferSize;
607 const size_t bulkTransferPayloadSize = std::min(maxRead, maxWrite);
608 // prepare payload.
609 if (!readError) {
610 const ssize_t result = readExactBytes(
611 fd,
612 mBuffer + bulkTransferSize,
613 bulkTransferPayloadSize);
614 if (result < 0) {
615 ALOGE("Found an error while reading data from FD. Send 0 data instead.");
616 readError = true;
617 }
618 }
619 if (readError) {
620 memset(mBuffer + bulkTransferSize, 0, bulkTransferPayloadSize);
621 }
622 bulkTransferSize += bulkTransferPayloadSize;
623 }
624
625 // Bulk transfer.
626 mPacketSize = bulkTransferSize;
627 request->buffer_length = bulkTransferSize;
628 const int result = transfer(request);
629 if (result != static_cast<ssize_t>(bulkTransferSize)) {
630 // Cannot recover writing error.
631 ALOGE("Found an error while write data to MtpDevice.");
632 return -1;
633 }
634
635 // Update variables.
636 processedBytes += bulkTransferSize;
637 }
638
639 return readError ? -1 : processedBytes;
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400640}
641
Mike Lockwood16864ba2010-05-11 17:16:59 -0400642#endif // MTP_HOST
Mike Lockwood7850ef92010-05-14 10:10:36 -0400643
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900644void* MtpDataPacket::getData(int* outLength) const {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400645 int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE;
646 if (length > 0) {
647 void* result = malloc(length);
648 if (result) {
649 memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length);
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900650 *outLength = length;
Mike Lockwood3e072b32010-06-10 16:34:20 -0400651 return result;
652 }
653 }
Daichi Hirono4fd9a8b2015-08-20 15:13:40 +0900654 *outLength = 0;
Mike Lockwood3e072b32010-06-10 16:34:20 -0400655 return NULL;
656}
657
Mike Lockwood7850ef92010-05-14 10:10:36 -0400658} // namespace android