blob: 27dc79663079e81fece4b2ecabb4ec2c004ca32f [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 Lockwood7850ef92010-05-14 10:10:36 -040028namespace android {
29
Mike Lockwood16864ba2010-05-11 17:16:59 -040030MtpDataPacket::MtpDataPacket()
31 : MtpPacket(512),
32 mOffset(MTP_CONTAINER_HEADER_SIZE)
33{
34}
35
36MtpDataPacket::~MtpDataPacket() {
37}
38
39void MtpDataPacket::reset() {
40 MtpPacket::reset();
41 mOffset = MTP_CONTAINER_HEADER_SIZE;
42}
43
44void MtpDataPacket::setOperationCode(MtpOperationCode code) {
45 MtpPacket::putUInt16(MTP_CONTAINER_CODE_OFFSET, code);
46}
47
48void MtpDataPacket::setTransactionID(MtpTransactionID id) {
49 MtpPacket::putUInt32(MTP_CONTAINER_TRANSACTION_ID_OFFSET, id);
50}
51
52uint16_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
59uint32_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
67uint64_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
Mike Lockwooda6c490b2010-06-05 22:45:01 -040077void MtpDataPacket::getUInt128(uint128_t& value) {
78 value[0] = getUInt32();
79 value[1] = getUInt32();
80 value[2] = getUInt32();
81 value[3] = getUInt32();
82}
83
Mike Lockwood16864ba2010-05-11 17:16:59 -040084void MtpDataPacket::getString(MtpStringBuffer& string)
85{
86 string.readFromPacket(this);
87}
88
Mike Lockwood335dd2b2010-05-19 10:33:39 -040089Int8List* 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
97UInt8List* 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
105Int16List* 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
113UInt16List* 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
121Int32List* 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
129UInt32List* 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
137Int64List* 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
145UInt64List* 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
Mike Lockwood16864ba2010-05-11 17:16:59 -0400153void MtpDataPacket::putInt8(int8_t value) {
154 allocate(mOffset + 1);
155 mBuffer[mOffset++] = (uint8_t)value;
156 if (mPacketSize < mOffset)
157 mPacketSize = mOffset;
158}
159
160void MtpDataPacket::putUInt8(uint8_t value) {
161 allocate(mOffset + 1);
162 mBuffer[mOffset++] = (uint8_t)value;
163 if (mPacketSize < mOffset)
164 mPacketSize = mOffset;
165}
166
167void 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
175void 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
183void 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
193void 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
203void 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
217void 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
Mike Lockwooda6c490b2010-06-05 22:45:01 -0400231void MtpDataPacket::putInt128(const int128_t& value) {
232 putInt32(value[0]);
233 putInt32(value[1]);
234 putInt32(value[2]);
235 putInt32(value[3]);
236}
237
238void MtpDataPacket::putUInt128(const uint128_t& value) {
239 putUInt32(value[0]);
240 putUInt32(value[1]);
241 putUInt32(value[2]);
242 putUInt32(value[3]);
243}
244
Mike Lockwood16864ba2010-05-11 17:16:59 -0400245void MtpDataPacket::putAInt8(const int8_t* values, int count) {
246 putUInt32(count);
247 for (int i = 0; i < count; i++)
248 putInt8(*values++);
249}
250
251void MtpDataPacket::putAUInt8(const uint8_t* values, int count) {
252 putUInt32(count);
253 for (int i = 0; i < count; i++)
254 putUInt8(*values++);
255}
256
257void MtpDataPacket::putAInt16(const int16_t* values, int count) {
258 putUInt32(count);
259 for (int i = 0; i < count; i++)
260 putInt16(*values++);
261}
262
263void MtpDataPacket::putAUInt16(const uint16_t* values, int count) {
264 putUInt32(count);
265 for (int i = 0; i < count; i++)
266 putUInt16(*values++);
267}
268
Mike Lockwood782aef12010-08-10 07:37:50 -0400269void MtpDataPacket::putAUInt16(const UInt16List* values) {
270 size_t count = (values ? values->size() : 0);
271 putUInt32(count);
272 for (size_t i = 0; i < count; i++)
273 putUInt16((*values)[i]);
274}
275
Mike Lockwood16864ba2010-05-11 17:16:59 -0400276void MtpDataPacket::putAInt32(const int32_t* values, int count) {
277 putUInt32(count);
278 for (int i = 0; i < count; i++)
279 putInt32(*values++);
280}
281
282void MtpDataPacket::putAUInt32(const uint32_t* values, int count) {
283 putUInt32(count);
284 for (int i = 0; i < count; i++)
285 putUInt32(*values++);
286}
287
288void MtpDataPacket::putAUInt32(const UInt32List* list) {
289 if (!list) {
290 putEmptyArray();
291 } else {
292 size_t size = list->size();
293 putUInt32(size);
294 for (size_t i = 0; i < size; i++)
295 putUInt32((*list)[i]);
296 }
297}
298
299void MtpDataPacket::putAInt64(const int64_t* values, int count) {
300 putUInt32(count);
301 for (int i = 0; i < count; i++)
302 putInt64(*values++);
303}
304
305void MtpDataPacket::putAUInt64(const uint64_t* values, int count) {
306 putUInt32(count);
307 for (int i = 0; i < count; i++)
308 putUInt64(*values++);
309}
310
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400311void MtpDataPacket::putString(const MtpStringBuffer& string) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400312 string.writeToPacket(this);
313}
314
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400315void MtpDataPacket::putString(const char* s) {
Mike Lockwood16864ba2010-05-11 17:16:59 -0400316 MtpStringBuffer string(s);
317 string.writeToPacket(this);
318}
319
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400320void MtpDataPacket::putString(const uint16_t* string) {
321 int count = 0;
322 for (int i = 0; i < 256; i++) {
323 if (string[i])
324 count++;
325 else
326 break;
327 }
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400328 putUInt8(count > 0 ? count + 1 : 0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400329 for (int i = 0; i < count; i++)
330 putUInt16(string[i]);
Mike Lockwoodde1e37a2010-08-18 12:31:09 -0400331 // only terminate with zero if string is not empty
332 if (count > 0)
333 putUInt16(0);
Mike Lockwood1865a5d2010-07-03 00:44:05 -0400334}
335
Mike Lockwood16864ba2010-05-11 17:16:59 -0400336#ifdef MTP_DEVICE
337int MtpDataPacket::read(int fd) {
338 // first read the header
339 int ret = ::read(fd, mBuffer, MTP_CONTAINER_HEADER_SIZE);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400340 if (ret != MTP_CONTAINER_HEADER_SIZE)
341 return -1;
342 // then the following data
343 int total = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET);
344 int remaining = total - MTP_CONTAINER_HEADER_SIZE;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400345 ret = ::read(fd, &mBuffer[0] + MTP_CONTAINER_HEADER_SIZE, remaining);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400346 if (ret != remaining)
347 return -1;
348
349 mPacketSize = total;
350 mOffset = MTP_CONTAINER_HEADER_SIZE;
351 return total;
352}
353
354int MtpDataPacket::readDataHeader(int fd) {
355 int ret = ::read(fd, mBuffer, MTP_CONTAINER_HEADER_SIZE);
356 if (ret > 0)
357 mPacketSize = ret;
358 else
359 mPacketSize = 0;
360 return ret;
361}
362
363int MtpDataPacket::write(int fd) {
364 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
365 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
366
367 // send header separately from data
368 int ret = ::write(fd, mBuffer, MTP_CONTAINER_HEADER_SIZE);
369 if (ret == MTP_CONTAINER_HEADER_SIZE)
370 ret = ::write(fd, mBuffer + MTP_CONTAINER_HEADER_SIZE,
371 mPacketSize - MTP_CONTAINER_HEADER_SIZE);
372 return (ret < 0 ? ret : 0);
373}
374
375int MtpDataPacket::writeDataHeader(int fd, uint32_t length) {
376 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
377 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
378 int ret = ::write(fd, mBuffer, MTP_CONTAINER_HEADER_SIZE);
379 return (ret < 0 ? ret : 0);
380}
381#endif // MTP_DEVICE
382
383#ifdef MTP_HOST
384int MtpDataPacket::read(struct usb_endpoint *ep) {
385 // first read the header
Mike Lockwood3e072b32010-06-10 16:34:20 -0400386 int length = transfer(ep, mBuffer, mBufferSize);
Mike Lockwood437e9452010-07-20 12:01:36 -0400387 if (length >= MTP_CONTAINER_HEADER_SIZE) {
Mike Lockwood3e072b32010-06-10 16:34:20 -0400388 // look at the length field to see if the data spans multiple packets
389 uint32_t totalLength = MtpPacket::getUInt32(MTP_CONTAINER_LENGTH_OFFSET);
390 while (totalLength > length) {
391 allocate(length + mAllocationIncrement);
392 int ret = transfer(ep, mBuffer + length, mAllocationIncrement);
393 if (ret >= 0)
394 length += ret;
395 else {
396 length = ret;
397 break;
398 }
399 }
400 }
401 if (length >= 0)
402 mPacketSize = length;
403 return length;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400404}
405
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400406int MtpDataPacket::readData(struct usb_endpoint *ep, void* buffer, int length) {
407 int packetSize = usb_endpoint_max_packet(ep);
408 int read = 0;
409 while (read < length) {
410 int ret = transfer(ep, (char *)buffer + read, packetSize);
411 if (ret < 0) {
412printf("MtpDataPacket::readData returning %d\n", ret);
413 return ret;
414 }
415 read += ret;
416 }
417printf("MtpDataPacket::readData returning %d\n", read);
418 return read;
419}
420
421int MtpDataPacket::readDataHeader(struct usb_endpoint *ep) {
422 int length = transfer(ep, mBuffer, usb_endpoint_max_packet(ep));
423 if (length >= 0)
424 mPacketSize = length;
425 return length;
426}
427
428int MtpDataPacket::writeDataHeader(struct usb_endpoint *ep, uint32_t length) {
429 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, length);
430 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
431 int ret = transfer(ep, mBuffer, MTP_CONTAINER_HEADER_SIZE);
432 return (ret < 0 ? ret : 0);
433}
434
Mike Lockwood16864ba2010-05-11 17:16:59 -0400435int MtpDataPacket::write(struct usb_endpoint *ep) {
436 MtpPacket::putUInt32(MTP_CONTAINER_LENGTH_OFFSET, mPacketSize);
437 MtpPacket::putUInt16(MTP_CONTAINER_TYPE_OFFSET, MTP_CONTAINER_TYPE_DATA);
438
439 // send header separately from data
440 int ret = transfer(ep, mBuffer, MTP_CONTAINER_HEADER_SIZE);
441 if (ret == MTP_CONTAINER_HEADER_SIZE)
442 ret = transfer(ep, mBuffer + MTP_CONTAINER_HEADER_SIZE,
443 mPacketSize - MTP_CONTAINER_HEADER_SIZE);
444 return (ret < 0 ? ret : 0);
445}
446
Mike Lockwood0cf89f22010-07-26 20:40:45 -0400447int MtpDataPacket::write(struct usb_endpoint *ep, void* buffer, uint32_t length) {
448 int ret = 0;
449 int packetSize = usb_endpoint_max_packet(ep);
450 while (length > 0) {
451 int write = (length > packetSize ? packetSize : length);
452 int ret = transfer(ep, buffer, write);
453 if (ret < 0)
454 break;
455 length -= ret;
456 }
457 return (ret < 0 ? ret : 0);
458}
459
Mike Lockwood16864ba2010-05-11 17:16:59 -0400460#endif // MTP_HOST
Mike Lockwood7850ef92010-05-14 10:10:36 -0400461
Mike Lockwood3e072b32010-06-10 16:34:20 -0400462void* MtpDataPacket::getData(int& outLength) const {
463 int length = mPacketSize - MTP_CONTAINER_HEADER_SIZE;
464 if (length > 0) {
465 void* result = malloc(length);
466 if (result) {
467 memcpy(result, mBuffer + MTP_CONTAINER_HEADER_SIZE, length);
468 outLength = length;
469 return result;
470 }
471 }
472 outLength = 0;
473 return NULL;
474}
475
Mike Lockwood7850ef92010-05-14 10:10:36 -0400476} // namespace android