1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /****************************************************************************
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Filename: cpia2_usb.c
5*4882a593Smuzhiyun *
6*4882a593Smuzhiyun * Copyright 2001, STMicrolectronics, Inc.
7*4882a593Smuzhiyun * Contact: steve.miller@st.com
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun * Description:
10*4882a593Smuzhiyun * This is a USB driver for CPia2 based video cameras.
11*4882a593Smuzhiyun * The infrastructure of this driver is based on the cpia usb driver by
12*4882a593Smuzhiyun * Jochen Scharrlach and Johannes Erdfeldt.
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * Stripped of 2.4 stuff ready for main kernel submit by
15*4882a593Smuzhiyun * Alan Cox <alan@lxorguk.ukuu.org.uk>
16*4882a593Smuzhiyun ****************************************************************************/
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #include <linux/kernel.h>
19*4882a593Smuzhiyun #include <linux/slab.h>
20*4882a593Smuzhiyun #include <linux/usb.h>
21*4882a593Smuzhiyun #include <linux/module.h>
22*4882a593Smuzhiyun
23*4882a593Smuzhiyun #include "cpia2.h"
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun static int frame_sizes[] = {
26*4882a593Smuzhiyun 0, // USBIF_CMDONLY
27*4882a593Smuzhiyun 0, // USBIF_BULK
28*4882a593Smuzhiyun 128, // USBIF_ISO_1
29*4882a593Smuzhiyun 384, // USBIF_ISO_2
30*4882a593Smuzhiyun 640, // USBIF_ISO_3
31*4882a593Smuzhiyun 768, // USBIF_ISO_4
32*4882a593Smuzhiyun 896, // USBIF_ISO_5
33*4882a593Smuzhiyun 1023, // USBIF_ISO_6
34*4882a593Smuzhiyun };
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun #define FRAMES_PER_DESC 10
37*4882a593Smuzhiyun #define FRAME_SIZE_PER_DESC frame_sizes[cam->cur_alt]
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun static void process_frame(struct camera_data *cam);
40*4882a593Smuzhiyun static void cpia2_usb_complete(struct urb *urb);
41*4882a593Smuzhiyun static int cpia2_usb_probe(struct usb_interface *intf,
42*4882a593Smuzhiyun const struct usb_device_id *id);
43*4882a593Smuzhiyun static void cpia2_usb_disconnect(struct usb_interface *intf);
44*4882a593Smuzhiyun static int cpia2_usb_suspend(struct usb_interface *intf, pm_message_t message);
45*4882a593Smuzhiyun static int cpia2_usb_resume(struct usb_interface *intf);
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun static void free_sbufs(struct camera_data *cam);
48*4882a593Smuzhiyun static void add_APPn(struct camera_data *cam);
49*4882a593Smuzhiyun static void add_COM(struct camera_data *cam);
50*4882a593Smuzhiyun static int submit_urbs(struct camera_data *cam);
51*4882a593Smuzhiyun static int set_alternate(struct camera_data *cam, unsigned int alt);
52*4882a593Smuzhiyun static int configure_transfer_mode(struct camera_data *cam, unsigned int alt);
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun static const struct usb_device_id cpia2_id_table[] = {
55*4882a593Smuzhiyun {USB_DEVICE(0x0553, 0x0100)},
56*4882a593Smuzhiyun {USB_DEVICE(0x0553, 0x0140)},
57*4882a593Smuzhiyun {USB_DEVICE(0x0553, 0x0151)}, /* STV0676 */
58*4882a593Smuzhiyun {} /* Terminating entry */
59*4882a593Smuzhiyun };
60*4882a593Smuzhiyun MODULE_DEVICE_TABLE(usb, cpia2_id_table);
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun static struct usb_driver cpia2_driver = {
63*4882a593Smuzhiyun .name = "cpia2",
64*4882a593Smuzhiyun .probe = cpia2_usb_probe,
65*4882a593Smuzhiyun .disconnect = cpia2_usb_disconnect,
66*4882a593Smuzhiyun .suspend = cpia2_usb_suspend,
67*4882a593Smuzhiyun .resume = cpia2_usb_resume,
68*4882a593Smuzhiyun .reset_resume = cpia2_usb_resume,
69*4882a593Smuzhiyun .id_table = cpia2_id_table
70*4882a593Smuzhiyun };
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun /******************************************************************************
74*4882a593Smuzhiyun *
75*4882a593Smuzhiyun * process_frame
76*4882a593Smuzhiyun *
77*4882a593Smuzhiyun *****************************************************************************/
process_frame(struct camera_data * cam)78*4882a593Smuzhiyun static void process_frame(struct camera_data *cam)
79*4882a593Smuzhiyun {
80*4882a593Smuzhiyun static int frame_count;
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun unsigned char *inbuff = cam->workbuff->data;
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun DBG("Processing frame #%d, current:%d\n",
85*4882a593Smuzhiyun cam->workbuff->num, cam->curbuff->num);
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun if(cam->workbuff->length > cam->workbuff->max_length)
88*4882a593Smuzhiyun cam->workbuff->max_length = cam->workbuff->length;
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun if ((inbuff[0] == 0xFF) && (inbuff[1] == 0xD8)) {
91*4882a593Smuzhiyun frame_count++;
92*4882a593Smuzhiyun } else {
93*4882a593Smuzhiyun cam->workbuff->status = FRAME_ERROR;
94*4882a593Smuzhiyun DBG("Start of frame not found\n");
95*4882a593Smuzhiyun return;
96*4882a593Smuzhiyun }
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun /***
99*4882a593Smuzhiyun * Now the output buffer should have a JPEG image in it.
100*4882a593Smuzhiyun ***/
101*4882a593Smuzhiyun if(!cam->first_image_seen) {
102*4882a593Smuzhiyun /* Always skip the first image after streaming
103*4882a593Smuzhiyun * starts. It is almost certainly corrupt. */
104*4882a593Smuzhiyun cam->first_image_seen = 1;
105*4882a593Smuzhiyun cam->workbuff->status = FRAME_EMPTY;
106*4882a593Smuzhiyun return;
107*4882a593Smuzhiyun }
108*4882a593Smuzhiyun if (cam->workbuff->length > 3) {
109*4882a593Smuzhiyun if(cam->mmapped &&
110*4882a593Smuzhiyun cam->workbuff->length < cam->workbuff->max_length) {
111*4882a593Smuzhiyun /* No junk in the buffers */
112*4882a593Smuzhiyun memset(cam->workbuff->data+cam->workbuff->length,
113*4882a593Smuzhiyun 0, cam->workbuff->max_length-
114*4882a593Smuzhiyun cam->workbuff->length);
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun cam->workbuff->max_length = cam->workbuff->length;
117*4882a593Smuzhiyun cam->workbuff->status = FRAME_READY;
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun if(!cam->mmapped && cam->num_frames > 2) {
120*4882a593Smuzhiyun /* During normal reading, the most recent
121*4882a593Smuzhiyun * frame will be read. If the current frame
122*4882a593Smuzhiyun * hasn't started reading yet, it will never
123*4882a593Smuzhiyun * be read, so mark it empty. If the buffer is
124*4882a593Smuzhiyun * mmapped, or we have few buffers, we need to
125*4882a593Smuzhiyun * wait for the user to free the buffer.
126*4882a593Smuzhiyun *
127*4882a593Smuzhiyun * NOTE: This is not entirely foolproof with 3
128*4882a593Smuzhiyun * buffers, but it would take an EXTREMELY
129*4882a593Smuzhiyun * overloaded system to cause problems (possible
130*4882a593Smuzhiyun * image data corruption). Basically, it would
131*4882a593Smuzhiyun * need to take more time to execute cpia2_read
132*4882a593Smuzhiyun * than it would for the camera to send
133*4882a593Smuzhiyun * cam->num_frames-2 frames before problems
134*4882a593Smuzhiyun * could occur.
135*4882a593Smuzhiyun */
136*4882a593Smuzhiyun cam->curbuff->status = FRAME_EMPTY;
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun cam->curbuff = cam->workbuff;
139*4882a593Smuzhiyun cam->workbuff = cam->workbuff->next;
140*4882a593Smuzhiyun DBG("Changed buffers, work:%d, current:%d\n",
141*4882a593Smuzhiyun cam->workbuff->num, cam->curbuff->num);
142*4882a593Smuzhiyun return;
143*4882a593Smuzhiyun } else {
144*4882a593Smuzhiyun DBG("Not enough data for an image.\n");
145*4882a593Smuzhiyun }
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun cam->workbuff->status = FRAME_ERROR;
148*4882a593Smuzhiyun return;
149*4882a593Smuzhiyun }
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun /******************************************************************************
152*4882a593Smuzhiyun *
153*4882a593Smuzhiyun * add_APPn
154*4882a593Smuzhiyun *
155*4882a593Smuzhiyun * Adds a user specified APPn record
156*4882a593Smuzhiyun *****************************************************************************/
add_APPn(struct camera_data * cam)157*4882a593Smuzhiyun static void add_APPn(struct camera_data *cam)
158*4882a593Smuzhiyun {
159*4882a593Smuzhiyun if(cam->APP_len > 0) {
160*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0xFF;
161*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0xE0+cam->APPn;
162*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0;
163*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = cam->APP_len+2;
164*4882a593Smuzhiyun memcpy(cam->workbuff->data+cam->workbuff->length,
165*4882a593Smuzhiyun cam->APP_data, cam->APP_len);
166*4882a593Smuzhiyun cam->workbuff->length += cam->APP_len;
167*4882a593Smuzhiyun }
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun /******************************************************************************
171*4882a593Smuzhiyun *
172*4882a593Smuzhiyun * add_COM
173*4882a593Smuzhiyun *
174*4882a593Smuzhiyun * Adds a user specified COM record
175*4882a593Smuzhiyun *****************************************************************************/
add_COM(struct camera_data * cam)176*4882a593Smuzhiyun static void add_COM(struct camera_data *cam)
177*4882a593Smuzhiyun {
178*4882a593Smuzhiyun if(cam->COM_len > 0) {
179*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0xFF;
180*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0xFE;
181*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = 0;
182*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length++] = cam->COM_len+2;
183*4882a593Smuzhiyun memcpy(cam->workbuff->data+cam->workbuff->length,
184*4882a593Smuzhiyun cam->COM_data, cam->COM_len);
185*4882a593Smuzhiyun cam->workbuff->length += cam->COM_len;
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun /******************************************************************************
190*4882a593Smuzhiyun *
191*4882a593Smuzhiyun * cpia2_usb_complete
192*4882a593Smuzhiyun *
193*4882a593Smuzhiyun * callback when incoming packet is received
194*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_complete(struct urb * urb)195*4882a593Smuzhiyun static void cpia2_usb_complete(struct urb *urb)
196*4882a593Smuzhiyun {
197*4882a593Smuzhiyun int i;
198*4882a593Smuzhiyun unsigned char *cdata;
199*4882a593Smuzhiyun static bool frame_ready = false;
200*4882a593Smuzhiyun struct camera_data *cam = (struct camera_data *) urb->context;
201*4882a593Smuzhiyun
202*4882a593Smuzhiyun if (urb->status!=0) {
203*4882a593Smuzhiyun if (!(urb->status == -ENOENT ||
204*4882a593Smuzhiyun urb->status == -ECONNRESET ||
205*4882a593Smuzhiyun urb->status == -ESHUTDOWN))
206*4882a593Smuzhiyun {
207*4882a593Smuzhiyun DBG("urb->status = %d!\n", urb->status);
208*4882a593Smuzhiyun }
209*4882a593Smuzhiyun DBG("Stopping streaming\n");
210*4882a593Smuzhiyun return;
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun if (!cam->streaming || !video_is_registered(&cam->vdev)) {
214*4882a593Smuzhiyun LOG("Will now stop the streaming: streaming = %d, present=%d\n",
215*4882a593Smuzhiyun cam->streaming, video_is_registered(&cam->vdev));
216*4882a593Smuzhiyun return;
217*4882a593Smuzhiyun }
218*4882a593Smuzhiyun
219*4882a593Smuzhiyun /***
220*4882a593Smuzhiyun * Packet collater
221*4882a593Smuzhiyun ***/
222*4882a593Smuzhiyun //DBG("Collating %d packets\n", urb->number_of_packets);
223*4882a593Smuzhiyun for (i = 0; i < urb->number_of_packets; i++) {
224*4882a593Smuzhiyun u16 checksum, iso_checksum;
225*4882a593Smuzhiyun int j;
226*4882a593Smuzhiyun int n = urb->iso_frame_desc[i].actual_length;
227*4882a593Smuzhiyun int st = urb->iso_frame_desc[i].status;
228*4882a593Smuzhiyun
229*4882a593Smuzhiyun if(cam->workbuff->status == FRAME_READY) {
230*4882a593Smuzhiyun struct framebuf *ptr;
231*4882a593Smuzhiyun /* Try to find an available buffer */
232*4882a593Smuzhiyun DBG("workbuff full, searching\n");
233*4882a593Smuzhiyun for (ptr = cam->workbuff->next;
234*4882a593Smuzhiyun ptr != cam->workbuff;
235*4882a593Smuzhiyun ptr = ptr->next)
236*4882a593Smuzhiyun {
237*4882a593Smuzhiyun if (ptr->status == FRAME_EMPTY) {
238*4882a593Smuzhiyun ptr->status = FRAME_READING;
239*4882a593Smuzhiyun ptr->length = 0;
240*4882a593Smuzhiyun break;
241*4882a593Smuzhiyun }
242*4882a593Smuzhiyun }
243*4882a593Smuzhiyun if (ptr == cam->workbuff)
244*4882a593Smuzhiyun break; /* No READING or EMPTY buffers left */
245*4882a593Smuzhiyun
246*4882a593Smuzhiyun cam->workbuff = ptr;
247*4882a593Smuzhiyun }
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun if (cam->workbuff->status == FRAME_EMPTY ||
250*4882a593Smuzhiyun cam->workbuff->status == FRAME_ERROR) {
251*4882a593Smuzhiyun cam->workbuff->status = FRAME_READING;
252*4882a593Smuzhiyun cam->workbuff->length = 0;
253*4882a593Smuzhiyun }
254*4882a593Smuzhiyun
255*4882a593Smuzhiyun //DBG(" Packet %d length = %d, status = %d\n", i, n, st);
256*4882a593Smuzhiyun cdata = urb->transfer_buffer + urb->iso_frame_desc[i].offset;
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun if (st) {
259*4882a593Smuzhiyun LOG("cpia2 data error: [%d] len=%d, status = %d\n",
260*4882a593Smuzhiyun i, n, st);
261*4882a593Smuzhiyun if(!ALLOW_CORRUPT)
262*4882a593Smuzhiyun cam->workbuff->status = FRAME_ERROR;
263*4882a593Smuzhiyun continue;
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun if(n<=2)
267*4882a593Smuzhiyun continue;
268*4882a593Smuzhiyun
269*4882a593Smuzhiyun checksum = 0;
270*4882a593Smuzhiyun for(j=0; j<n-2; ++j)
271*4882a593Smuzhiyun checksum += cdata[j];
272*4882a593Smuzhiyun iso_checksum = cdata[j] + cdata[j+1]*256;
273*4882a593Smuzhiyun if(checksum != iso_checksum) {
274*4882a593Smuzhiyun LOG("checksum mismatch: [%d] len=%d, calculated = %x, checksum = %x\n",
275*4882a593Smuzhiyun i, n, (int)checksum, (int)iso_checksum);
276*4882a593Smuzhiyun if(!ALLOW_CORRUPT) {
277*4882a593Smuzhiyun cam->workbuff->status = FRAME_ERROR;
278*4882a593Smuzhiyun continue;
279*4882a593Smuzhiyun }
280*4882a593Smuzhiyun }
281*4882a593Smuzhiyun n -= 2;
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun if(cam->workbuff->status != FRAME_READING) {
284*4882a593Smuzhiyun if((0xFF == cdata[0] && 0xD8 == cdata[1]) ||
285*4882a593Smuzhiyun (0xD8 == cdata[0] && 0xFF == cdata[1] &&
286*4882a593Smuzhiyun 0 != cdata[2])) {
287*4882a593Smuzhiyun /* frame is skipped, but increment total
288*4882a593Smuzhiyun * frame count anyway */
289*4882a593Smuzhiyun cam->frame_count++;
290*4882a593Smuzhiyun }
291*4882a593Smuzhiyun DBG("workbuff not reading, status=%d\n",
292*4882a593Smuzhiyun cam->workbuff->status);
293*4882a593Smuzhiyun continue;
294*4882a593Smuzhiyun }
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun if (cam->frame_size < cam->workbuff->length + n) {
297*4882a593Smuzhiyun ERR("buffer overflow! length: %d, n: %d\n",
298*4882a593Smuzhiyun cam->workbuff->length, n);
299*4882a593Smuzhiyun cam->workbuff->status = FRAME_ERROR;
300*4882a593Smuzhiyun if(cam->workbuff->length > cam->workbuff->max_length)
301*4882a593Smuzhiyun cam->workbuff->max_length =
302*4882a593Smuzhiyun cam->workbuff->length;
303*4882a593Smuzhiyun continue;
304*4882a593Smuzhiyun }
305*4882a593Smuzhiyun
306*4882a593Smuzhiyun if (cam->workbuff->length == 0) {
307*4882a593Smuzhiyun int data_offset;
308*4882a593Smuzhiyun if ((0xD8 == cdata[0]) && (0xFF == cdata[1])) {
309*4882a593Smuzhiyun data_offset = 1;
310*4882a593Smuzhiyun } else if((0xFF == cdata[0]) && (0xD8 == cdata[1])
311*4882a593Smuzhiyun && (0xFF == cdata[2])) {
312*4882a593Smuzhiyun data_offset = 2;
313*4882a593Smuzhiyun } else {
314*4882a593Smuzhiyun DBG("Ignoring packet, not beginning!\n");
315*4882a593Smuzhiyun continue;
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun DBG("Start of frame pattern found\n");
318*4882a593Smuzhiyun cam->workbuff->ts = ktime_get_ns();
319*4882a593Smuzhiyun cam->workbuff->seq = cam->frame_count++;
320*4882a593Smuzhiyun cam->workbuff->data[0] = 0xFF;
321*4882a593Smuzhiyun cam->workbuff->data[1] = 0xD8;
322*4882a593Smuzhiyun cam->workbuff->length = 2;
323*4882a593Smuzhiyun add_APPn(cam);
324*4882a593Smuzhiyun add_COM(cam);
325*4882a593Smuzhiyun memcpy(cam->workbuff->data+cam->workbuff->length,
326*4882a593Smuzhiyun cdata+data_offset, n-data_offset);
327*4882a593Smuzhiyun cam->workbuff->length += n-data_offset;
328*4882a593Smuzhiyun } else if (cam->workbuff->length > 0) {
329*4882a593Smuzhiyun memcpy(cam->workbuff->data + cam->workbuff->length,
330*4882a593Smuzhiyun cdata, n);
331*4882a593Smuzhiyun cam->workbuff->length += n;
332*4882a593Smuzhiyun }
333*4882a593Smuzhiyun
334*4882a593Smuzhiyun if ((cam->workbuff->length >= 3) &&
335*4882a593Smuzhiyun (cam->workbuff->data[cam->workbuff->length - 3] == 0xFF) &&
336*4882a593Smuzhiyun (cam->workbuff->data[cam->workbuff->length - 2] == 0xD9) &&
337*4882a593Smuzhiyun (cam->workbuff->data[cam->workbuff->length - 1] == 0xFF)) {
338*4882a593Smuzhiyun frame_ready = true;
339*4882a593Smuzhiyun cam->workbuff->data[cam->workbuff->length - 1] = 0;
340*4882a593Smuzhiyun cam->workbuff->length -= 1;
341*4882a593Smuzhiyun } else if ((cam->workbuff->length >= 2) &&
342*4882a593Smuzhiyun (cam->workbuff->data[cam->workbuff->length - 2] == 0xFF) &&
343*4882a593Smuzhiyun (cam->workbuff->data[cam->workbuff->length - 1] == 0xD9)) {
344*4882a593Smuzhiyun frame_ready = true;
345*4882a593Smuzhiyun }
346*4882a593Smuzhiyun
347*4882a593Smuzhiyun if (frame_ready) {
348*4882a593Smuzhiyun DBG("Workbuff image size = %d\n",cam->workbuff->length);
349*4882a593Smuzhiyun process_frame(cam);
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun frame_ready = false;
352*4882a593Smuzhiyun
353*4882a593Smuzhiyun if (waitqueue_active(&cam->wq_stream))
354*4882a593Smuzhiyun wake_up_interruptible(&cam->wq_stream);
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun }
357*4882a593Smuzhiyun
358*4882a593Smuzhiyun if(cam->streaming) {
359*4882a593Smuzhiyun /* resubmit */
360*4882a593Smuzhiyun urb->dev = cam->dev;
361*4882a593Smuzhiyun if ((i = usb_submit_urb(urb, GFP_ATOMIC)) != 0)
362*4882a593Smuzhiyun ERR("%s: usb_submit_urb ret %d!\n", __func__, i);
363*4882a593Smuzhiyun }
364*4882a593Smuzhiyun }
365*4882a593Smuzhiyun
366*4882a593Smuzhiyun /******************************************************************************
367*4882a593Smuzhiyun *
368*4882a593Smuzhiyun * configure_transfer_mode
369*4882a593Smuzhiyun *
370*4882a593Smuzhiyun *****************************************************************************/
configure_transfer_mode(struct camera_data * cam,unsigned int alt)371*4882a593Smuzhiyun static int configure_transfer_mode(struct camera_data *cam, unsigned int alt)
372*4882a593Smuzhiyun {
373*4882a593Smuzhiyun static unsigned char iso_regs[8][4] = {
374*4882a593Smuzhiyun {0x00, 0x00, 0x00, 0x00},
375*4882a593Smuzhiyun {0x00, 0x00, 0x00, 0x00},
376*4882a593Smuzhiyun {0xB9, 0x00, 0x00, 0x7E},
377*4882a593Smuzhiyun {0xB9, 0x00, 0x01, 0x7E},
378*4882a593Smuzhiyun {0xB9, 0x00, 0x02, 0x7E},
379*4882a593Smuzhiyun {0xB9, 0x00, 0x02, 0xFE},
380*4882a593Smuzhiyun {0xB9, 0x00, 0x03, 0x7E},
381*4882a593Smuzhiyun {0xB9, 0x00, 0x03, 0xFD}
382*4882a593Smuzhiyun };
383*4882a593Smuzhiyun struct cpia2_command cmd;
384*4882a593Smuzhiyun unsigned char reg;
385*4882a593Smuzhiyun
386*4882a593Smuzhiyun if (!video_is_registered(&cam->vdev))
387*4882a593Smuzhiyun return -ENODEV;
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun /***
390*4882a593Smuzhiyun * Write the isoc registers according to the alternate selected
391*4882a593Smuzhiyun ***/
392*4882a593Smuzhiyun cmd.direction = TRANSFER_WRITE;
393*4882a593Smuzhiyun cmd.buffer.block_data[0] = iso_regs[alt][0];
394*4882a593Smuzhiyun cmd.buffer.block_data[1] = iso_regs[alt][1];
395*4882a593Smuzhiyun cmd.buffer.block_data[2] = iso_regs[alt][2];
396*4882a593Smuzhiyun cmd.buffer.block_data[3] = iso_regs[alt][3];
397*4882a593Smuzhiyun cmd.req_mode = CAMERAACCESS_TYPE_BLOCK | CAMERAACCESS_VC;
398*4882a593Smuzhiyun cmd.start = CPIA2_VC_USB_ISOLIM;
399*4882a593Smuzhiyun cmd.reg_count = 4;
400*4882a593Smuzhiyun cpia2_send_command(cam, &cmd);
401*4882a593Smuzhiyun
402*4882a593Smuzhiyun /***
403*4882a593Smuzhiyun * Enable relevant streams before starting polling.
404*4882a593Smuzhiyun * First read USB Stream Config Register.
405*4882a593Smuzhiyun ***/
406*4882a593Smuzhiyun cmd.direction = TRANSFER_READ;
407*4882a593Smuzhiyun cmd.req_mode = CAMERAACCESS_TYPE_BLOCK | CAMERAACCESS_VC;
408*4882a593Smuzhiyun cmd.start = CPIA2_VC_USB_STRM;
409*4882a593Smuzhiyun cmd.reg_count = 1;
410*4882a593Smuzhiyun cpia2_send_command(cam, &cmd);
411*4882a593Smuzhiyun reg = cmd.buffer.block_data[0];
412*4882a593Smuzhiyun
413*4882a593Smuzhiyun /* Clear iso, bulk, and int */
414*4882a593Smuzhiyun reg &= ~(CPIA2_VC_USB_STRM_BLK_ENABLE |
415*4882a593Smuzhiyun CPIA2_VC_USB_STRM_ISO_ENABLE |
416*4882a593Smuzhiyun CPIA2_VC_USB_STRM_INT_ENABLE);
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun if (alt == USBIF_BULK) {
419*4882a593Smuzhiyun DBG("Enabling bulk xfer\n");
420*4882a593Smuzhiyun reg |= CPIA2_VC_USB_STRM_BLK_ENABLE; /* Enable Bulk */
421*4882a593Smuzhiyun cam->xfer_mode = XFER_BULK;
422*4882a593Smuzhiyun } else if (alt >= USBIF_ISO_1) {
423*4882a593Smuzhiyun DBG("Enabling ISOC xfer\n");
424*4882a593Smuzhiyun reg |= CPIA2_VC_USB_STRM_ISO_ENABLE;
425*4882a593Smuzhiyun cam->xfer_mode = XFER_ISOC;
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun
428*4882a593Smuzhiyun cmd.buffer.block_data[0] = reg;
429*4882a593Smuzhiyun cmd.direction = TRANSFER_WRITE;
430*4882a593Smuzhiyun cmd.start = CPIA2_VC_USB_STRM;
431*4882a593Smuzhiyun cmd.reg_count = 1;
432*4882a593Smuzhiyun cmd.req_mode = CAMERAACCESS_TYPE_BLOCK | CAMERAACCESS_VC;
433*4882a593Smuzhiyun cpia2_send_command(cam, &cmd);
434*4882a593Smuzhiyun
435*4882a593Smuzhiyun return 0;
436*4882a593Smuzhiyun }
437*4882a593Smuzhiyun
438*4882a593Smuzhiyun /******************************************************************************
439*4882a593Smuzhiyun *
440*4882a593Smuzhiyun * cpia2_usb_change_streaming_alternate
441*4882a593Smuzhiyun *
442*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_change_streaming_alternate(struct camera_data * cam,unsigned int alt)443*4882a593Smuzhiyun int cpia2_usb_change_streaming_alternate(struct camera_data *cam,
444*4882a593Smuzhiyun unsigned int alt)
445*4882a593Smuzhiyun {
446*4882a593Smuzhiyun int ret = 0;
447*4882a593Smuzhiyun
448*4882a593Smuzhiyun if(alt < USBIF_ISO_1 || alt > USBIF_ISO_6)
449*4882a593Smuzhiyun return -EINVAL;
450*4882a593Smuzhiyun
451*4882a593Smuzhiyun if(alt == cam->params.camera_state.stream_mode)
452*4882a593Smuzhiyun return 0;
453*4882a593Smuzhiyun
454*4882a593Smuzhiyun cpia2_usb_stream_pause(cam);
455*4882a593Smuzhiyun
456*4882a593Smuzhiyun configure_transfer_mode(cam, alt);
457*4882a593Smuzhiyun
458*4882a593Smuzhiyun cam->params.camera_state.stream_mode = alt;
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun /* Reset the camera to prevent image quality degradation */
461*4882a593Smuzhiyun cpia2_reset_camera(cam);
462*4882a593Smuzhiyun
463*4882a593Smuzhiyun cpia2_usb_stream_resume(cam);
464*4882a593Smuzhiyun
465*4882a593Smuzhiyun return ret;
466*4882a593Smuzhiyun }
467*4882a593Smuzhiyun
468*4882a593Smuzhiyun /******************************************************************************
469*4882a593Smuzhiyun *
470*4882a593Smuzhiyun * set_alternate
471*4882a593Smuzhiyun *
472*4882a593Smuzhiyun *****************************************************************************/
set_alternate(struct camera_data * cam,unsigned int alt)473*4882a593Smuzhiyun static int set_alternate(struct camera_data *cam, unsigned int alt)
474*4882a593Smuzhiyun {
475*4882a593Smuzhiyun int ret = 0;
476*4882a593Smuzhiyun
477*4882a593Smuzhiyun if(alt == cam->cur_alt)
478*4882a593Smuzhiyun return 0;
479*4882a593Smuzhiyun
480*4882a593Smuzhiyun if (cam->cur_alt != USBIF_CMDONLY) {
481*4882a593Smuzhiyun DBG("Changing from alt %d to %d\n", cam->cur_alt, USBIF_CMDONLY);
482*4882a593Smuzhiyun ret = usb_set_interface(cam->dev, cam->iface, USBIF_CMDONLY);
483*4882a593Smuzhiyun if (ret != 0)
484*4882a593Smuzhiyun return ret;
485*4882a593Smuzhiyun }
486*4882a593Smuzhiyun if (alt != USBIF_CMDONLY) {
487*4882a593Smuzhiyun DBG("Changing from alt %d to %d\n", USBIF_CMDONLY, alt);
488*4882a593Smuzhiyun ret = usb_set_interface(cam->dev, cam->iface, alt);
489*4882a593Smuzhiyun if (ret != 0)
490*4882a593Smuzhiyun return ret;
491*4882a593Smuzhiyun }
492*4882a593Smuzhiyun
493*4882a593Smuzhiyun cam->old_alt = cam->cur_alt;
494*4882a593Smuzhiyun cam->cur_alt = alt;
495*4882a593Smuzhiyun
496*4882a593Smuzhiyun return ret;
497*4882a593Smuzhiyun }
498*4882a593Smuzhiyun
499*4882a593Smuzhiyun /******************************************************************************
500*4882a593Smuzhiyun *
501*4882a593Smuzhiyun * free_sbufs
502*4882a593Smuzhiyun *
503*4882a593Smuzhiyun * Free all cam->sbuf[]. All non-NULL .data and .urb members that are non-NULL
504*4882a593Smuzhiyun * are assumed to be allocated. Non-NULL .urb members are also assumed to be
505*4882a593Smuzhiyun * submitted (and must therefore be killed before they are freed).
506*4882a593Smuzhiyun *****************************************************************************/
free_sbufs(struct camera_data * cam)507*4882a593Smuzhiyun static void free_sbufs(struct camera_data *cam)
508*4882a593Smuzhiyun {
509*4882a593Smuzhiyun int i;
510*4882a593Smuzhiyun
511*4882a593Smuzhiyun for (i = 0; i < NUM_SBUF; i++) {
512*4882a593Smuzhiyun if(cam->sbuf[i].urb) {
513*4882a593Smuzhiyun usb_kill_urb(cam->sbuf[i].urb);
514*4882a593Smuzhiyun usb_free_urb(cam->sbuf[i].urb);
515*4882a593Smuzhiyun cam->sbuf[i].urb = NULL;
516*4882a593Smuzhiyun }
517*4882a593Smuzhiyun if(cam->sbuf[i].data) {
518*4882a593Smuzhiyun kfree(cam->sbuf[i].data);
519*4882a593Smuzhiyun cam->sbuf[i].data = NULL;
520*4882a593Smuzhiyun }
521*4882a593Smuzhiyun }
522*4882a593Smuzhiyun }
523*4882a593Smuzhiyun
524*4882a593Smuzhiyun /*******
525*4882a593Smuzhiyun * Convenience functions
526*4882a593Smuzhiyun *******/
527*4882a593Smuzhiyun /****************************************************************************
528*4882a593Smuzhiyun *
529*4882a593Smuzhiyun * write_packet
530*4882a593Smuzhiyun *
531*4882a593Smuzhiyun ***************************************************************************/
write_packet(struct usb_device * udev,u8 request,u8 * registers,u16 start,size_t size)532*4882a593Smuzhiyun static int write_packet(struct usb_device *udev,
533*4882a593Smuzhiyun u8 request, u8 * registers, u16 start, size_t size)
534*4882a593Smuzhiyun {
535*4882a593Smuzhiyun unsigned char *buf;
536*4882a593Smuzhiyun int ret;
537*4882a593Smuzhiyun
538*4882a593Smuzhiyun if (!registers || size <= 0)
539*4882a593Smuzhiyun return -EINVAL;
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun buf = kmemdup(registers, size, GFP_KERNEL);
542*4882a593Smuzhiyun if (!buf)
543*4882a593Smuzhiyun return -ENOMEM;
544*4882a593Smuzhiyun
545*4882a593Smuzhiyun ret = usb_control_msg(udev,
546*4882a593Smuzhiyun usb_sndctrlpipe(udev, 0),
547*4882a593Smuzhiyun request,
548*4882a593Smuzhiyun USB_TYPE_VENDOR | USB_RECIP_DEVICE,
549*4882a593Smuzhiyun start, /* value */
550*4882a593Smuzhiyun 0, /* index */
551*4882a593Smuzhiyun buf, /* buffer */
552*4882a593Smuzhiyun size,
553*4882a593Smuzhiyun 1000);
554*4882a593Smuzhiyun
555*4882a593Smuzhiyun kfree(buf);
556*4882a593Smuzhiyun return ret;
557*4882a593Smuzhiyun }
558*4882a593Smuzhiyun
559*4882a593Smuzhiyun /****************************************************************************
560*4882a593Smuzhiyun *
561*4882a593Smuzhiyun * read_packet
562*4882a593Smuzhiyun *
563*4882a593Smuzhiyun ***************************************************************************/
read_packet(struct usb_device * udev,u8 request,u8 * registers,u16 start,size_t size)564*4882a593Smuzhiyun static int read_packet(struct usb_device *udev,
565*4882a593Smuzhiyun u8 request, u8 * registers, u16 start, size_t size)
566*4882a593Smuzhiyun {
567*4882a593Smuzhiyun unsigned char *buf;
568*4882a593Smuzhiyun int ret;
569*4882a593Smuzhiyun
570*4882a593Smuzhiyun if (!registers || size <= 0)
571*4882a593Smuzhiyun return -EINVAL;
572*4882a593Smuzhiyun
573*4882a593Smuzhiyun buf = kmalloc(size, GFP_KERNEL);
574*4882a593Smuzhiyun if (!buf)
575*4882a593Smuzhiyun return -ENOMEM;
576*4882a593Smuzhiyun
577*4882a593Smuzhiyun ret = usb_control_msg(udev,
578*4882a593Smuzhiyun usb_rcvctrlpipe(udev, 0),
579*4882a593Smuzhiyun request,
580*4882a593Smuzhiyun USB_DIR_IN|USB_TYPE_VENDOR|USB_RECIP_DEVICE,
581*4882a593Smuzhiyun start, /* value */
582*4882a593Smuzhiyun 0, /* index */
583*4882a593Smuzhiyun buf, /* buffer */
584*4882a593Smuzhiyun size,
585*4882a593Smuzhiyun 1000);
586*4882a593Smuzhiyun
587*4882a593Smuzhiyun if (ret >= 0)
588*4882a593Smuzhiyun memcpy(registers, buf, size);
589*4882a593Smuzhiyun
590*4882a593Smuzhiyun kfree(buf);
591*4882a593Smuzhiyun
592*4882a593Smuzhiyun return ret;
593*4882a593Smuzhiyun }
594*4882a593Smuzhiyun
595*4882a593Smuzhiyun /******************************************************************************
596*4882a593Smuzhiyun *
597*4882a593Smuzhiyun * cpia2_usb_transfer_cmd
598*4882a593Smuzhiyun *
599*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_transfer_cmd(struct camera_data * cam,void * registers,u8 request,u8 start,u8 count,u8 direction)600*4882a593Smuzhiyun int cpia2_usb_transfer_cmd(struct camera_data *cam,
601*4882a593Smuzhiyun void *registers,
602*4882a593Smuzhiyun u8 request, u8 start, u8 count, u8 direction)
603*4882a593Smuzhiyun {
604*4882a593Smuzhiyun int err = 0;
605*4882a593Smuzhiyun struct usb_device *udev = cam->dev;
606*4882a593Smuzhiyun
607*4882a593Smuzhiyun if (!udev) {
608*4882a593Smuzhiyun ERR("%s: Internal driver error: udev is NULL\n", __func__);
609*4882a593Smuzhiyun return -EINVAL;
610*4882a593Smuzhiyun }
611*4882a593Smuzhiyun
612*4882a593Smuzhiyun if (!registers) {
613*4882a593Smuzhiyun ERR("%s: Internal driver error: register array is NULL\n", __func__);
614*4882a593Smuzhiyun return -EINVAL;
615*4882a593Smuzhiyun }
616*4882a593Smuzhiyun
617*4882a593Smuzhiyun if (direction == TRANSFER_READ) {
618*4882a593Smuzhiyun err = read_packet(udev, request, (u8 *)registers, start, count);
619*4882a593Smuzhiyun if (err > 0)
620*4882a593Smuzhiyun err = 0;
621*4882a593Smuzhiyun } else if (direction == TRANSFER_WRITE) {
622*4882a593Smuzhiyun err =write_packet(udev, request, (u8 *)registers, start, count);
623*4882a593Smuzhiyun if (err < 0) {
624*4882a593Smuzhiyun LOG("Control message failed, err val = %d\n", err);
625*4882a593Smuzhiyun LOG("Message: request = 0x%0X, start = 0x%0X\n",
626*4882a593Smuzhiyun request, start);
627*4882a593Smuzhiyun LOG("Message: count = %d, register[0] = 0x%0X\n",
628*4882a593Smuzhiyun count, ((unsigned char *) registers)[0]);
629*4882a593Smuzhiyun } else
630*4882a593Smuzhiyun err=0;
631*4882a593Smuzhiyun } else {
632*4882a593Smuzhiyun LOG("Unexpected first byte of direction: %d\n",
633*4882a593Smuzhiyun direction);
634*4882a593Smuzhiyun return -EINVAL;
635*4882a593Smuzhiyun }
636*4882a593Smuzhiyun
637*4882a593Smuzhiyun if(err != 0)
638*4882a593Smuzhiyun LOG("Unexpected error: %d\n", err);
639*4882a593Smuzhiyun return err;
640*4882a593Smuzhiyun }
641*4882a593Smuzhiyun
642*4882a593Smuzhiyun
643*4882a593Smuzhiyun /******************************************************************************
644*4882a593Smuzhiyun *
645*4882a593Smuzhiyun * submit_urbs
646*4882a593Smuzhiyun *
647*4882a593Smuzhiyun *****************************************************************************/
submit_urbs(struct camera_data * cam)648*4882a593Smuzhiyun static int submit_urbs(struct camera_data *cam)
649*4882a593Smuzhiyun {
650*4882a593Smuzhiyun struct urb *urb;
651*4882a593Smuzhiyun int fx, err, i, j;
652*4882a593Smuzhiyun
653*4882a593Smuzhiyun for(i=0; i<NUM_SBUF; ++i) {
654*4882a593Smuzhiyun if (cam->sbuf[i].data)
655*4882a593Smuzhiyun continue;
656*4882a593Smuzhiyun cam->sbuf[i].data =
657*4882a593Smuzhiyun kmalloc_array(FRAME_SIZE_PER_DESC, FRAMES_PER_DESC,
658*4882a593Smuzhiyun GFP_KERNEL);
659*4882a593Smuzhiyun if (!cam->sbuf[i].data) {
660*4882a593Smuzhiyun while (--i >= 0) {
661*4882a593Smuzhiyun kfree(cam->sbuf[i].data);
662*4882a593Smuzhiyun cam->sbuf[i].data = NULL;
663*4882a593Smuzhiyun }
664*4882a593Smuzhiyun return -ENOMEM;
665*4882a593Smuzhiyun }
666*4882a593Smuzhiyun }
667*4882a593Smuzhiyun
668*4882a593Smuzhiyun /* We double buffer the Isoc lists, and also know the polling
669*4882a593Smuzhiyun * interval is every frame (1 == (1 << (bInterval -1))).
670*4882a593Smuzhiyun */
671*4882a593Smuzhiyun for(i=0; i<NUM_SBUF; ++i) {
672*4882a593Smuzhiyun if(cam->sbuf[i].urb) {
673*4882a593Smuzhiyun continue;
674*4882a593Smuzhiyun }
675*4882a593Smuzhiyun urb = usb_alloc_urb(FRAMES_PER_DESC, GFP_KERNEL);
676*4882a593Smuzhiyun if (!urb) {
677*4882a593Smuzhiyun for (j = 0; j < i; j++)
678*4882a593Smuzhiyun usb_free_urb(cam->sbuf[j].urb);
679*4882a593Smuzhiyun for (j = 0; j < NUM_SBUF; j++) {
680*4882a593Smuzhiyun kfree(cam->sbuf[j].data);
681*4882a593Smuzhiyun cam->sbuf[j].data = NULL;
682*4882a593Smuzhiyun }
683*4882a593Smuzhiyun return -ENOMEM;
684*4882a593Smuzhiyun }
685*4882a593Smuzhiyun
686*4882a593Smuzhiyun cam->sbuf[i].urb = urb;
687*4882a593Smuzhiyun urb->dev = cam->dev;
688*4882a593Smuzhiyun urb->context = cam;
689*4882a593Smuzhiyun urb->pipe = usb_rcvisocpipe(cam->dev, 1 /*ISOC endpoint*/);
690*4882a593Smuzhiyun urb->transfer_flags = URB_ISO_ASAP;
691*4882a593Smuzhiyun urb->transfer_buffer = cam->sbuf[i].data;
692*4882a593Smuzhiyun urb->complete = cpia2_usb_complete;
693*4882a593Smuzhiyun urb->number_of_packets = FRAMES_PER_DESC;
694*4882a593Smuzhiyun urb->interval = 1;
695*4882a593Smuzhiyun urb->transfer_buffer_length =
696*4882a593Smuzhiyun FRAME_SIZE_PER_DESC * FRAMES_PER_DESC;
697*4882a593Smuzhiyun
698*4882a593Smuzhiyun for (fx = 0; fx < FRAMES_PER_DESC; fx++) {
699*4882a593Smuzhiyun urb->iso_frame_desc[fx].offset =
700*4882a593Smuzhiyun FRAME_SIZE_PER_DESC * fx;
701*4882a593Smuzhiyun urb->iso_frame_desc[fx].length = FRAME_SIZE_PER_DESC;
702*4882a593Smuzhiyun }
703*4882a593Smuzhiyun }
704*4882a593Smuzhiyun
705*4882a593Smuzhiyun
706*4882a593Smuzhiyun /* Queue the ISO urbs, and resubmit in the completion handler */
707*4882a593Smuzhiyun for(i=0; i<NUM_SBUF; ++i) {
708*4882a593Smuzhiyun err = usb_submit_urb(cam->sbuf[i].urb, GFP_KERNEL);
709*4882a593Smuzhiyun if (err) {
710*4882a593Smuzhiyun ERR("usb_submit_urb[%d]() = %d\n", i, err);
711*4882a593Smuzhiyun return err;
712*4882a593Smuzhiyun }
713*4882a593Smuzhiyun }
714*4882a593Smuzhiyun
715*4882a593Smuzhiyun return 0;
716*4882a593Smuzhiyun }
717*4882a593Smuzhiyun
718*4882a593Smuzhiyun /******************************************************************************
719*4882a593Smuzhiyun *
720*4882a593Smuzhiyun * cpia2_usb_stream_start
721*4882a593Smuzhiyun *
722*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_stream_start(struct camera_data * cam,unsigned int alternate)723*4882a593Smuzhiyun int cpia2_usb_stream_start(struct camera_data *cam, unsigned int alternate)
724*4882a593Smuzhiyun {
725*4882a593Smuzhiyun int ret;
726*4882a593Smuzhiyun int old_alt;
727*4882a593Smuzhiyun
728*4882a593Smuzhiyun if(cam->streaming)
729*4882a593Smuzhiyun return 0;
730*4882a593Smuzhiyun
731*4882a593Smuzhiyun if (cam->flush) {
732*4882a593Smuzhiyun int i;
733*4882a593Smuzhiyun DBG("Flushing buffers\n");
734*4882a593Smuzhiyun for(i=0; i<cam->num_frames; ++i) {
735*4882a593Smuzhiyun cam->buffers[i].status = FRAME_EMPTY;
736*4882a593Smuzhiyun cam->buffers[i].length = 0;
737*4882a593Smuzhiyun }
738*4882a593Smuzhiyun cam->curbuff = &cam->buffers[0];
739*4882a593Smuzhiyun cam->workbuff = cam->curbuff->next;
740*4882a593Smuzhiyun cam->flush = false;
741*4882a593Smuzhiyun }
742*4882a593Smuzhiyun
743*4882a593Smuzhiyun old_alt = cam->params.camera_state.stream_mode;
744*4882a593Smuzhiyun cam->params.camera_state.stream_mode = 0;
745*4882a593Smuzhiyun ret = cpia2_usb_change_streaming_alternate(cam, alternate);
746*4882a593Smuzhiyun if (ret < 0) {
747*4882a593Smuzhiyun int ret2;
748*4882a593Smuzhiyun ERR("cpia2_usb_change_streaming_alternate() = %d!\n", ret);
749*4882a593Smuzhiyun cam->params.camera_state.stream_mode = old_alt;
750*4882a593Smuzhiyun ret2 = set_alternate(cam, USBIF_CMDONLY);
751*4882a593Smuzhiyun if (ret2 < 0) {
752*4882a593Smuzhiyun ERR("cpia2_usb_change_streaming_alternate(%d) =%d has already failed. Then tried to call set_alternate(USBIF_CMDONLY) = %d.\n",
753*4882a593Smuzhiyun alternate, ret, ret2);
754*4882a593Smuzhiyun }
755*4882a593Smuzhiyun } else {
756*4882a593Smuzhiyun cam->frame_count = 0;
757*4882a593Smuzhiyun cam->streaming = 1;
758*4882a593Smuzhiyun ret = cpia2_usb_stream_resume(cam);
759*4882a593Smuzhiyun }
760*4882a593Smuzhiyun return ret;
761*4882a593Smuzhiyun }
762*4882a593Smuzhiyun
763*4882a593Smuzhiyun /******************************************************************************
764*4882a593Smuzhiyun *
765*4882a593Smuzhiyun * cpia2_usb_stream_pause
766*4882a593Smuzhiyun *
767*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_stream_pause(struct camera_data * cam)768*4882a593Smuzhiyun int cpia2_usb_stream_pause(struct camera_data *cam)
769*4882a593Smuzhiyun {
770*4882a593Smuzhiyun int ret = 0;
771*4882a593Smuzhiyun if(cam->streaming) {
772*4882a593Smuzhiyun free_sbufs(cam);
773*4882a593Smuzhiyun ret = set_alternate(cam, USBIF_CMDONLY);
774*4882a593Smuzhiyun }
775*4882a593Smuzhiyun return ret;
776*4882a593Smuzhiyun }
777*4882a593Smuzhiyun
778*4882a593Smuzhiyun /******************************************************************************
779*4882a593Smuzhiyun *
780*4882a593Smuzhiyun * cpia2_usb_stream_resume
781*4882a593Smuzhiyun *
782*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_stream_resume(struct camera_data * cam)783*4882a593Smuzhiyun int cpia2_usb_stream_resume(struct camera_data *cam)
784*4882a593Smuzhiyun {
785*4882a593Smuzhiyun int ret = 0;
786*4882a593Smuzhiyun if(cam->streaming) {
787*4882a593Smuzhiyun cam->first_image_seen = 0;
788*4882a593Smuzhiyun ret = set_alternate(cam, cam->params.camera_state.stream_mode);
789*4882a593Smuzhiyun if(ret == 0) {
790*4882a593Smuzhiyun /* for some reason the user effects need to be set
791*4882a593Smuzhiyun again when starting streaming. */
792*4882a593Smuzhiyun cpia2_do_command(cam, CPIA2_CMD_SET_USER_EFFECTS, TRANSFER_WRITE,
793*4882a593Smuzhiyun cam->params.vp_params.user_effects);
794*4882a593Smuzhiyun ret = submit_urbs(cam);
795*4882a593Smuzhiyun }
796*4882a593Smuzhiyun }
797*4882a593Smuzhiyun return ret;
798*4882a593Smuzhiyun }
799*4882a593Smuzhiyun
800*4882a593Smuzhiyun /******************************************************************************
801*4882a593Smuzhiyun *
802*4882a593Smuzhiyun * cpia2_usb_stream_stop
803*4882a593Smuzhiyun *
804*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_stream_stop(struct camera_data * cam)805*4882a593Smuzhiyun int cpia2_usb_stream_stop(struct camera_data *cam)
806*4882a593Smuzhiyun {
807*4882a593Smuzhiyun int ret;
808*4882a593Smuzhiyun
809*4882a593Smuzhiyun ret = cpia2_usb_stream_pause(cam);
810*4882a593Smuzhiyun cam->streaming = 0;
811*4882a593Smuzhiyun configure_transfer_mode(cam, 0);
812*4882a593Smuzhiyun return ret;
813*4882a593Smuzhiyun }
814*4882a593Smuzhiyun
815*4882a593Smuzhiyun /******************************************************************************
816*4882a593Smuzhiyun *
817*4882a593Smuzhiyun * cpia2_usb_probe
818*4882a593Smuzhiyun *
819*4882a593Smuzhiyun * Probe and initialize.
820*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_probe(struct usb_interface * intf,const struct usb_device_id * id)821*4882a593Smuzhiyun static int cpia2_usb_probe(struct usb_interface *intf,
822*4882a593Smuzhiyun const struct usb_device_id *id)
823*4882a593Smuzhiyun {
824*4882a593Smuzhiyun struct usb_device *udev = interface_to_usbdev(intf);
825*4882a593Smuzhiyun struct usb_interface_descriptor *interface;
826*4882a593Smuzhiyun struct camera_data *cam;
827*4882a593Smuzhiyun int ret;
828*4882a593Smuzhiyun
829*4882a593Smuzhiyun /* A multi-config CPiA2 camera? */
830*4882a593Smuzhiyun if (udev->descriptor.bNumConfigurations != 1)
831*4882a593Smuzhiyun return -ENODEV;
832*4882a593Smuzhiyun interface = &intf->cur_altsetting->desc;
833*4882a593Smuzhiyun
834*4882a593Smuzhiyun /* If we get to this point, we found a CPiA2 camera */
835*4882a593Smuzhiyun LOG("CPiA2 USB camera found\n");
836*4882a593Smuzhiyun
837*4882a593Smuzhiyun cam = cpia2_init_camera_struct(intf);
838*4882a593Smuzhiyun if (cam == NULL)
839*4882a593Smuzhiyun return -ENOMEM;
840*4882a593Smuzhiyun
841*4882a593Smuzhiyun cam->dev = udev;
842*4882a593Smuzhiyun cam->iface = interface->bInterfaceNumber;
843*4882a593Smuzhiyun
844*4882a593Smuzhiyun ret = set_alternate(cam, USBIF_CMDONLY);
845*4882a593Smuzhiyun if (ret < 0) {
846*4882a593Smuzhiyun ERR("%s: usb_set_interface error (ret = %d)\n", __func__, ret);
847*4882a593Smuzhiyun goto alt_err;
848*4882a593Smuzhiyun }
849*4882a593Smuzhiyun
850*4882a593Smuzhiyun
851*4882a593Smuzhiyun if((ret = cpia2_init_camera(cam)) < 0) {
852*4882a593Smuzhiyun ERR("%s: failed to initialize cpia2 camera (ret = %d)\n", __func__, ret);
853*4882a593Smuzhiyun goto alt_err;
854*4882a593Smuzhiyun }
855*4882a593Smuzhiyun LOG(" CPiA Version: %d.%02d (%d.%d)\n",
856*4882a593Smuzhiyun cam->params.version.firmware_revision_hi,
857*4882a593Smuzhiyun cam->params.version.firmware_revision_lo,
858*4882a593Smuzhiyun cam->params.version.asic_id,
859*4882a593Smuzhiyun cam->params.version.asic_rev);
860*4882a593Smuzhiyun LOG(" CPiA PnP-ID: %04x:%04x:%04x\n",
861*4882a593Smuzhiyun cam->params.pnp_id.vendor,
862*4882a593Smuzhiyun cam->params.pnp_id.product,
863*4882a593Smuzhiyun cam->params.pnp_id.device_revision);
864*4882a593Smuzhiyun LOG(" SensorID: %d.(version %d)\n",
865*4882a593Smuzhiyun cam->params.version.sensor_flags,
866*4882a593Smuzhiyun cam->params.version.sensor_rev);
867*4882a593Smuzhiyun
868*4882a593Smuzhiyun usb_set_intfdata(intf, cam);
869*4882a593Smuzhiyun
870*4882a593Smuzhiyun ret = cpia2_register_camera(cam);
871*4882a593Smuzhiyun if (ret < 0) {
872*4882a593Smuzhiyun ERR("%s: Failed to register cpia2 camera (ret = %d)\n", __func__, ret);
873*4882a593Smuzhiyun goto alt_err;
874*4882a593Smuzhiyun }
875*4882a593Smuzhiyun
876*4882a593Smuzhiyun return 0;
877*4882a593Smuzhiyun
878*4882a593Smuzhiyun alt_err:
879*4882a593Smuzhiyun cpia2_deinit_camera_struct(cam, intf);
880*4882a593Smuzhiyun return ret;
881*4882a593Smuzhiyun }
882*4882a593Smuzhiyun
883*4882a593Smuzhiyun /******************************************************************************
884*4882a593Smuzhiyun *
885*4882a593Smuzhiyun * cpia2_disconnect
886*4882a593Smuzhiyun *
887*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_disconnect(struct usb_interface * intf)888*4882a593Smuzhiyun static void cpia2_usb_disconnect(struct usb_interface *intf)
889*4882a593Smuzhiyun {
890*4882a593Smuzhiyun struct camera_data *cam = usb_get_intfdata(intf);
891*4882a593Smuzhiyun usb_set_intfdata(intf, NULL);
892*4882a593Smuzhiyun
893*4882a593Smuzhiyun DBG("Stopping stream\n");
894*4882a593Smuzhiyun cpia2_usb_stream_stop(cam);
895*4882a593Smuzhiyun
896*4882a593Smuzhiyun mutex_lock(&cam->v4l2_lock);
897*4882a593Smuzhiyun DBG("Unregistering camera\n");
898*4882a593Smuzhiyun cpia2_unregister_camera(cam);
899*4882a593Smuzhiyun v4l2_device_disconnect(&cam->v4l2_dev);
900*4882a593Smuzhiyun mutex_unlock(&cam->v4l2_lock);
901*4882a593Smuzhiyun
902*4882a593Smuzhiyun if(cam->buffers) {
903*4882a593Smuzhiyun DBG("Wakeup waiting processes\n");
904*4882a593Smuzhiyun cam->curbuff->status = FRAME_READY;
905*4882a593Smuzhiyun cam->curbuff->length = 0;
906*4882a593Smuzhiyun wake_up_interruptible(&cam->wq_stream);
907*4882a593Smuzhiyun }
908*4882a593Smuzhiyun
909*4882a593Smuzhiyun v4l2_device_put(&cam->v4l2_dev);
910*4882a593Smuzhiyun
911*4882a593Smuzhiyun LOG("CPiA2 camera disconnected.\n");
912*4882a593Smuzhiyun }
913*4882a593Smuzhiyun
cpia2_usb_suspend(struct usb_interface * intf,pm_message_t message)914*4882a593Smuzhiyun static int cpia2_usb_suspend(struct usb_interface *intf, pm_message_t message)
915*4882a593Smuzhiyun {
916*4882a593Smuzhiyun struct camera_data *cam = usb_get_intfdata(intf);
917*4882a593Smuzhiyun
918*4882a593Smuzhiyun mutex_lock(&cam->v4l2_lock);
919*4882a593Smuzhiyun if (cam->streaming) {
920*4882a593Smuzhiyun cpia2_usb_stream_stop(cam);
921*4882a593Smuzhiyun cam->streaming = 1;
922*4882a593Smuzhiyun }
923*4882a593Smuzhiyun mutex_unlock(&cam->v4l2_lock);
924*4882a593Smuzhiyun
925*4882a593Smuzhiyun dev_info(&intf->dev, "going into suspend..\n");
926*4882a593Smuzhiyun return 0;
927*4882a593Smuzhiyun }
928*4882a593Smuzhiyun
929*4882a593Smuzhiyun /* Resume device - start device. */
cpia2_usb_resume(struct usb_interface * intf)930*4882a593Smuzhiyun static int cpia2_usb_resume(struct usb_interface *intf)
931*4882a593Smuzhiyun {
932*4882a593Smuzhiyun struct camera_data *cam = usb_get_intfdata(intf);
933*4882a593Smuzhiyun
934*4882a593Smuzhiyun mutex_lock(&cam->v4l2_lock);
935*4882a593Smuzhiyun v4l2_ctrl_handler_setup(&cam->hdl);
936*4882a593Smuzhiyun if (cam->streaming) {
937*4882a593Smuzhiyun cam->streaming = 0;
938*4882a593Smuzhiyun cpia2_usb_stream_start(cam,
939*4882a593Smuzhiyun cam->params.camera_state.stream_mode);
940*4882a593Smuzhiyun }
941*4882a593Smuzhiyun mutex_unlock(&cam->v4l2_lock);
942*4882a593Smuzhiyun
943*4882a593Smuzhiyun dev_info(&intf->dev, "coming out of suspend..\n");
944*4882a593Smuzhiyun return 0;
945*4882a593Smuzhiyun }
946*4882a593Smuzhiyun
947*4882a593Smuzhiyun /******************************************************************************
948*4882a593Smuzhiyun *
949*4882a593Smuzhiyun * usb_cpia2_init
950*4882a593Smuzhiyun *
951*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_init(void)952*4882a593Smuzhiyun int cpia2_usb_init(void)
953*4882a593Smuzhiyun {
954*4882a593Smuzhiyun return usb_register(&cpia2_driver);
955*4882a593Smuzhiyun }
956*4882a593Smuzhiyun
957*4882a593Smuzhiyun /******************************************************************************
958*4882a593Smuzhiyun *
959*4882a593Smuzhiyun * usb_cpia_cleanup
960*4882a593Smuzhiyun *
961*4882a593Smuzhiyun *****************************************************************************/
cpia2_usb_cleanup(void)962*4882a593Smuzhiyun void cpia2_usb_cleanup(void)
963*4882a593Smuzhiyun {
964*4882a593Smuzhiyun schedule_timeout(2 * HZ);
965*4882a593Smuzhiyun usb_deregister(&cpia2_driver);
966*4882a593Smuzhiyun }
967