Commit 6c493f8b authored by Hans Verkuil's avatar Hans Verkuil Committed by Mauro Carvalho Chehab

[media] cpia2: major overhaul to get it in a working state again

This driver was severely broken. This patch makes it work again, and updates
it to the latest V4L2 frameworks (except for videobuf2). It passes the
v4l2-compliance tests and it now handles suspend/resume correctly.

Several custom controls are replaced by new standard controls, only the
USB_ALTERNATE control remains.

Tested with the Hanse HVS-CM500PC USB microscope.
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 04ef0524
...@@ -32,11 +32,12 @@ ...@@ -32,11 +32,12 @@
#define __CPIA2_H__ #define __CPIA2_H__
#include <linux/videodev2.h> #include <linux/videodev2.h>
#include <media/v4l2-common.h>
#include <linux/usb.h> #include <linux/usb.h>
#include <linux/poll.h> #include <linux/poll.h>
#include <media/v4l2-common.h>
#include <media/v4l2-device.h>
#include <media/v4l2-ctrls.h>
#include "cpia2dev.h"
#include "cpia2_registers.h" #include "cpia2_registers.h"
/* define for verbose debug output */ /* define for verbose debug output */
...@@ -65,7 +66,6 @@ ...@@ -65,7 +66,6 @@
/* Flicker Modes */ /* Flicker Modes */
#define NEVER_FLICKER 0 #define NEVER_FLICKER 0
#define ANTI_FLICKER_ON 1
#define FLICKER_60 60 #define FLICKER_60 60
#define FLICKER_50 50 #define FLICKER_50 50
...@@ -148,7 +148,6 @@ enum { ...@@ -148,7 +148,6 @@ enum {
#define DEFAULT_BRIGHTNESS 0x46 #define DEFAULT_BRIGHTNESS 0x46
#define DEFAULT_CONTRAST 0x93 #define DEFAULT_CONTRAST 0x93
#define DEFAULT_SATURATION 0x7f #define DEFAULT_SATURATION 0x7f
#define DEFAULT_TARGET_KB 0x30
/* Power state */ /* Power state */
#define HI_POWER_MODE CPIA2_SYSTEM_CONTROL_HIGH_POWER #define HI_POWER_MODE CPIA2_SYSTEM_CONTROL_HIGH_POWER
...@@ -287,7 +286,6 @@ struct camera_params { ...@@ -287,7 +286,6 @@ struct camera_params {
struct { struct {
u8 cam_register; u8 cam_register;
u8 flicker_mode_req; /* 1 if flicker on, else never flicker */ u8 flicker_mode_req; /* 1 if flicker on, else never flicker */
int mains_frequency;
} flicker_control; } flicker_control;
struct { struct {
...@@ -337,7 +335,7 @@ struct camera_params { ...@@ -337,7 +335,7 @@ struct camera_params {
u8 vc_control; u8 vc_control;
u8 vc_mp_direction; u8 vc_mp_direction;
u8 vc_mp_data; u8 vc_mp_data;
u8 target_kb; u8 quality;
} vc_params; } vc_params;
struct { struct {
...@@ -366,23 +364,23 @@ struct framebuf { ...@@ -366,23 +364,23 @@ struct framebuf {
struct framebuf *next; struct framebuf *next;
}; };
struct cpia2_fh {
enum v4l2_priority prio;
u8 mmapped;
};
struct camera_data { struct camera_data {
/* locks */ /* locks */
struct v4l2_device v4l2_dev;
struct mutex v4l2_lock; /* serialize file operations */ struct mutex v4l2_lock; /* serialize file operations */
struct v4l2_prio_state prio; struct v4l2_ctrl_handler hdl;
struct {
/* Lights control cluster */
struct v4l2_ctrl *top_light;
struct v4l2_ctrl *bottom_light;
};
struct v4l2_ctrl *usb_alt;
/* camera status */ /* camera status */
volatile int present; /* Is the camera still present? */
int open_count; /* # of process that have camera open */
int first_image_seen; int first_image_seen;
u8 mains_freq; /* for flicker control */
enum sensors sensor_type; enum sensors sensor_type;
u8 flush; u8 flush;
struct v4l2_fh *stream_fh;
u8 mmapped; u8 mmapped;
int streaming; /* 0 = no, 1 = yes */ int streaming; /* 0 = no, 1 = yes */
int xfer_mode; /* XFER_BULK or XFER_ISOC */ int xfer_mode; /* XFER_BULK or XFER_ISOC */
...@@ -390,7 +388,7 @@ struct camera_data { ...@@ -390,7 +388,7 @@ struct camera_data {
/* v4l */ /* v4l */
int video_size; /* VIDEO_SIZE_ */ int video_size; /* VIDEO_SIZE_ */
struct video_device *vdev; /* v4l videodev */ struct video_device vdev; /* v4l videodev */
u32 width; u32 width;
u32 height; /* Its size */ u32 height; /* Its size */
__u32 pixelformat; /* Format fourcc */ __u32 pixelformat; /* Format fourcc */
...@@ -425,6 +423,7 @@ struct camera_data { ...@@ -425,6 +423,7 @@ struct camera_data {
/* v4l */ /* v4l */
int cpia2_register_camera(struct camera_data *cam); int cpia2_register_camera(struct camera_data *cam);
void cpia2_unregister_camera(struct camera_data *cam); void cpia2_unregister_camera(struct camera_data *cam);
void cpia2_camera_release(struct v4l2_device *v4l2_dev);
/* core */ /* core */
int cpia2_reset_camera(struct camera_data *cam); int cpia2_reset_camera(struct camera_data *cam);
...@@ -443,7 +442,7 @@ int cpia2_send_command(struct camera_data *cam, struct cpia2_command *cmd); ...@@ -443,7 +442,7 @@ int cpia2_send_command(struct camera_data *cam, struct cpia2_command *cmd);
int cpia2_do_command(struct camera_data *cam, int cpia2_do_command(struct camera_data *cam,
unsigned int command, unsigned int command,
unsigned char direction, unsigned char param); unsigned char direction, unsigned char param);
struct camera_data *cpia2_init_camera_struct(void); struct camera_data *cpia2_init_camera_struct(struct usb_interface *intf);
int cpia2_init_camera(struct camera_data *cam); int cpia2_init_camera(struct camera_data *cam);
int cpia2_allocate_buffers(struct camera_data *cam); int cpia2_allocate_buffers(struct camera_data *cam);
void cpia2_free_buffers(struct camera_data *cam); void cpia2_free_buffers(struct camera_data *cam);
...@@ -454,7 +453,6 @@ unsigned int cpia2_poll(struct camera_data *cam, ...@@ -454,7 +453,6 @@ unsigned int cpia2_poll(struct camera_data *cam,
int cpia2_remap_buffer(struct camera_data *cam, struct vm_area_struct *vma); int cpia2_remap_buffer(struct camera_data *cam, struct vm_area_struct *vma);
void cpia2_set_property_flip(struct camera_data *cam, int prop_val); void cpia2_set_property_flip(struct camera_data *cam, int prop_val);
void cpia2_set_property_mirror(struct camera_data *cam, int prop_val); void cpia2_set_property_mirror(struct camera_data *cam, int prop_val);
int cpia2_set_target_kb(struct camera_data *cam, unsigned char value);
int cpia2_set_gpio(struct camera_data *cam, unsigned char setting); int cpia2_set_gpio(struct camera_data *cam, unsigned char setting);
int cpia2_set_fps(struct camera_data *cam, int framerate); int cpia2_set_fps(struct camera_data *cam, int framerate);
......
This diff is collapsed.
...@@ -54,6 +54,8 @@ static void cpia2_usb_complete(struct urb *urb); ...@@ -54,6 +54,8 @@ static void cpia2_usb_complete(struct urb *urb);
static int cpia2_usb_probe(struct usb_interface *intf, static int cpia2_usb_probe(struct usb_interface *intf,
const struct usb_device_id *id); const struct usb_device_id *id);
static void cpia2_usb_disconnect(struct usb_interface *intf); static void cpia2_usb_disconnect(struct usb_interface *intf);
static int cpia2_usb_suspend(struct usb_interface *intf, pm_message_t message);
static int cpia2_usb_resume(struct usb_interface *intf);
static void free_sbufs(struct camera_data *cam); static void free_sbufs(struct camera_data *cam);
static void add_APPn(struct camera_data *cam); static void add_APPn(struct camera_data *cam);
...@@ -74,6 +76,9 @@ static struct usb_driver cpia2_driver = { ...@@ -74,6 +76,9 @@ static struct usb_driver cpia2_driver = {
.name = "cpia2", .name = "cpia2",
.probe = cpia2_usb_probe, .probe = cpia2_usb_probe,
.disconnect = cpia2_usb_disconnect, .disconnect = cpia2_usb_disconnect,
.suspend = cpia2_usb_suspend,
.resume = cpia2_usb_resume,
.reset_resume = cpia2_usb_resume,
.id_table = cpia2_id_table .id_table = cpia2_id_table
}; };
...@@ -218,10 +223,9 @@ static void cpia2_usb_complete(struct urb *urb) ...@@ -218,10 +223,9 @@ static void cpia2_usb_complete(struct urb *urb)
return; return;
} }
if (!cam->streaming || !cam->present || cam->open_count == 0) { if (!cam->streaming || !video_is_registered(&cam->vdev)) {
LOG("Will now stop the streaming: streaming = %d, " LOG("Will now stop the streaming: streaming = %d, present=%d\n",
"present=%d, open_count=%d\n", cam->streaming, video_is_registered(&cam->vdev));
cam->streaming, cam->present, cam->open_count);
return; return;
} }
...@@ -392,7 +396,7 @@ static int configure_transfer_mode(struct camera_data *cam, unsigned int alt) ...@@ -392,7 +396,7 @@ static int configure_transfer_mode(struct camera_data *cam, unsigned int alt)
struct cpia2_command cmd; struct cpia2_command cmd;
unsigned char reg; unsigned char reg;
if(!cam->present) if (!video_is_registered(&cam->vdev))
return -ENODEV; return -ENODEV;
/*** /***
...@@ -752,8 +756,8 @@ int cpia2_usb_stream_pause(struct camera_data *cam) ...@@ -752,8 +756,8 @@ int cpia2_usb_stream_pause(struct camera_data *cam)
{ {
int ret = 0; int ret = 0;
if(cam->streaming) { if(cam->streaming) {
ret = set_alternate(cam, USBIF_CMDONLY);
free_sbufs(cam); free_sbufs(cam);
ret = set_alternate(cam, USBIF_CMDONLY);
} }
return ret; return ret;
} }
...@@ -770,6 +774,10 @@ int cpia2_usb_stream_resume(struct camera_data *cam) ...@@ -770,6 +774,10 @@ int cpia2_usb_stream_resume(struct camera_data *cam)
cam->first_image_seen = 0; cam->first_image_seen = 0;
ret = set_alternate(cam, cam->params.camera_state.stream_mode); ret = set_alternate(cam, cam->params.camera_state.stream_mode);
if(ret == 0) { if(ret == 0) {
/* for some reason the user effects need to be set
again when starting streaming. */
cpia2_do_command(cam, CPIA2_CMD_SET_USER_EFFECTS, TRANSFER_WRITE,
cam->params.vp_params.user_effects);
ret = submit_urbs(cam); ret = submit_urbs(cam);
} }
} }
...@@ -784,6 +792,7 @@ int cpia2_usb_stream_resume(struct camera_data *cam) ...@@ -784,6 +792,7 @@ int cpia2_usb_stream_resume(struct camera_data *cam)
int cpia2_usb_stream_stop(struct camera_data *cam) int cpia2_usb_stream_stop(struct camera_data *cam)
{ {
int ret; int ret;
ret = cpia2_usb_stream_pause(cam); ret = cpia2_usb_stream_pause(cam);
cam->streaming = 0; cam->streaming = 0;
configure_transfer_mode(cam, 0); configure_transfer_mode(cam, 0);
...@@ -812,7 +821,8 @@ static int cpia2_usb_probe(struct usb_interface *intf, ...@@ -812,7 +821,8 @@ static int cpia2_usb_probe(struct usb_interface *intf,
/* If we get to this point, we found a CPiA2 camera */ /* If we get to this point, we found a CPiA2 camera */
LOG("CPiA2 USB camera found\n"); LOG("CPiA2 USB camera found\n");
if((cam = cpia2_init_camera_struct()) == NULL) cam = cpia2_init_camera_struct(intf);
if (cam == NULL)
return -ENOMEM; return -ENOMEM;
cam->dev = udev; cam->dev = udev;
...@@ -825,16 +835,9 @@ static int cpia2_usb_probe(struct usb_interface *intf, ...@@ -825,16 +835,9 @@ static int cpia2_usb_probe(struct usb_interface *intf,
return ret; return ret;
} }
if ((ret = cpia2_register_camera(cam)) < 0) {
ERR("%s: Failed to register cpia2 camera (ret = %d)\n", __func__, ret);
kfree(cam);
return ret;
}
if((ret = cpia2_init_camera(cam)) < 0) { if((ret = cpia2_init_camera(cam)) < 0) {
ERR("%s: failed to initialize cpia2 camera (ret = %d)\n", __func__, ret); ERR("%s: failed to initialize cpia2 camera (ret = %d)\n", __func__, ret);
cpia2_unregister_camera(cam);
kfree(cam); kfree(cam);
return ret; return ret;
} }
...@@ -853,6 +856,13 @@ static int cpia2_usb_probe(struct usb_interface *intf, ...@@ -853,6 +856,13 @@ static int cpia2_usb_probe(struct usb_interface *intf,
usb_set_intfdata(intf, cam); usb_set_intfdata(intf, cam);
ret = cpia2_register_camera(cam);
if (ret < 0) {
ERR("%s: Failed to register cpia2 camera (ret = %d)\n", __func__, ret);
kfree(cam);
return ret;
}
return 0; return 0;
} }
...@@ -865,13 +875,16 @@ static void cpia2_usb_disconnect(struct usb_interface *intf) ...@@ -865,13 +875,16 @@ static void cpia2_usb_disconnect(struct usb_interface *intf)
{ {
struct camera_data *cam = usb_get_intfdata(intf); struct camera_data *cam = usb_get_intfdata(intf);
usb_set_intfdata(intf, NULL); usb_set_intfdata(intf, NULL);
cam->present = 0;
DBG("Stopping stream\n"); DBG("Stopping stream\n");
cpia2_usb_stream_stop(cam); cpia2_usb_stream_stop(cam);
mutex_lock(&cam->v4l2_lock);
DBG("Unregistering camera\n"); DBG("Unregistering camera\n");
cpia2_unregister_camera(cam); cpia2_unregister_camera(cam);
v4l2_device_disconnect(&cam->v4l2_dev);
mutex_unlock(&cam->v4l2_lock);
v4l2_device_put(&cam->v4l2_dev);
if(cam->buffers) { if(cam->buffers) {
DBG("Wakeup waiting processes\n"); DBG("Wakeup waiting processes\n");
...@@ -884,14 +897,41 @@ static void cpia2_usb_disconnect(struct usb_interface *intf) ...@@ -884,14 +897,41 @@ static void cpia2_usb_disconnect(struct usb_interface *intf)
DBG("Releasing interface\n"); DBG("Releasing interface\n");
usb_driver_release_interface(&cpia2_driver, intf); usb_driver_release_interface(&cpia2_driver, intf);
if (cam->open_count == 0) { LOG("CPiA2 camera disconnected.\n");
DBG("Freeing camera structure\n"); }
kfree(cam);
static int cpia2_usb_suspend(struct usb_interface *intf, pm_message_t message)
{
struct camera_data *cam = usb_get_intfdata(intf);
mutex_lock(&cam->v4l2_lock);
if (cam->streaming) {
cpia2_usb_stream_stop(cam);
cam->streaming = 1;
} }
mutex_unlock(&cam->v4l2_lock);
LOG("CPiA2 camera disconnected.\n"); dev_info(&intf->dev, "going into suspend..\n");
return 0;
} }
/* Resume device - start device. */
static int cpia2_usb_resume(struct usb_interface *intf)
{
struct camera_data *cam = usb_get_intfdata(intf);
mutex_lock(&cam->v4l2_lock);
v4l2_ctrl_handler_setup(&cam->hdl);
if (cam->streaming) {
cam->streaming = 0;
cpia2_usb_stream_start(cam,
cam->params.camera_state.stream_mode);
}
mutex_unlock(&cam->v4l2_lock);
dev_info(&intf->dev, "coming out of suspend..\n");
return 0;
}
/****************************************************************************** /******************************************************************************
* *
......
This diff is collapsed.
/****************************************************************************
*
* Filename: cpia2dev.h
*
* Copyright 2001, STMicrolectronics, Inc.
*
* Contact: steve.miller@st.com
*
* Description:
* This file provides definitions for applications wanting to use the
* cpia2 driver beyond the generic v4l capabilities.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
****************************************************************************/
#ifndef CPIA2_DEV_HEADER
#define CPIA2_DEV_HEADER
#include <linux/videodev2.h>
/***
* The following defines are ioctl numbers based on video4linux private ioctls,
* which can range from 192 (BASE_VIDIOCPRIVATE) to 255. All of these take int
* args
*/
#define CPIA2_IOC_SET_GPIO _IOW('v', BASE_VIDIOC_PRIVATE + 17, __u32)
/* V4L2 driver specific controls */
#define CPIA2_CID_TARGET_KB (V4L2_CID_PRIVATE_BASE+0)
#define CPIA2_CID_GPIO (V4L2_CID_PRIVATE_BASE+1)
#define CPIA2_CID_FLICKER_MODE (V4L2_CID_PRIVATE_BASE+2)
#define CPIA2_CID_FRAMERATE (V4L2_CID_PRIVATE_BASE+3)
#define CPIA2_CID_USB_ALT (V4L2_CID_PRIVATE_BASE+4)
#define CPIA2_CID_LIGHTS (V4L2_CID_PRIVATE_BASE+5)
#define CPIA2_CID_RESET_CAMERA (V4L2_CID_PRIVATE_BASE+6)
#endif
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment