blob: b4c47af88e7eac3e138fd50c5bf01aa763fb4b22 [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 Lockwood5ed68d22010-05-25 19:08:48 -040017#define LOG_TAG "MtpClient"
Mike Lockwoodb14e5882010-06-29 18:11:52 -040018
19#include "MtpDebug.h"
20#include "MtpClient.h"
21#include "MtpDevice.h"
Mike Lockwood5ed68d22010-05-25 19:08:48 -040022
Mike Lockwood16864ba2010-05-11 17:16:59 -040023#include <stdio.h>
24#include <stdlib.h>
25#include <sys/types.h>
26#include <sys/ioctl.h>
27#include <sys/stat.h>
28#include <fcntl.h>
29#include <errno.h>
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040030#include <utils/threads.h>
Mike Lockwood16864ba2010-05-11 17:16:59 -040031
32#include <usbhost/usbhost.h>
Mike Lockwood5ed68d22010-05-25 19:08:48 -040033#include <linux/version.h>
34#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 20)
35#include <linux/usb/ch9.h>
36#else
37#include <linux/usb_ch9.h>
38#endif
Mike Lockwood16864ba2010-05-11 17:16:59 -040039
Mike Lockwood7850ef92010-05-14 10:10:36 -040040namespace android {
41
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040042class MtpClientThread : public Thread {
43private:
44 MtpClient* mClient;
45
46public:
47 MtpClientThread(MtpClient* client)
48 : mClient(client)
49 {
50 }
51
52 virtual bool threadLoop() {
53 return mClient->threadLoop();
54 }
55};
56
57
Mike Lockwood5ed68d22010-05-25 19:08:48 -040058MtpClient::MtpClient()
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040059 : mThread(NULL),
60 mUsbHostContext(NULL),
61 mDone(false)
Mike Lockwood16864ba2010-05-11 17:16:59 -040062{
Mike Lockwood16864ba2010-05-11 17:16:59 -040063}
64
65MtpClient::~MtpClient() {
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040066 usb_host_cleanup(mUsbHostContext);
Mike Lockwood16864ba2010-05-11 17:16:59 -040067}
68
Mike Lockwood5ed68d22010-05-25 19:08:48 -040069bool MtpClient::start() {
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040070 if (mThread)
Mike Lockwood16864ba2010-05-11 17:16:59 -040071 return true;
Mike Lockwood5ed68d22010-05-25 19:08:48 -040072
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040073 mUsbHostContext = usb_host_init();
74 if (!mUsbHostContext)
Mike Lockwood16864ba2010-05-11 17:16:59 -040075 return false;
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040076
77 mThread = new MtpClientThread(this);
78 mThread->run("MtpClientThread");
79
Mike Lockwood5ed68d22010-05-25 19:08:48 -040080 return true;
81}
82
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040083void MtpClient::stop() {
84 mDone = true;
85}
86
87bool MtpClient::usbDeviceAdded(const char *devname) {
Mike Lockwood5ed68d22010-05-25 19:08:48 -040088 struct usb_descriptor_header* desc;
89 struct usb_descriptor_iter iter;
90
91 struct usb_device *device = usb_device_open(devname);
92 if (!device) {
93 LOGE("usb_device_open failed\n");
Mike Lockwoodcff0ef92010-07-01 11:32:08 -040094 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -040095 }
96
97 usb_descriptor_iter_init(device, &iter);
98
99 while ((desc = usb_descriptor_iter_next(&iter)) != NULL) {
100 if (desc->bDescriptorType == USB_DT_INTERFACE) {
101 struct usb_interface_descriptor *interface = (struct usb_interface_descriptor *)desc;
102
103 if (interface->bInterfaceClass == USB_CLASS_STILL_IMAGE &&
104 interface->bInterfaceSubClass == 1 && // Still Image Capture
105 interface->bInterfaceProtocol == 1) // Picture Transfer Protocol (PIMA 15470)
106 {
107 LOGD("Found camera: \"%s\" \"%s\"\n", usb_device_get_manufacturer_name(device),
108 usb_device_get_product_name(device));
109
110 // interface should be followed by three endpoints
111 struct usb_endpoint_descriptor *ep;
112 struct usb_endpoint_descriptor *ep_in_desc = NULL;
113 struct usb_endpoint_descriptor *ep_out_desc = NULL;
114 struct usb_endpoint_descriptor *ep_intr_desc = NULL;
115 for (int i = 0; i < 3; i++) {
116 ep = (struct usb_endpoint_descriptor *)usb_descriptor_iter_next(&iter);
117 if (!ep || ep->bDescriptorType != USB_DT_ENDPOINT) {
118 LOGE("endpoints not found\n");
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400119 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400120 }
121 if (ep->bmAttributes == USB_ENDPOINT_XFER_BULK) {
122 if (ep->bEndpointAddress & USB_ENDPOINT_DIR_MASK)
123 ep_in_desc = ep;
124 else
125 ep_out_desc = ep;
126 } else if (ep->bmAttributes == USB_ENDPOINT_XFER_INT &&
127 ep->bEndpointAddress & USB_ENDPOINT_DIR_MASK) {
128 ep_intr_desc = ep;
129 }
130 }
131 if (!ep_in_desc || !ep_out_desc || !ep_intr_desc) {
132 LOGE("endpoints not found\n");
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400133 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400134 }
135
136 struct usb_endpoint *ep_in = usb_endpoint_open(device, ep_in_desc);
137 struct usb_endpoint *ep_out = usb_endpoint_open(device, ep_out_desc);
138 struct usb_endpoint *ep_intr = usb_endpoint_open(device, ep_intr_desc);
139
140 if (usb_device_claim_interface(device, interface->bInterfaceNumber)) {
141 LOGE("usb_device_claim_interface failed\n");
142 usb_endpoint_close(ep_in);
143 usb_endpoint_close(ep_out);
144 usb_endpoint_close(ep_intr);
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400145 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400146 }
147
148 MtpDevice* mtpDevice = new MtpDevice(device, interface->bInterfaceNumber,
149 ep_in, ep_out, ep_intr);
150 mDeviceList.add(mtpDevice);
151 mtpDevice->initialize();
152 deviceAdded(mtpDevice);
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400153 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400154 }
155 }
156 }
157
158 usb_device_close(device);
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400159 return mDone;
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400160}
161
162MtpDevice* MtpClient::getDevice(int id) {
163 for (int i = 0; i < mDeviceList.size(); i++) {
164 MtpDevice* device = mDeviceList[i];
165 if (device->getID() == id)
166 return device;
167 }
168 return NULL;
169}
170
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400171bool MtpClient::usbDeviceRemoved(const char *devname) {
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400172 for (int i = 0; i < mDeviceList.size(); i++) {
173 MtpDevice* device = mDeviceList[i];
174 if (!strcmp(devname, device->getDeviceName())) {
175 deviceRemoved(device);
176 mDeviceList.removeAt(i);
177 delete device;
178 LOGD("Camera removed!\n");
179 break;
180 }
181 }
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400182 return mDone;
Mike Lockwood16864ba2010-05-11 17:16:59 -0400183}
184
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400185bool MtpClient::threadLoop() {
186 usb_host_run(mUsbHostContext, usb_device_added, usb_device_removed, this);
187 return false;
188}
189
190int MtpClient::usb_device_added(const char *devname, void* client_data) {
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400191 LOGD("usb_device_added %s\n", devname);
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400192 return ((MtpClient *)client_data)->usbDeviceAdded(devname);
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400193}
194
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400195int MtpClient::usb_device_removed(const char *devname, void* client_data) {
Mike Lockwood5ed68d22010-05-25 19:08:48 -0400196 LOGD("usb_device_removed %s\n", devname);
Mike Lockwoodcff0ef92010-07-01 11:32:08 -0400197 return ((MtpClient *)client_data)->usbDeviceRemoved(devname);
Mike Lockwood16864ba2010-05-11 17:16:59 -0400198}
199
Mike Lockwood7850ef92010-05-14 10:10:36 -0400200} // namespace android