2007-01-21 00:58:17 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* device driver for Conexant 2388x based TV cards
|
|
|
|
* video4linux video interface
|
|
|
|
*
|
|
|
|
* (c) 2003-04 Gerd Knorr <kraxel@bytesex.org> [SuSE Labs]
|
|
|
|
*
|
2007-01-21 00:58:17 +08:00
|
|
|
* (c) 2005-2006 Mauro Carvalho Chehab <mchehab@infradead.org>
|
|
|
|
* - Multituner support
|
|
|
|
* - video_ioctl2 conversion
|
|
|
|
* - PAL/M fixes
|
|
|
|
*
|
2005-04-17 06:20:36 +08:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
#include <linux/kmod.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/kthread.h>
|
|
|
|
#include <asm/div64.h>
|
|
|
|
|
|
|
|
#include "cx88.h"
|
2006-01-10 01:32:31 +08:00
|
|
|
#include <media/v4l2-common.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-04-10 02:43:41 +08:00
|
|
|
#ifdef CONFIG_VIDEO_V4L1_COMPAT
|
2005-11-09 13:37:49 +08:00
|
|
|
/* Include V4L1 specific functions. Should be removed soon */
|
|
|
|
#include <linux/videodev.h>
|
2006-04-10 02:43:41 +08:00
|
|
|
#endif
|
2005-11-09 13:37:49 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
MODULE_DESCRIPTION("v4l2 driver module for cx2388x based TV cards");
|
|
|
|
MODULE_AUTHOR("Gerd Knorr <kraxel@bytesex.org> [SuSE Labs]");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
static unsigned int video_nr[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
|
|
|
|
static unsigned int vbi_nr[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
|
|
|
|
static unsigned int radio_nr[] = {[0 ... (CX88_MAXBOARDS - 1)] = UNSET };
|
|
|
|
|
|
|
|
module_param_array(video_nr, int, NULL, 0444);
|
|
|
|
module_param_array(vbi_nr, int, NULL, 0444);
|
|
|
|
module_param_array(radio_nr, int, NULL, 0444);
|
|
|
|
|
|
|
|
MODULE_PARM_DESC(video_nr,"video device numbers");
|
|
|
|
MODULE_PARM_DESC(vbi_nr,"vbi device numbers");
|
|
|
|
MODULE_PARM_DESC(radio_nr,"radio device numbers");
|
|
|
|
|
|
|
|
static unsigned int video_debug = 0;
|
|
|
|
module_param(video_debug,int,0644);
|
|
|
|
MODULE_PARM_DESC(video_debug,"enable debug messages [video]");
|
|
|
|
|
|
|
|
static unsigned int irq_debug = 0;
|
|
|
|
module_param(irq_debug,int,0644);
|
|
|
|
MODULE_PARM_DESC(irq_debug,"enable debug messages [IRQ handler]");
|
|
|
|
|
|
|
|
static unsigned int vid_limit = 16;
|
|
|
|
module_param(vid_limit,int,0644);
|
|
|
|
MODULE_PARM_DESC(vid_limit,"capture memory limit in megabytes");
|
|
|
|
|
|
|
|
#define dprintk(level,fmt, arg...) if (video_debug >= level) \
|
2005-09-10 04:03:41 +08:00
|
|
|
printk(KERN_DEBUG "%s/0: " fmt, core->name , ## arg)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
static LIST_HEAD(cx8800_devlist);
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------- */
|
|
|
|
/* static data */
|
|
|
|
|
|
|
|
static struct cx8800_fmt formats[] = {
|
|
|
|
{
|
|
|
|
.name = "8 bpp, gray",
|
|
|
|
.fourcc = V4L2_PIX_FMT_GREY,
|
|
|
|
.cxformat = ColorFormatY8,
|
|
|
|
.depth = 8,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "15 bpp RGB, le",
|
|
|
|
.fourcc = V4L2_PIX_FMT_RGB555,
|
|
|
|
.cxformat = ColorFormatRGB15,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "15 bpp RGB, be",
|
|
|
|
.fourcc = V4L2_PIX_FMT_RGB555X,
|
|
|
|
.cxformat = ColorFormatRGB15 | ColorFormatBSWAP,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "16 bpp RGB, le",
|
|
|
|
.fourcc = V4L2_PIX_FMT_RGB565,
|
|
|
|
.cxformat = ColorFormatRGB16,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "16 bpp RGB, be",
|
|
|
|
.fourcc = V4L2_PIX_FMT_RGB565X,
|
|
|
|
.cxformat = ColorFormatRGB16 | ColorFormatBSWAP,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "24 bpp RGB, le",
|
|
|
|
.fourcc = V4L2_PIX_FMT_BGR24,
|
|
|
|
.cxformat = ColorFormatRGB24,
|
|
|
|
.depth = 24,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "32 bpp RGB, le",
|
|
|
|
.fourcc = V4L2_PIX_FMT_BGR32,
|
|
|
|
.cxformat = ColorFormatRGB32,
|
|
|
|
.depth = 32,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "32 bpp RGB, be",
|
|
|
|
.fourcc = V4L2_PIX_FMT_RGB32,
|
|
|
|
.cxformat = ColorFormatRGB32 | ColorFormatBSWAP | ColorFormatWSWAP,
|
|
|
|
.depth = 32,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "4:2:2, packed, YUYV",
|
|
|
|
.fourcc = V4L2_PIX_FMT_YUYV,
|
|
|
|
.cxformat = ColorFormatYUY2,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},{
|
|
|
|
.name = "4:2:2, packed, UYVY",
|
|
|
|
.fourcc = V4L2_PIX_FMT_UYVY,
|
|
|
|
.cxformat = ColorFormatYUY2 | ColorFormatBSWAP,
|
|
|
|
.depth = 16,
|
|
|
|
.flags = FORMAT_FLAGS_PACKED,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct cx8800_fmt* format_by_fourcc(unsigned int fourcc)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(formats); i++)
|
|
|
|
if (formats[i].fourcc == fourcc)
|
|
|
|
return formats+i;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static const struct v4l2_queryctrl no_ctl = {
|
|
|
|
.name = "42",
|
|
|
|
.flags = V4L2_CTRL_FLAG_DISABLED,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct cx88_ctrl cx8800_ctls[] = {
|
|
|
|
/* --- video --- */
|
|
|
|
{
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_BRIGHTNESS,
|
|
|
|
.name = "Brightness",
|
|
|
|
.minimum = 0x00,
|
|
|
|
.maximum = 0xff,
|
|
|
|
.step = 1,
|
2006-03-12 11:03:47 +08:00
|
|
|
.default_value = 0x7f,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
|
|
|
.off = 128,
|
|
|
|
.reg = MO_CONTR_BRIGHT,
|
|
|
|
.mask = 0x00ff,
|
|
|
|
.shift = 0,
|
|
|
|
},{
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_CONTRAST,
|
|
|
|
.name = "Contrast",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 0xff,
|
|
|
|
.step = 1,
|
2006-01-10 01:25:26 +08:00
|
|
|
.default_value = 0x3f,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
2005-07-13 04:58:44 +08:00
|
|
|
.off = 0,
|
2005-04-17 06:20:36 +08:00
|
|
|
.reg = MO_CONTR_BRIGHT,
|
|
|
|
.mask = 0xff00,
|
|
|
|
.shift = 8,
|
|
|
|
},{
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_HUE,
|
|
|
|
.name = "Hue",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 0xff,
|
|
|
|
.step = 1,
|
2006-03-12 11:03:47 +08:00
|
|
|
.default_value = 0x7f,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
2005-07-08 08:58:38 +08:00
|
|
|
.off = 128,
|
2005-04-17 06:20:36 +08:00
|
|
|
.reg = MO_HUE,
|
|
|
|
.mask = 0x00ff,
|
|
|
|
.shift = 0,
|
|
|
|
},{
|
|
|
|
/* strictly, this only describes only U saturation.
|
|
|
|
* V saturation is handled specially through code.
|
|
|
|
*/
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_SATURATION,
|
|
|
|
.name = "Saturation",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 0xff,
|
|
|
|
.step = 1,
|
2006-01-10 01:25:26 +08:00
|
|
|
.default_value = 0x7f,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
|
|
|
.off = 0,
|
|
|
|
.reg = MO_UV_SATURATION,
|
|
|
|
.mask = 0x00ff,
|
|
|
|
.shift = 0,
|
|
|
|
},{
|
|
|
|
/* --- audio --- */
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_AUDIO_MUTE,
|
|
|
|
.name = "Mute",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 1,
|
2006-01-10 01:25:26 +08:00
|
|
|
.default_value = 1,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_BOOLEAN,
|
|
|
|
},
|
|
|
|
.reg = AUD_VOL_CTL,
|
|
|
|
.sreg = SHADOW_AUD_VOL_CTL,
|
|
|
|
.mask = (1 << 6),
|
|
|
|
.shift = 6,
|
|
|
|
},{
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_AUDIO_VOLUME,
|
|
|
|
.name = "Volume",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 0x3f,
|
|
|
|
.step = 1,
|
2006-03-12 11:03:47 +08:00
|
|
|
.default_value = 0x3f,
|
2005-04-17 06:20:36 +08:00
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
|
|
|
.reg = AUD_VOL_CTL,
|
|
|
|
.sreg = SHADOW_AUD_VOL_CTL,
|
|
|
|
.mask = 0x3f,
|
|
|
|
.shift = 0,
|
|
|
|
},{
|
|
|
|
.v = {
|
|
|
|
.id = V4L2_CID_AUDIO_BALANCE,
|
|
|
|
.name = "Balance",
|
|
|
|
.minimum = 0,
|
|
|
|
.maximum = 0x7f,
|
|
|
|
.step = 1,
|
|
|
|
.default_value = 0x40,
|
|
|
|
.type = V4L2_CTRL_TYPE_INTEGER,
|
|
|
|
},
|
|
|
|
.reg = AUD_BAL_CTL,
|
|
|
|
.sreg = SHADOW_AUD_BAL_CTL,
|
|
|
|
.mask = 0x7f,
|
|
|
|
.shift = 0,
|
|
|
|
}
|
|
|
|
};
|
2005-05-01 23:59:29 +08:00
|
|
|
static const int CX8800_CTLS = ARRAY_SIZE(cx8800_ctls);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-06-27 10:42:39 +08:00
|
|
|
const u32 cx88_user_ctrls[] = {
|
|
|
|
V4L2_CID_USER_CLASS,
|
|
|
|
V4L2_CID_BRIGHTNESS,
|
|
|
|
V4L2_CID_CONTRAST,
|
|
|
|
V4L2_CID_SATURATION,
|
|
|
|
V4L2_CID_HUE,
|
|
|
|
V4L2_CID_AUDIO_VOLUME,
|
|
|
|
V4L2_CID_AUDIO_BALANCE,
|
|
|
|
V4L2_CID_AUDIO_MUTE,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
EXPORT_SYMBOL(cx88_user_ctrls);
|
|
|
|
|
|
|
|
static const u32 *ctrl_classes[] = {
|
|
|
|
cx88_user_ctrls,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
int cx8800_ctrl_query(struct v4l2_queryctrl *qctrl)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (qctrl->id < V4L2_CID_BASE ||
|
|
|
|
qctrl->id >= V4L2_CID_LASTP1)
|
|
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < CX8800_CTLS; i++)
|
|
|
|
if (cx8800_ctls[i].v.id == qctrl->id)
|
|
|
|
break;
|
|
|
|
if (i == CX8800_CTLS) {
|
|
|
|
*qctrl = no_ctl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*qctrl = cx8800_ctls[i].v;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(cx8800_ctrl_query);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* ------------------------------------------------------------------- */
|
|
|
|
/* resource management */
|
|
|
|
|
|
|
|
static int res_get(struct cx8800_dev *dev, struct cx8800_fh *fh, unsigned int bit)
|
|
|
|
{
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (fh->resources & bit)
|
|
|
|
/* have it already allocated */
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* is it free? */
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_lock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (dev->resources & bit) {
|
|
|
|
/* no, someone else uses it */
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* it's free, grab it */
|
|
|
|
fh->resources |= bit;
|
|
|
|
dev->resources |= bit;
|
|
|
|
dprintk(1,"res: get %d\n",bit);
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
int res_check(struct cx8800_fh *fh, unsigned int bit)
|
|
|
|
{
|
|
|
|
return (fh->resources & bit);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
int res_locked(struct cx8800_dev *dev, unsigned int bit)
|
|
|
|
{
|
|
|
|
return (dev->resources & bit);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
void res_free(struct cx8800_dev *dev, struct cx8800_fh *fh, unsigned int bits)
|
|
|
|
{
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2006-03-14 00:17:11 +08:00
|
|
|
BUG_ON((fh->resources & bits) != bits);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_lock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
fh->resources &= ~bits;
|
|
|
|
dev->resources &= ~bits;
|
|
|
|
dprintk(1,"res: put %d\n",bits);
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
2007-01-21 00:58:29 +08:00
|
|
|
int cx88_video_mux(struct cx88_core *core, unsigned int input)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-09-10 04:03:41 +08:00
|
|
|
/* struct cx88_core *core = dev->core; */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
dprintk(1,"video_mux: %d [vmux=%d,gpio=0x%x,0x%x,0x%x,0x%x]\n",
|
|
|
|
input, INPUT(input)->vmux,
|
|
|
|
INPUT(input)->gpio0,INPUT(input)->gpio1,
|
|
|
|
INPUT(input)->gpio2,INPUT(input)->gpio3);
|
2005-09-10 04:03:41 +08:00
|
|
|
core->input = input;
|
2005-04-17 06:20:36 +08:00
|
|
|
cx_andor(MO_INPUT_FORMAT, 0x03 << 14, INPUT(input)->vmux << 14);
|
|
|
|
cx_write(MO_GP3_IO, INPUT(input)->gpio3);
|
|
|
|
cx_write(MO_GP0_IO, INPUT(input)->gpio0);
|
|
|
|
cx_write(MO_GP1_IO, INPUT(input)->gpio1);
|
|
|
|
cx_write(MO_GP2_IO, INPUT(input)->gpio2);
|
|
|
|
|
|
|
|
switch (INPUT(input)->type) {
|
|
|
|
case CX88_VMUX_SVIDEO:
|
|
|
|
cx_set(MO_AFECFG_IO, 0x00000001);
|
|
|
|
cx_set(MO_INPUT_FORMAT, 0x00010010);
|
|
|
|
cx_set(MO_FILTER_EVEN, 0x00002020);
|
|
|
|
cx_set(MO_FILTER_ODD, 0x00002020);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
cx_clear(MO_AFECFG_IO, 0x00000001);
|
|
|
|
cx_clear(MO_INPUT_FORMAT, 0x00010010);
|
|
|
|
cx_clear(MO_FILTER_EVEN, 0x00002020);
|
|
|
|
cx_clear(MO_FILTER_ODD, 0x00002020);
|
|
|
|
break;
|
|
|
|
}
|
2006-10-17 03:51:11 +08:00
|
|
|
|
|
|
|
if (cx88_boards[core->board].mpeg & CX88_MPEG_BLACKBIRD) {
|
|
|
|
/* sets sound input from external adc */
|
|
|
|
if (INPUT(input)->extadc)
|
|
|
|
cx_set(AUD_CTL, EN_I2SIN_ENABLE);
|
|
|
|
else
|
|
|
|
cx_clear(AUD_CTL, EN_I2SIN_ENABLE);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2007-01-21 00:58:29 +08:00
|
|
|
EXPORT_SYMBOL(cx88_video_mux);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
static int start_video_dma(struct cx8800_dev *dev,
|
|
|
|
struct cx88_dmaqueue *q,
|
|
|
|
struct cx88_buffer *buf)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
|
|
|
|
/* setup fifo + format */
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_sram_channel_setup(core, &cx88_sram_channels[SRAM_CH21],
|
2005-04-17 06:20:36 +08:00
|
|
|
buf->bpl, buf->risc.dma);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_set_scale(core, buf->vb.width, buf->vb.height, buf->vb.field);
|
2005-04-17 06:20:36 +08:00
|
|
|
cx_write(MO_COLOR_CTRL, buf->fmt->cxformat | ColorFormatGamma);
|
|
|
|
|
|
|
|
/* reset counter */
|
|
|
|
cx_write(MO_VIDY_GPCNTRL,GP_COUNT_CONTROL_RESET);
|
|
|
|
q->count = 1;
|
|
|
|
|
|
|
|
/* enable irqs */
|
|
|
|
cx_set(MO_PCI_INTMSK, core->pci_irqmask | 0x01);
|
2005-09-10 04:03:41 +08:00
|
|
|
|
|
|
|
/* Enables corresponding bits at PCI_INT_STAT:
|
|
|
|
bits 0 to 4: video, audio, transport stream, VIP, Host
|
|
|
|
bit 7: timer
|
|
|
|
bits 8 and 9: DMA complete for: SRC, DST
|
|
|
|
bits 10 and 11: BERR signal asserted for RISC: RD, WR
|
|
|
|
bits 12 to 15: BERR signal asserted for: BRDG, SRC, DST, IPB
|
|
|
|
*/
|
2005-04-17 06:20:36 +08:00
|
|
|
cx_set(MO_VID_INTMSK, 0x0f0011);
|
|
|
|
|
|
|
|
/* enable capture */
|
|
|
|
cx_set(VID_CAPTURE_CONTROL,0x06);
|
|
|
|
|
|
|
|
/* start dma */
|
|
|
|
cx_set(MO_DEV_CNTRL2, (1<<5));
|
2005-09-10 04:03:41 +08:00
|
|
|
cx_set(MO_VID_DMACNTRL, 0x11); /* Planar Y and packed FIFO and RISC enable */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-08-13 09:01:27 +08:00
|
|
|
#ifdef CONFIG_PM
|
2005-04-17 06:20:36 +08:00
|
|
|
static int stop_video_dma(struct cx8800_dev *dev)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
|
|
|
|
/* stop dma */
|
|
|
|
cx_clear(MO_VID_DMACNTRL, 0x11);
|
|
|
|
|
|
|
|
/* disable capture */
|
|
|
|
cx_clear(VID_CAPTURE_CONTROL,0x06);
|
|
|
|
|
|
|
|
/* disable irqs */
|
|
|
|
cx_clear(MO_PCI_INTMSK, 0x000001);
|
|
|
|
cx_clear(MO_VID_INTMSK, 0x0f0011);
|
|
|
|
return 0;
|
|
|
|
}
|
2006-08-13 09:01:27 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
static int restart_video_queue(struct cx8800_dev *dev,
|
|
|
|
struct cx88_dmaqueue *q)
|
|
|
|
{
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_buffer *buf, *prev;
|
|
|
|
struct list_head *item;
|
|
|
|
|
|
|
|
if (!list_empty(&q->active)) {
|
2005-11-09 13:37:43 +08:00
|
|
|
buf = list_entry(q->active.next, struct cx88_buffer, vb.queue);
|
2005-04-17 06:20:36 +08:00
|
|
|
dprintk(2,"restart_queue [%p/%d]: restart dma\n",
|
|
|
|
buf, buf->vb.i);
|
|
|
|
start_video_dma(dev, q, buf);
|
|
|
|
list_for_each(item,&q->active) {
|
|
|
|
buf = list_entry(item, struct cx88_buffer, vb.queue);
|
|
|
|
buf->count = q->count++;
|
|
|
|
}
|
|
|
|
mod_timer(&q->timeout, jiffies+BUFFER_TIMEOUT);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev = NULL;
|
|
|
|
for (;;) {
|
|
|
|
if (list_empty(&q->queued))
|
|
|
|
return 0;
|
2005-11-09 13:37:43 +08:00
|
|
|
buf = list_entry(q->queued.next, struct cx88_buffer, vb.queue);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (NULL == prev) {
|
2006-06-26 15:24:41 +08:00
|
|
|
list_move_tail(&buf->vb.queue, &q->active);
|
2005-04-17 06:20:36 +08:00
|
|
|
start_video_dma(dev, q, buf);
|
|
|
|
buf->vb.state = STATE_ACTIVE;
|
|
|
|
buf->count = q->count++;
|
|
|
|
mod_timer(&q->timeout, jiffies+BUFFER_TIMEOUT);
|
|
|
|
dprintk(2,"[%p/%d] restart_queue - first active\n",
|
|
|
|
buf,buf->vb.i);
|
|
|
|
|
|
|
|
} else if (prev->vb.width == buf->vb.width &&
|
|
|
|
prev->vb.height == buf->vb.height &&
|
|
|
|
prev->fmt == buf->fmt) {
|
2006-06-26 15:24:41 +08:00
|
|
|
list_move_tail(&buf->vb.queue, &q->active);
|
2005-04-17 06:20:36 +08:00
|
|
|
buf->vb.state = STATE_ACTIVE;
|
|
|
|
buf->count = q->count++;
|
|
|
|
prev->risc.jmp[1] = cpu_to_le32(buf->risc.dma);
|
|
|
|
dprintk(2,"[%p/%d] restart_queue - move to active\n",
|
|
|
|
buf,buf->vb.i);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
prev = buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
static int
|
|
|
|
buffer_setup(struct videobuf_queue *q, unsigned int *count, unsigned int *size)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = q->priv_data;
|
|
|
|
|
|
|
|
*size = fh->fmt->depth*fh->width*fh->height >> 3;
|
|
|
|
if (0 == *count)
|
|
|
|
*count = 32;
|
|
|
|
while (*size * *count > vid_limit * 1024 * 1024)
|
|
|
|
(*count)--;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
buffer_prepare(struct videobuf_queue *q, struct videobuf_buffer *vb,
|
|
|
|
enum v4l2_field field)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = q->priv_data;
|
|
|
|
struct cx8800_dev *dev = fh->dev;
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_buffer *buf = container_of(vb,struct cx88_buffer,vb);
|
|
|
|
int rc, init_buffer = 0;
|
|
|
|
|
|
|
|
BUG_ON(NULL == fh->fmt);
|
2005-09-10 04:03:41 +08:00
|
|
|
if (fh->width < 48 || fh->width > norm_maxw(core->tvnorm) ||
|
|
|
|
fh->height < 32 || fh->height > norm_maxh(core->tvnorm))
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
buf->vb.size = (fh->width * fh->height * fh->fmt->depth) >> 3;
|
|
|
|
if (0 != buf->vb.baddr && buf->vb.bsize < buf->vb.size)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (buf->fmt != fh->fmt ||
|
|
|
|
buf->vb.width != fh->width ||
|
|
|
|
buf->vb.height != fh->height ||
|
|
|
|
buf->vb.field != field) {
|
|
|
|
buf->fmt = fh->fmt;
|
|
|
|
buf->vb.width = fh->width;
|
|
|
|
buf->vb.height = fh->height;
|
|
|
|
buf->vb.field = field;
|
|
|
|
init_buffer = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STATE_NEEDS_INIT == buf->vb.state) {
|
|
|
|
init_buffer = 1;
|
2006-03-10 23:29:15 +08:00
|
|
|
if (0 != (rc = videobuf_iolock(q,&buf->vb,NULL)))
|
2005-04-17 06:20:36 +08:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (init_buffer) {
|
|
|
|
buf->bpl = buf->vb.width * buf->fmt->depth >> 3;
|
|
|
|
switch (buf->vb.field) {
|
|
|
|
case V4L2_FIELD_TOP:
|
|
|
|
cx88_risc_buffer(dev->pci, &buf->risc,
|
|
|
|
buf->vb.dma.sglist, 0, UNSET,
|
|
|
|
buf->bpl, 0, buf->vb.height);
|
|
|
|
break;
|
|
|
|
case V4L2_FIELD_BOTTOM:
|
|
|
|
cx88_risc_buffer(dev->pci, &buf->risc,
|
|
|
|
buf->vb.dma.sglist, UNSET, 0,
|
|
|
|
buf->bpl, 0, buf->vb.height);
|
|
|
|
break;
|
|
|
|
case V4L2_FIELD_INTERLACED:
|
|
|
|
cx88_risc_buffer(dev->pci, &buf->risc,
|
|
|
|
buf->vb.dma.sglist, 0, buf->bpl,
|
|
|
|
buf->bpl, buf->bpl,
|
|
|
|
buf->vb.height >> 1);
|
|
|
|
break;
|
|
|
|
case V4L2_FIELD_SEQ_TB:
|
|
|
|
cx88_risc_buffer(dev->pci, &buf->risc,
|
|
|
|
buf->vb.dma.sglist,
|
|
|
|
0, buf->bpl * (buf->vb.height >> 1),
|
|
|
|
buf->bpl, 0,
|
|
|
|
buf->vb.height >> 1);
|
|
|
|
break;
|
|
|
|
case V4L2_FIELD_SEQ_BT:
|
|
|
|
cx88_risc_buffer(dev->pci, &buf->risc,
|
|
|
|
buf->vb.dma.sglist,
|
|
|
|
buf->bpl * (buf->vb.height >> 1), 0,
|
|
|
|
buf->bpl, 0,
|
|
|
|
buf->vb.height >> 1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dprintk(2,"[%p/%d] buffer_prepare - %dx%d %dbpp \"%s\" - dma=0x%08lx\n",
|
|
|
|
buf, buf->vb.i,
|
|
|
|
fh->width, fh->height, fh->fmt->depth, fh->fmt->name,
|
|
|
|
(unsigned long)buf->risc.dma);
|
|
|
|
|
|
|
|
buf->vb.state = STATE_PREPARED;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2006-03-10 23:29:15 +08:00
|
|
|
cx88_free_buffer(q,buf);
|
2005-04-17 06:20:36 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
buffer_queue(struct videobuf_queue *vq, struct videobuf_buffer *vb)
|
|
|
|
{
|
|
|
|
struct cx88_buffer *buf = container_of(vb,struct cx88_buffer,vb);
|
|
|
|
struct cx88_buffer *prev;
|
|
|
|
struct cx8800_fh *fh = vq->priv_data;
|
|
|
|
struct cx8800_dev *dev = fh->dev;
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_dmaqueue *q = &dev->vidq;
|
|
|
|
|
|
|
|
/* add jump to stopper */
|
|
|
|
buf->risc.jmp[0] = cpu_to_le32(RISC_JUMP | RISC_IRQ1 | RISC_CNT_INC);
|
|
|
|
buf->risc.jmp[1] = cpu_to_le32(q->stopper.dma);
|
|
|
|
|
|
|
|
if (!list_empty(&q->queued)) {
|
|
|
|
list_add_tail(&buf->vb.queue,&q->queued);
|
|
|
|
buf->vb.state = STATE_QUEUED;
|
|
|
|
dprintk(2,"[%p/%d] buffer_queue - append to queued\n",
|
|
|
|
buf, buf->vb.i);
|
|
|
|
|
|
|
|
} else if (list_empty(&q->active)) {
|
|
|
|
list_add_tail(&buf->vb.queue,&q->active);
|
|
|
|
start_video_dma(dev, q, buf);
|
|
|
|
buf->vb.state = STATE_ACTIVE;
|
|
|
|
buf->count = q->count++;
|
|
|
|
mod_timer(&q->timeout, jiffies+BUFFER_TIMEOUT);
|
|
|
|
dprintk(2,"[%p/%d] buffer_queue - first active\n",
|
|
|
|
buf, buf->vb.i);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
prev = list_entry(q->active.prev, struct cx88_buffer, vb.queue);
|
|
|
|
if (prev->vb.width == buf->vb.width &&
|
|
|
|
prev->vb.height == buf->vb.height &&
|
|
|
|
prev->fmt == buf->fmt) {
|
|
|
|
list_add_tail(&buf->vb.queue,&q->active);
|
|
|
|
buf->vb.state = STATE_ACTIVE;
|
|
|
|
buf->count = q->count++;
|
|
|
|
prev->risc.jmp[1] = cpu_to_le32(buf->risc.dma);
|
|
|
|
dprintk(2,"[%p/%d] buffer_queue - append to active\n",
|
|
|
|
buf, buf->vb.i);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
list_add_tail(&buf->vb.queue,&q->queued);
|
|
|
|
buf->vb.state = STATE_QUEUED;
|
|
|
|
dprintk(2,"[%p/%d] buffer_queue - first queued\n",
|
|
|
|
buf, buf->vb.i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void buffer_release(struct videobuf_queue *q, struct videobuf_buffer *vb)
|
|
|
|
{
|
|
|
|
struct cx88_buffer *buf = container_of(vb,struct cx88_buffer,vb);
|
|
|
|
|
2006-03-10 23:29:15 +08:00
|
|
|
cx88_free_buffer(q,buf);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2005-05-01 23:59:29 +08:00
|
|
|
static struct videobuf_queue_ops cx8800_video_qops = {
|
2005-04-17 06:20:36 +08:00
|
|
|
.buf_setup = buffer_setup,
|
|
|
|
.buf_prepare = buffer_prepare,
|
|
|
|
.buf_queue = buffer_queue,
|
|
|
|
.buf_release = buffer_release,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
|
|
static struct videobuf_queue* get_queue(struct cx8800_fh *fh)
|
|
|
|
{
|
|
|
|
switch (fh->type) {
|
|
|
|
case V4L2_BUF_TYPE_VIDEO_CAPTURE:
|
|
|
|
return &fh->vidq;
|
|
|
|
case V4L2_BUF_TYPE_VBI_CAPTURE:
|
|
|
|
return &fh->vbiq;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int get_ressource(struct cx8800_fh *fh)
|
|
|
|
{
|
|
|
|
switch (fh->type) {
|
|
|
|
case V4L2_BUF_TYPE_VIDEO_CAPTURE:
|
|
|
|
return RESOURCE_VIDEO;
|
|
|
|
case V4L2_BUF_TYPE_VBI_CAPTURE:
|
|
|
|
return RESOURCE_VBI;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int video_open(struct inode *inode, struct file *file)
|
|
|
|
{
|
|
|
|
int minor = iminor(inode);
|
|
|
|
struct cx8800_dev *h,*dev = NULL;
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx8800_fh *fh;
|
|
|
|
struct list_head *list;
|
|
|
|
enum v4l2_buf_type type = 0;
|
|
|
|
int radio = 0;
|
|
|
|
|
|
|
|
list_for_each(list,&cx8800_devlist) {
|
|
|
|
h = list_entry(list, struct cx8800_dev, devlist);
|
|
|
|
if (h->video_dev->minor == minor) {
|
|
|
|
dev = h;
|
|
|
|
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
}
|
|
|
|
if (h->vbi_dev->minor == minor) {
|
|
|
|
dev = h;
|
|
|
|
type = V4L2_BUF_TYPE_VBI_CAPTURE;
|
|
|
|
}
|
|
|
|
if (h->radio_dev &&
|
|
|
|
h->radio_dev->minor == minor) {
|
|
|
|
radio = 1;
|
|
|
|
dev = h;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (NULL == dev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
2005-09-10 04:03:41 +08:00
|
|
|
core = dev->core;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
dprintk(1,"open minor=%d radio=%d type=%s\n",
|
|
|
|
minor,radio,v4l2_type_names[type]);
|
|
|
|
|
|
|
|
/* allocate + initialize per filehandle data */
|
2006-01-12 05:40:56 +08:00
|
|
|
fh = kzalloc(sizeof(*fh),GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (NULL == fh)
|
|
|
|
return -ENOMEM;
|
|
|
|
file->private_data = fh;
|
|
|
|
fh->dev = dev;
|
|
|
|
fh->radio = radio;
|
|
|
|
fh->type = type;
|
|
|
|
fh->width = 320;
|
|
|
|
fh->height = 240;
|
|
|
|
fh->fmt = format_by_fourcc(V4L2_PIX_FMT_BGR24);
|
|
|
|
|
|
|
|
videobuf_queue_init(&fh->vidq, &cx8800_video_qops,
|
|
|
|
dev->pci, &dev->slock,
|
|
|
|
V4L2_BUF_TYPE_VIDEO_CAPTURE,
|
|
|
|
V4L2_FIELD_INTERLACED,
|
|
|
|
sizeof(struct cx88_buffer),
|
|
|
|
fh);
|
|
|
|
videobuf_queue_init(&fh->vbiq, &cx8800_vbi_qops,
|
|
|
|
dev->pci, &dev->slock,
|
|
|
|
V4L2_BUF_TYPE_VBI_CAPTURE,
|
|
|
|
V4L2_FIELD_SEQ_TB,
|
|
|
|
sizeof(struct cx88_buffer),
|
|
|
|
fh);
|
|
|
|
|
|
|
|
if (fh->radio) {
|
|
|
|
int board = core->board;
|
|
|
|
dprintk(1,"video_open: setting radio device\n");
|
2005-08-01 13:34:43 +08:00
|
|
|
cx_write(MO_GP3_IO, cx88_boards[board].radio.gpio3);
|
2005-04-17 06:20:36 +08:00
|
|
|
cx_write(MO_GP0_IO, cx88_boards[board].radio.gpio0);
|
|
|
|
cx_write(MO_GP1_IO, cx88_boards[board].radio.gpio1);
|
|
|
|
cx_write(MO_GP2_IO, cx88_boards[board].radio.gpio2);
|
2005-09-10 04:03:41 +08:00
|
|
|
core->tvaudio = WW_FM;
|
2005-04-17 06:20:36 +08:00
|
|
|
cx88_set_tvaudio(core);
|
|
|
|
cx88_set_stereo(core,V4L2_TUNER_MODE_STEREO,1);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_call_i2c_clients(core,AUDC_SET_RADIO,NULL);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2005-11-09 13:37:43 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
2005-11-09 13:36:29 +08:00
|
|
|
video_read(struct file *file, char __user *data, size_t count, loff_t *ppos)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = file->private_data;
|
|
|
|
|
|
|
|
switch (fh->type) {
|
|
|
|
case V4L2_BUF_TYPE_VIDEO_CAPTURE:
|
|
|
|
if (res_locked(fh->dev,RESOURCE_VIDEO))
|
|
|
|
return -EBUSY;
|
|
|
|
return videobuf_read_one(&fh->vidq, data, count, ppos,
|
|
|
|
file->f_flags & O_NONBLOCK);
|
|
|
|
case V4L2_BUF_TYPE_VBI_CAPTURE:
|
|
|
|
if (!res_get(fh->dev,fh,RESOURCE_VBI))
|
|
|
|
return -EBUSY;
|
|
|
|
return videobuf_read_stream(&fh->vbiq, data, count, ppos, 1,
|
|
|
|
file->f_flags & O_NONBLOCK);
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int
|
|
|
|
video_poll(struct file *file, struct poll_table_struct *wait)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = file->private_data;
|
|
|
|
struct cx88_buffer *buf;
|
|
|
|
|
|
|
|
if (V4L2_BUF_TYPE_VBI_CAPTURE == fh->type) {
|
|
|
|
if (!res_get(fh->dev,fh,RESOURCE_VBI))
|
|
|
|
return POLLERR;
|
|
|
|
return videobuf_poll_stream(file, &fh->vbiq, wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res_check(fh,RESOURCE_VIDEO)) {
|
|
|
|
/* streaming capture */
|
|
|
|
if (list_empty(&fh->vidq.stream))
|
|
|
|
return POLLERR;
|
|
|
|
buf = list_entry(fh->vidq.stream.next,struct cx88_buffer,vb.stream);
|
|
|
|
} else {
|
|
|
|
/* read() capture */
|
|
|
|
buf = (struct cx88_buffer*)fh->vidq.read_buf;
|
|
|
|
if (NULL == buf)
|
|
|
|
return POLLERR;
|
|
|
|
}
|
|
|
|
poll_wait(file, &buf->vb.done, wait);
|
|
|
|
if (buf->vb.state == STATE_DONE ||
|
|
|
|
buf->vb.state == STATE_ERROR)
|
|
|
|
return POLLIN|POLLRDNORM;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int video_release(struct inode *inode, struct file *file)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = file->private_data;
|
|
|
|
struct cx8800_dev *dev = fh->dev;
|
|
|
|
|
|
|
|
/* turn off overlay */
|
|
|
|
if (res_check(fh, RESOURCE_OVERLAY)) {
|
|
|
|
/* FIXME */
|
|
|
|
res_free(dev,fh,RESOURCE_OVERLAY);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stop video capture */
|
|
|
|
if (res_check(fh, RESOURCE_VIDEO)) {
|
|
|
|
videobuf_queue_cancel(&fh->vidq);
|
|
|
|
res_free(dev,fh,RESOURCE_VIDEO);
|
|
|
|
}
|
|
|
|
if (fh->vidq.read_buf) {
|
|
|
|
buffer_release(&fh->vidq,fh->vidq.read_buf);
|
|
|
|
kfree(fh->vidq.read_buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stop vbi capture */
|
|
|
|
if (res_check(fh, RESOURCE_VBI)) {
|
|
|
|
if (fh->vbiq.streaming)
|
|
|
|
videobuf_streamoff(&fh->vbiq);
|
|
|
|
if (fh->vbiq.reading)
|
|
|
|
videobuf_read_stop(&fh->vbiq);
|
|
|
|
res_free(dev,fh,RESOURCE_VBI);
|
|
|
|
}
|
|
|
|
|
|
|
|
videobuf_mmap_free(&fh->vidq);
|
|
|
|
videobuf_mmap_free(&fh->vbiq);
|
|
|
|
file->private_data = NULL;
|
|
|
|
kfree(fh);
|
2005-09-10 04:03:41 +08:00
|
|
|
|
|
|
|
cx88_call_i2c_clients (dev->core, TUNER_SET_STANDBY, NULL);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
video_mmap(struct file *file, struct vm_area_struct * vma)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = file->private_data;
|
|
|
|
|
|
|
|
return videobuf_mmap_mapper(get_queue(fh), vma);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
2007-01-21 00:58:17 +08:00
|
|
|
/* VIDEO CTRL IOCTLS */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
int cx88_get_control (struct cx88_core *core, struct v4l2_control *ctl)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx88_ctrl *c = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
u32 value;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CX8800_CTLS; i++)
|
|
|
|
if (cx8800_ctls[i].v.id == ctl->id)
|
|
|
|
c = &cx8800_ctls[i];
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(NULL == c))
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
value = c->sreg ? cx_sread(c->sreg) : cx_read(c->reg);
|
|
|
|
switch (ctl->id) {
|
|
|
|
case V4L2_CID_AUDIO_BALANCE:
|
2006-03-12 11:03:47 +08:00
|
|
|
ctl->value = ((value & 0x7f) < 0x40) ? ((value & 0x7f) + 0x40)
|
|
|
|
: (0x7f - (value & 0x7f));
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
case V4L2_CID_AUDIO_VOLUME:
|
|
|
|
ctl->value = 0x3f - (value & 0x3f);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ctl->value = ((value + (c->off << c->shift)) & c->mask) >> c->shift;
|
|
|
|
break;
|
|
|
|
}
|
2006-02-27 11:09:45 +08:00
|
|
|
dprintk(1,"get_control id=0x%X(%s) ctrl=0x%02x, reg=0x%02x val=0x%02x (mask 0x%02x)%s\n",
|
|
|
|
ctl->id, c->v.name, ctl->value, c->reg,
|
|
|
|
value,c->mask, c->sreg ? " [shadowed]" : "");
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2007-01-21 00:58:26 +08:00
|
|
|
EXPORT_SYMBOL(cx88_get_control);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
int cx88_set_control(struct cx88_core *core, struct v4l2_control *ctl)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct cx88_ctrl *c = NULL;
|
2006-01-10 01:25:26 +08:00
|
|
|
u32 value,mask;
|
2005-04-17 06:20:36 +08:00
|
|
|
int i;
|
2007-01-21 00:58:17 +08:00
|
|
|
|
2006-01-10 01:25:26 +08:00
|
|
|
for (i = 0; i < CX8800_CTLS; i++) {
|
|
|
|
if (cx8800_ctls[i].v.id == ctl->id) {
|
2005-04-17 06:20:36 +08:00
|
|
|
c = &cx8800_ctls[i];
|
2006-01-10 01:25:26 +08:00
|
|
|
}
|
|
|
|
}
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(NULL == c))
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (ctl->value < c->v.minimum)
|
2005-09-10 04:03:41 +08:00
|
|
|
ctl->value = c->v.minimum;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (ctl->value > c->v.maximum)
|
2005-09-10 04:03:41 +08:00
|
|
|
ctl->value = c->v.maximum;
|
2006-01-10 01:25:26 +08:00
|
|
|
mask=c->mask;
|
2005-04-17 06:20:36 +08:00
|
|
|
switch (ctl->id) {
|
|
|
|
case V4L2_CID_AUDIO_BALANCE:
|
2006-03-12 11:03:47 +08:00
|
|
|
value = (ctl->value < 0x40) ? (0x7f - ctl->value) : (ctl->value - 0x40);
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
case V4L2_CID_AUDIO_VOLUME:
|
|
|
|
value = 0x3f - (ctl->value & 0x3f);
|
|
|
|
break;
|
|
|
|
case V4L2_CID_SATURATION:
|
|
|
|
/* special v_sat handling */
|
2006-01-10 01:25:26 +08:00
|
|
|
|
|
|
|
value = ((ctl->value - c->off) << c->shift) & c->mask;
|
|
|
|
|
2007-01-21 00:58:33 +08:00
|
|
|
if (core->tvnorm & V4L2_STD_SECAM) {
|
2006-01-10 01:25:26 +08:00
|
|
|
/* For SECAM, both U and V sat should be equal */
|
|
|
|
value=value<<8|value;
|
|
|
|
} else {
|
|
|
|
/* Keeps U Saturation proportional to V Sat */
|
|
|
|
value=(value*0x5a)/0x7f<<8|value;
|
|
|
|
}
|
|
|
|
mask=0xffff;
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
default:
|
|
|
|
value = ((ctl->value - c->off) << c->shift) & c->mask;
|
|
|
|
break;
|
|
|
|
}
|
2006-02-27 11:09:45 +08:00
|
|
|
dprintk(1,"set_control id=0x%X(%s) ctrl=0x%02x, reg=0x%02x val=0x%02x (mask 0x%02x)%s\n",
|
|
|
|
ctl->id, c->v.name, ctl->value, c->reg, value,
|
|
|
|
mask, c->sreg ? " [shadowed]" : "");
|
2005-04-17 06:20:36 +08:00
|
|
|
if (c->sreg) {
|
2006-01-10 01:25:26 +08:00
|
|
|
cx_sandor(c->sreg, c->reg, mask, value);
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
2006-01-10 01:25:26 +08:00
|
|
|
cx_andor(c->reg, mask, value);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2007-01-21 00:58:26 +08:00
|
|
|
EXPORT_SYMBOL(cx88_set_control);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-09-10 04:03:41 +08:00
|
|
|
static void init_controls(struct cx88_core *core)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-01-10 01:25:26 +08:00
|
|
|
struct v4l2_control ctrl;
|
|
|
|
int i;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-01-10 01:25:26 +08:00
|
|
|
for (i = 0; i < CX8800_CTLS; i++) {
|
|
|
|
ctrl.id=cx8800_ctls[i].v.id;
|
2006-03-12 11:03:47 +08:00
|
|
|
ctrl.value=cx8800_ctls[i].v.default_value;
|
2007-01-21 00:58:17 +08:00
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
cx88_set_control(core, &ctrl);
|
2006-01-10 01:25:26 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------ */
|
2007-01-21 00:58:17 +08:00
|
|
|
/* VIDEO IOCTLS */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_g_fmt_cap (struct file *file, void *priv,
|
|
|
|
struct v4l2_format *f)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
|
|
|
|
f->fmt.pix.width = fh->width;
|
|
|
|
f->fmt.pix.height = fh->height;
|
|
|
|
f->fmt.pix.field = fh->vidq.field;
|
|
|
|
f->fmt.pix.pixelformat = fh->fmt->fourcc;
|
|
|
|
f->fmt.pix.bytesperline =
|
|
|
|
(f->fmt.pix.width * fh->fmt->depth) >> 3;
|
|
|
|
f->fmt.pix.sizeimage =
|
|
|
|
f->fmt.pix.height * f->fmt.pix.bytesperline;
|
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_try_fmt_cap (struct file *file, void *priv,
|
|
|
|
struct v4l2_format *f)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
|
|
|
struct cx8800_fmt *fmt;
|
|
|
|
enum v4l2_field field;
|
|
|
|
unsigned int maxw, maxh;
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
fmt = format_by_fourcc(f->fmt.pix.pixelformat);
|
|
|
|
if (NULL == fmt)
|
|
|
|
return -EINVAL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
field = f->fmt.pix.field;
|
|
|
|
maxw = norm_maxw(core->tvnorm);
|
|
|
|
maxh = norm_maxh(core->tvnorm);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (V4L2_FIELD_ANY == field) {
|
|
|
|
field = (f->fmt.pix.height > maxh/2)
|
|
|
|
? V4L2_FIELD_INTERLACED
|
|
|
|
: V4L2_FIELD_BOTTOM;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2007-01-21 00:58:17 +08:00
|
|
|
|
|
|
|
switch (field) {
|
|
|
|
case V4L2_FIELD_TOP:
|
|
|
|
case V4L2_FIELD_BOTTOM:
|
|
|
|
maxh = maxh / 2;
|
|
|
|
break;
|
|
|
|
case V4L2_FIELD_INTERLACED:
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2007-01-21 00:58:17 +08:00
|
|
|
|
|
|
|
f->fmt.pix.field = field;
|
|
|
|
if (f->fmt.pix.height < 32)
|
|
|
|
f->fmt.pix.height = 32;
|
|
|
|
if (f->fmt.pix.height > maxh)
|
|
|
|
f->fmt.pix.height = maxh;
|
|
|
|
if (f->fmt.pix.width < 48)
|
|
|
|
f->fmt.pix.width = 48;
|
|
|
|
if (f->fmt.pix.width > maxw)
|
|
|
|
f->fmt.pix.width = maxw;
|
|
|
|
f->fmt.pix.width &= ~0x03;
|
|
|
|
f->fmt.pix.bytesperline =
|
|
|
|
(f->fmt.pix.width * fmt->depth) >> 3;
|
|
|
|
f->fmt.pix.sizeimage =
|
|
|
|
f->fmt.pix.height * f->fmt.pix.bytesperline;
|
|
|
|
|
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_s_fmt_cap (struct file *file, void *priv,
|
|
|
|
struct v4l2_format *f)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
int err = vidioc_try_fmt_cap (file,priv,f);
|
|
|
|
|
|
|
|
if (0 != err)
|
|
|
|
return err;
|
|
|
|
fh->fmt = format_by_fourcc(f->fmt.pix.pixelformat);
|
|
|
|
fh->width = f->fmt.pix.width;
|
|
|
|
fh->height = f->fmt.pix.height;
|
|
|
|
fh->vidq.field = f->fmt.pix.field;
|
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_querycap (struct file *file, void *priv,
|
|
|
|
struct v4l2_capability *cap)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx8800_dev *dev = ((struct cx8800_fh *)priv)->dev;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
strcpy(cap->driver, "cx8800");
|
|
|
|
strlcpy(cap->card, cx88_boards[core->board].name,
|
|
|
|
sizeof(cap->card));
|
|
|
|
sprintf(cap->bus_info,"PCI:%s",pci_name(dev->pci));
|
|
|
|
cap->version = CX88_VERSION_CODE;
|
|
|
|
cap->capabilities =
|
|
|
|
V4L2_CAP_VIDEO_CAPTURE |
|
|
|
|
V4L2_CAP_READWRITE |
|
|
|
|
V4L2_CAP_STREAMING |
|
|
|
|
V4L2_CAP_VBI_CAPTURE;
|
|
|
|
if (UNSET != core->tuner_type)
|
|
|
|
cap->capabilities |= V4L2_CAP_TUNER;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_enum_fmt_cap (struct file *file, void *priv,
|
|
|
|
struct v4l2_fmtdesc *f)
|
|
|
|
{
|
|
|
|
if (unlikely(f->index >= ARRAY_SIZE(formats)))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
strlcpy(f->description,formats[f->index].name,sizeof(f->description));
|
|
|
|
f->pixelformat = formats[f->index].fourcc;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-08-08 20:10:10 +08:00
|
|
|
#ifdef CONFIG_VIDEO_V4L1_COMPAT
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidiocgmbuf (struct file *file, void *priv, struct video_mbuf *mbuf)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
struct videobuf_queue *q;
|
|
|
|
struct v4l2_requestbuffers req;
|
|
|
|
unsigned int i;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
q = get_queue(fh);
|
|
|
|
memset(&req,0,sizeof(req));
|
|
|
|
req.type = q->type;
|
|
|
|
req.count = 8;
|
|
|
|
req.memory = V4L2_MEMORY_MMAP;
|
|
|
|
err = videobuf_reqbufs(q,&req);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
mbuf->frames = req.count;
|
|
|
|
mbuf->size = 0;
|
|
|
|
for (i = 0; i < mbuf->frames; i++) {
|
|
|
|
mbuf->offsets[i] = q->bufs[i]->boff;
|
|
|
|
mbuf->size += q->bufs[i]->bsize;
|
2005-09-10 04:03:41 +08:00
|
|
|
}
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2005-11-09 13:37:49 +08:00
|
|
|
#endif
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_reqbufs (struct file *file, void *priv, struct v4l2_requestbuffers *p)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
return (videobuf_reqbufs(get_queue(fh), p));
|
|
|
|
}
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_querybuf (struct file *file, void *priv, struct v4l2_buffer *p)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
return (videobuf_querybuf(get_queue(fh), p));
|
|
|
|
}
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_qbuf (struct file *file, void *priv, struct v4l2_buffer *p)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
return (videobuf_qbuf(get_queue(fh), p));
|
|
|
|
}
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_dqbuf (struct file *file, void *priv, struct v4l2_buffer *p)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
return (videobuf_dqbuf(get_queue(fh), p,
|
|
|
|
file->f_flags & O_NONBLOCK));
|
|
|
|
}
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_streamon(struct file *file, void *priv, enum v4l2_buf_type i)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
struct cx8800_dev *dev = fh->dev;
|
|
|
|
|
|
|
|
if (unlikely(fh->type != V4L2_BUF_TYPE_VIDEO_CAPTURE))
|
|
|
|
return -EINVAL;
|
|
|
|
if (unlikely(i != fh->type))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (unlikely(!res_get(dev,fh,get_ressource(fh))))
|
|
|
|
return -EBUSY;
|
|
|
|
return videobuf_streamon(get_queue(fh));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vidioc_streamoff(struct file *file, void *priv, enum v4l2_buf_type i)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
struct cx8800_dev *dev = fh->dev;
|
|
|
|
int err, res;
|
|
|
|
|
|
|
|
if (fh->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
|
|
|
|
return -EINVAL;
|
|
|
|
if (i != fh->type)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
res = get_ressource(fh);
|
|
|
|
err = videobuf_streamoff(get_queue(fh));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
res_free(dev,fh,res);
|
2005-09-10 04:03:41 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:33 +08:00
|
|
|
static int vidioc_s_std (struct file *file, void *priv, v4l2_std_id *tvnorms)
|
2005-09-10 04:03:41 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_lock(&core->lock);
|
2007-01-21 00:58:33 +08:00
|
|
|
cx88_set_tvnorm(core,*tvnorms);
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2007-01-21 00:58:33 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
/* only one input in this sample driver */
|
2007-01-21 00:58:26 +08:00
|
|
|
int cx88_enum_input (struct cx88_core *core,struct v4l2_input *i)
|
2007-01-21 00:58:17 +08:00
|
|
|
{
|
|
|
|
static const char *iname[] = {
|
|
|
|
[ CX88_VMUX_COMPOSITE1 ] = "Composite1",
|
|
|
|
[ CX88_VMUX_COMPOSITE2 ] = "Composite2",
|
|
|
|
[ CX88_VMUX_COMPOSITE3 ] = "Composite3",
|
|
|
|
[ CX88_VMUX_COMPOSITE4 ] = "Composite4",
|
|
|
|
[ CX88_VMUX_SVIDEO ] = "S-Video",
|
|
|
|
[ CX88_VMUX_TELEVISION ] = "Television",
|
|
|
|
[ CX88_VMUX_CABLE ] = "Cable TV",
|
|
|
|
[ CX88_VMUX_DVB ] = "DVB",
|
|
|
|
[ CX88_VMUX_DEBUG ] = "for debug only",
|
|
|
|
};
|
|
|
|
unsigned int n;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
n = i->index;
|
|
|
|
if (n >= 4)
|
|
|
|
return -EINVAL;
|
|
|
|
if (0 == INPUT(n)->type)
|
|
|
|
return -EINVAL;
|
|
|
|
memset(i,0,sizeof(*i));
|
|
|
|
i->index = n;
|
|
|
|
i->type = V4L2_INPUT_TYPE_CAMERA;
|
|
|
|
strcpy(i->name,iname[INPUT(n)->type]);
|
|
|
|
if ((CX88_VMUX_TELEVISION == INPUT(n)->type) ||
|
|
|
|
(CX88_VMUX_CABLE == INPUT(n)->type))
|
|
|
|
i->type = V4L2_INPUT_TYPE_TUNER;
|
2007-01-21 00:58:33 +08:00
|
|
|
i->std = CX88_NORMS;
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2007-01-21 00:58:26 +08:00
|
|
|
EXPORT_SYMBOL(cx88_enum_input);
|
|
|
|
|
|
|
|
static int vidioc_enum_input (struct file *file, void *priv,
|
|
|
|
struct v4l2_input *i)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
|
|
|
return cx88_enum_input (core,i);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_g_input (struct file *file, void *priv, unsigned int *i)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
*i = core->input;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_s_input (struct file *file, void *priv, unsigned int i)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (i >= 4)
|
|
|
|
return -EINVAL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_lock(&core->lock);
|
|
|
|
cx88_newstation(core);
|
2007-01-21 00:58:29 +08:00
|
|
|
cx88_video_mux(core,i);
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_unlock(&core->lock);
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_queryctrl (struct file *file, void *priv,
|
|
|
|
struct v4l2_queryctrl *qctrl)
|
|
|
|
{
|
|
|
|
qctrl->id = v4l2_ctrl_next(ctrl_classes, qctrl->id);
|
|
|
|
if (unlikely(qctrl->id == 0))
|
|
|
|
return -EINVAL;
|
|
|
|
return cx8800_ctrl_query(qctrl);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
static int vidioc_g_ctrl (struct file *file, void *priv,
|
2007-01-21 00:58:17 +08:00
|
|
|
struct v4l2_control *ctl)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2007-01-21 00:58:26 +08:00
|
|
|
return
|
|
|
|
cx88_get_control(core,ctl);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
static int vidioc_s_ctrl (struct file *file, void *priv,
|
|
|
|
struct v4l2_control *ctl)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2007-01-21 00:58:17 +08:00
|
|
|
return
|
2007-01-21 00:58:26 +08:00
|
|
|
cx88_set_control(core,ctl);
|
2007-01-21 00:58:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vidioc_g_tuner (struct file *file, void *priv,
|
|
|
|
struct v4l2_tuner *t)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
|
|
|
u32 reg;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(UNSET == core->tuner_type))
|
|
|
|
return -EINVAL;
|
2007-01-21 00:58:36 +08:00
|
|
|
if (0 != t->index)
|
|
|
|
return -EINVAL;
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
strcpy(t->name, "Television");
|
|
|
|
t->type = V4L2_TUNER_ANALOG_TV;
|
|
|
|
t->capability = V4L2_TUNER_CAP_NORM;
|
|
|
|
t->rangehigh = 0xffffffffUL;
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
cx88_get_stereo(core ,t);
|
|
|
|
reg = cx_read(MO_DEVICE_STATUS);
|
|
|
|
t->signal = (reg & (1<<5)) ? 0xffff : 0x0000;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-07-13 04:58:44 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_s_tuner (struct file *file, void *priv,
|
|
|
|
struct v4l2_tuner *t)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-07-13 04:58:44 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (UNSET == core->tuner_type)
|
|
|
|
return -EINVAL;
|
|
|
|
if (0 != t->index)
|
|
|
|
return -EINVAL;
|
2005-07-15 18:56:28 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
cx88_set_stereo(core, t->audmode, 1);
|
|
|
|
return 0;
|
|
|
|
}
|
2006-10-05 07:33:51 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int vidioc_g_frequency (struct file *file, void *priv,
|
|
|
|
struct v4l2_frequency *f)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
struct cx88_core *core = fh->dev->core;
|
2006-10-05 07:33:51 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(UNSET == core->tuner_type))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* f->type = fh->radio ? V4L2_TUNER_RADIO : V4L2_TUNER_ANALOG_TV; */
|
|
|
|
f->type = fh->radio ? V4L2_TUNER_RADIO : V4L2_TUNER_ANALOG_TV;
|
|
|
|
f->frequency = core->freq;
|
|
|
|
|
|
|
|
cx88_call_i2c_clients(core,VIDIOC_G_FREQUENCY,f);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:26 +08:00
|
|
|
int cx88_set_freq (struct cx88_core *core,
|
2007-01-21 00:58:17 +08:00
|
|
|
struct v4l2_frequency *f)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(UNSET == core->tuner_type))
|
|
|
|
return -EINVAL;
|
|
|
|
if (unlikely(f->tuner != 0))
|
|
|
|
return -EINVAL;
|
2007-01-21 00:58:26 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_lock(&core->lock);
|
|
|
|
core->freq = f->frequency;
|
|
|
|
cx88_newstation(core);
|
|
|
|
cx88_call_i2c_clients(core,VIDIOC_S_FREQUENCY,f);
|
2006-03-10 23:29:15 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
/* When changing channels it is required to reset TVAUDIO */
|
|
|
|
msleep (10);
|
|
|
|
cx88_set_tvaudio(core);
|
2006-03-10 23:29:15 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2007-01-21 00:58:26 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2007-01-21 00:58:26 +08:00
|
|
|
EXPORT_SYMBOL(cx88_set_freq);
|
|
|
|
|
|
|
|
static int vidioc_s_frequency (struct file *file, void *priv,
|
|
|
|
struct v4l2_frequency *f)
|
|
|
|
{
|
|
|
|
struct cx8800_fh *fh = priv;
|
|
|
|
struct cx88_core *core = fh->dev->core;
|
|
|
|
|
|
|
|
if (unlikely(0 == fh->radio && f->type != V4L2_TUNER_ANALOG_TV))
|
|
|
|
return -EINVAL;
|
|
|
|
if (unlikely(1 == fh->radio && f->type != V4L2_TUNER_RADIO))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return
|
|
|
|
cx88_set_freq (core,f);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-23 10:31:53 +08:00
|
|
|
#ifdef CONFIG_VIDEO_ADV_DEBUG
|
|
|
|
static int vidioc_g_register (struct file *file, void *fh,
|
|
|
|
struct v4l2_register *reg)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh*)fh)->dev->core;
|
|
|
|
|
2007-02-24 07:55:14 +08:00
|
|
|
if (!v4l2_chip_match_host(reg->match_type, reg->match_chip))
|
2007-01-23 10:31:53 +08:00
|
|
|
return -EINVAL;
|
|
|
|
/* cx2388x has a 24-bit register space */
|
|
|
|
reg->val = cx_read(reg->reg&0xffffff);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vidioc_s_register (struct file *file, void *fh,
|
|
|
|
struct v4l2_register *reg)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh*)fh)->dev->core;
|
|
|
|
|
2007-02-24 07:55:14 +08:00
|
|
|
if (!v4l2_chip_match_host(reg->match_type, reg->match_chip))
|
2007-01-23 10:31:53 +08:00
|
|
|
return -EINVAL;
|
|
|
|
cx_write(reg->reg&0xffffff, reg->val);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
2007-01-21 00:58:17 +08:00
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
/* RADIO ESPECIFIC IOCTLS */
|
2005-04-17 06:20:36 +08:00
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_querycap (struct file *file, void *priv,
|
|
|
|
struct v4l2_capability *cap)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
struct cx8800_dev *dev = ((struct cx8800_fh *)priv)->dev;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
strcpy(cap->driver, "cx8800");
|
|
|
|
strlcpy(cap->card, cx88_boards[core->board].name,
|
|
|
|
sizeof(cap->card));
|
|
|
|
sprintf(cap->bus_info,"PCI:%s", pci_name(dev->pci));
|
|
|
|
cap->version = CX88_VERSION_CODE;
|
|
|
|
cap->capabilities = V4L2_CAP_TUNER;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_g_tuner (struct file *file, void *priv,
|
|
|
|
struct v4l2_tuner *t)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (unlikely(t->index > 0))
|
|
|
|
return -EINVAL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
strcpy(t->name, "Radio");
|
|
|
|
t->type = V4L2_TUNER_RADIO;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
cx88_call_i2c_clients(core,VIDIOC_G_TUNER,t);
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_enum_input (struct file *file, void *priv,
|
|
|
|
struct v4l2_input *i)
|
|
|
|
{
|
|
|
|
if (i->index != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
strcpy(i->name,"Radio");
|
|
|
|
i->type = V4L2_INPUT_TYPE_TUNER;
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_g_audio (struct file *file, void *priv, struct v4l2_audio *a)
|
|
|
|
{
|
|
|
|
if (unlikely(a->index))
|
|
|
|
return -EINVAL;
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
memset(a,0,sizeof(*a));
|
|
|
|
strcpy(a->name,"Radio");
|
|
|
|
return 0;
|
|
|
|
}
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
/* FIXME: Should add a standard for radio */
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_s_tuner (struct file *file, void *priv,
|
|
|
|
struct v4l2_tuner *t)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = ((struct cx8800_fh *)priv)->dev->core;
|
2005-07-08 08:58:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
if (0 != t->index)
|
|
|
|
return -EINVAL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
cx88_call_i2c_clients(core,VIDIOC_S_TUNER,t);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_s_audio (struct file *file, void *fh,
|
|
|
|
struct v4l2_audio *a)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_s_input (struct file *file, void *fh, unsigned int i)
|
|
|
|
{
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
2007-01-21 00:58:17 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static int radio_queryctrl (struct file *file, void *priv,
|
|
|
|
struct v4l2_queryctrl *c)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if (c->id < V4L2_CID_BASE ||
|
|
|
|
c->id >= V4L2_CID_LASTP1)
|
|
|
|
return -EINVAL;
|
|
|
|
if (c->id == V4L2_CID_AUDIO_MUTE) {
|
|
|
|
for (i = 0; i < CX8800_CTLS; i++)
|
|
|
|
if (cx8800_ctls[i].v.id == c->id)
|
|
|
|
break;
|
|
|
|
*c = cx8800_ctls[i].v;
|
|
|
|
} else
|
|
|
|
*c = no_ctl;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
|
|
|
|
static void cx8800_vid_timeout(unsigned long data)
|
|
|
|
{
|
|
|
|
struct cx8800_dev *dev = (struct cx8800_dev*)data;
|
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
struct cx88_dmaqueue *q = &dev->vidq;
|
|
|
|
struct cx88_buffer *buf;
|
|
|
|
unsigned long flags;
|
|
|
|
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_sram_channel_dump(core, &cx88_sram_channels[SRAM_CH21]);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
cx_clear(MO_VID_DMACNTRL, 0x11);
|
|
|
|
cx_clear(VID_CAPTURE_CONTROL, 0x06);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&dev->slock,flags);
|
|
|
|
while (!list_empty(&q->active)) {
|
|
|
|
buf = list_entry(q->active.next, struct cx88_buffer, vb.queue);
|
|
|
|
list_del(&buf->vb.queue);
|
|
|
|
buf->vb.state = STATE_ERROR;
|
|
|
|
wake_up(&buf->vb.done);
|
|
|
|
printk("%s/0: [%p/%d] timeout - dma=0x%08lx\n", core->name,
|
|
|
|
buf, buf->vb.i, (unsigned long)buf->risc.dma);
|
|
|
|
}
|
|
|
|
restart_video_queue(dev,q);
|
|
|
|
spin_unlock_irqrestore(&dev->slock,flags);
|
|
|
|
}
|
|
|
|
|
2005-07-13 04:58:44 +08:00
|
|
|
static char *cx88_vid_irqs[32] = {
|
|
|
|
"y_risci1", "u_risci1", "v_risci1", "vbi_risc1",
|
|
|
|
"y_risci2", "u_risci2", "v_risci2", "vbi_risc2",
|
|
|
|
"y_oflow", "u_oflow", "v_oflow", "vbi_oflow",
|
|
|
|
"y_sync", "u_sync", "v_sync", "vbi_sync",
|
|
|
|
"opc_err", "par_err", "rip_err", "pci_abort",
|
|
|
|
};
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
static void cx8800_vid_irq(struct cx8800_dev *dev)
|
|
|
|
{
|
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
u32 status, mask, count;
|
|
|
|
|
|
|
|
status = cx_read(MO_VID_INTSTAT);
|
|
|
|
mask = cx_read(MO_VID_INTMSK);
|
|
|
|
if (0 == (status & mask))
|
|
|
|
return;
|
|
|
|
cx_write(MO_VID_INTSTAT, status);
|
|
|
|
if (irq_debug || (status & mask & ~0xff))
|
|
|
|
cx88_print_irqbits(core->name, "irq vid",
|
|
|
|
cx88_vid_irqs, status, mask);
|
|
|
|
|
|
|
|
/* risc op code error */
|
|
|
|
if (status & (1 << 16)) {
|
|
|
|
printk(KERN_WARNING "%s/0: video risc op code error\n",core->name);
|
|
|
|
cx_clear(MO_VID_DMACNTRL, 0x11);
|
|
|
|
cx_clear(VID_CAPTURE_CONTROL, 0x06);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_sram_channel_dump(core, &cx88_sram_channels[SRAM_CH21]);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* risc1 y */
|
|
|
|
if (status & 0x01) {
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
count = cx_read(MO_VIDY_GPCNT);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_wakeup(core, &dev->vidq, count);
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* risc1 vbi */
|
|
|
|
if (status & 0x08) {
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
count = cx_read(MO_VBI_GPCNT);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_wakeup(core, &dev->vbiq, count);
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* risc2 y */
|
|
|
|
if (status & 0x10) {
|
|
|
|
dprintk(2,"stopper video\n");
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
restart_video_queue(dev,&dev->vidq);
|
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* risc2 vbi */
|
|
|
|
if (status & 0x80) {
|
|
|
|
dprintk(2,"stopper vbi\n");
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
cx8800_restart_vbi_queue(dev,&dev->vbiq);
|
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 21:55:46 +08:00
|
|
|
static irqreturn_t cx8800_irq(int irq, void *dev_id)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct cx8800_dev *dev = dev_id;
|
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
u32 status;
|
|
|
|
int loop, handled = 0;
|
|
|
|
|
|
|
|
for (loop = 0; loop < 10; loop++) {
|
|
|
|
status = cx_read(MO_PCI_INTSTAT) & (core->pci_irqmask | 0x01);
|
|
|
|
if (0 == status)
|
|
|
|
goto out;
|
|
|
|
cx_write(MO_PCI_INTSTAT, status);
|
|
|
|
handled = 1;
|
|
|
|
|
|
|
|
if (status & core->pci_irqmask)
|
|
|
|
cx88_core_irq(core,status);
|
|
|
|
if (status & 0x01)
|
|
|
|
cx8800_vid_irq(dev);
|
|
|
|
};
|
|
|
|
if (10 == loop) {
|
|
|
|
printk(KERN_WARNING "%s/0: irq loop -- clearing mask\n",
|
|
|
|
core->name);
|
|
|
|
cx_write(MO_PCI_INTMSK,0);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return IRQ_RETVAL(handled);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
/* exported stuff */
|
|
|
|
|
2007-02-12 16:55:33 +08:00
|
|
|
static const struct file_operations video_fops =
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = video_open,
|
|
|
|
.release = video_release,
|
|
|
|
.read = video_read,
|
|
|
|
.poll = video_poll,
|
|
|
|
.mmap = video_mmap,
|
2007-01-21 00:58:17 +08:00
|
|
|
.ioctl = video_ioctl2,
|
2006-01-10 01:24:57 +08:00
|
|
|
.compat_ioctl = v4l_compat_ioctl32,
|
2005-04-17 06:20:36 +08:00
|
|
|
.llseek = no_llseek,
|
|
|
|
};
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
static struct video_device cx8800_vbi_template;
|
2005-05-01 23:59:29 +08:00
|
|
|
static struct video_device cx8800_video_template =
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
.name = "cx8800-video",
|
|
|
|
.type = VID_TYPE_CAPTURE|VID_TYPE_TUNER|VID_TYPE_SCALES,
|
|
|
|
.fops = &video_fops,
|
|
|
|
.minor = -1,
|
|
|
|
.vidioc_querycap = vidioc_querycap,
|
|
|
|
.vidioc_enum_fmt_cap = vidioc_enum_fmt_cap,
|
|
|
|
.vidioc_g_fmt_cap = vidioc_g_fmt_cap,
|
|
|
|
.vidioc_try_fmt_cap = vidioc_try_fmt_cap,
|
|
|
|
.vidioc_s_fmt_cap = vidioc_s_fmt_cap,
|
|
|
|
.vidioc_g_fmt_vbi = cx8800_vbi_fmt,
|
|
|
|
.vidioc_try_fmt_vbi = cx8800_vbi_fmt,
|
|
|
|
.vidioc_s_fmt_vbi = cx8800_vbi_fmt,
|
|
|
|
.vidioc_reqbufs = vidioc_reqbufs,
|
|
|
|
.vidioc_querybuf = vidioc_querybuf,
|
|
|
|
.vidioc_qbuf = vidioc_qbuf,
|
|
|
|
.vidioc_dqbuf = vidioc_dqbuf,
|
|
|
|
.vidioc_s_std = vidioc_s_std,
|
|
|
|
.vidioc_enum_input = vidioc_enum_input,
|
|
|
|
.vidioc_g_input = vidioc_g_input,
|
|
|
|
.vidioc_s_input = vidioc_s_input,
|
|
|
|
.vidioc_queryctrl = vidioc_queryctrl,
|
|
|
|
.vidioc_g_ctrl = vidioc_g_ctrl,
|
|
|
|
.vidioc_s_ctrl = vidioc_s_ctrl,
|
|
|
|
.vidioc_streamon = vidioc_streamon,
|
|
|
|
.vidioc_streamoff = vidioc_streamoff,
|
|
|
|
#ifdef CONFIG_VIDEO_V4L1_COMPAT
|
|
|
|
.vidiocgmbuf = vidiocgmbuf,
|
|
|
|
#endif
|
|
|
|
.vidioc_g_tuner = vidioc_g_tuner,
|
|
|
|
.vidioc_s_tuner = vidioc_s_tuner,
|
|
|
|
.vidioc_g_frequency = vidioc_g_frequency,
|
|
|
|
.vidioc_s_frequency = vidioc_s_frequency,
|
2007-01-23 10:31:53 +08:00
|
|
|
#ifdef CONFIG_VIDEO_ADV_DEBUG
|
|
|
|
.vidioc_g_register = vidioc_g_register,
|
|
|
|
.vidioc_s_register = vidioc_s_register,
|
|
|
|
#endif
|
2007-01-21 00:58:33 +08:00
|
|
|
.tvnorms = CX88_NORMS,
|
2007-01-23 10:31:53 +08:00
|
|
|
.current_norm = V4L2_STD_NTSC_M,
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2007-02-12 16:55:33 +08:00
|
|
|
static const struct file_operations radio_fops =
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = video_open,
|
|
|
|
.release = video_release,
|
2007-01-21 00:58:17 +08:00
|
|
|
.ioctl = video_ioctl2,
|
2006-01-10 01:24:57 +08:00
|
|
|
.compat_ioctl = v4l_compat_ioctl32,
|
2005-04-17 06:20:36 +08:00
|
|
|
.llseek = no_llseek,
|
|
|
|
};
|
|
|
|
|
2005-05-01 23:59:29 +08:00
|
|
|
static struct video_device cx8800_radio_template =
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2007-01-21 00:58:17 +08:00
|
|
|
.name = "cx8800-radio",
|
|
|
|
.type = VID_TYPE_TUNER,
|
|
|
|
.hardware = 0,
|
|
|
|
.fops = &radio_fops,
|
|
|
|
.minor = -1,
|
|
|
|
.vidioc_querycap = radio_querycap,
|
|
|
|
.vidioc_g_tuner = radio_g_tuner,
|
|
|
|
.vidioc_enum_input = radio_enum_input,
|
|
|
|
.vidioc_g_audio = radio_g_audio,
|
|
|
|
.vidioc_s_tuner = radio_s_tuner,
|
|
|
|
.vidioc_s_audio = radio_s_audio,
|
|
|
|
.vidioc_s_input = radio_s_input,
|
|
|
|
.vidioc_queryctrl = radio_queryctrl,
|
|
|
|
.vidioc_g_ctrl = vidioc_g_ctrl,
|
|
|
|
.vidioc_s_ctrl = vidioc_s_ctrl,
|
|
|
|
.vidioc_g_frequency = vidioc_g_frequency,
|
|
|
|
.vidioc_s_frequency = vidioc_s_frequency,
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
|
|
|
|
static void cx8800_unregister_video(struct cx8800_dev *dev)
|
|
|
|
{
|
|
|
|
if (dev->radio_dev) {
|
|
|
|
if (-1 != dev->radio_dev->minor)
|
|
|
|
video_unregister_device(dev->radio_dev);
|
|
|
|
else
|
|
|
|
video_device_release(dev->radio_dev);
|
|
|
|
dev->radio_dev = NULL;
|
|
|
|
}
|
|
|
|
if (dev->vbi_dev) {
|
|
|
|
if (-1 != dev->vbi_dev->minor)
|
|
|
|
video_unregister_device(dev->vbi_dev);
|
|
|
|
else
|
|
|
|
video_device_release(dev->vbi_dev);
|
|
|
|
dev->vbi_dev = NULL;
|
|
|
|
}
|
|
|
|
if (dev->video_dev) {
|
|
|
|
if (-1 != dev->video_dev->minor)
|
|
|
|
video_unregister_device(dev->video_dev);
|
|
|
|
else
|
|
|
|
video_device_release(dev->video_dev);
|
|
|
|
dev->video_dev = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __devinit cx8800_initdev(struct pci_dev *pci_dev,
|
|
|
|
const struct pci_device_id *pci_id)
|
|
|
|
{
|
|
|
|
struct cx8800_dev *dev;
|
|
|
|
struct cx88_core *core;
|
2007-01-21 00:58:17 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
int err;
|
|
|
|
|
2006-01-12 05:40:56 +08:00
|
|
|
dev = kzalloc(sizeof(*dev),GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (NULL == dev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
/* pci init */
|
|
|
|
dev->pci = pci_dev;
|
|
|
|
if (pci_enable_device(pci_dev)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto fail_free;
|
|
|
|
}
|
|
|
|
core = cx88_core_get(dev->pci);
|
|
|
|
if (NULL == core) {
|
|
|
|
err = -EINVAL;
|
|
|
|
goto fail_free;
|
|
|
|
}
|
|
|
|
dev->core = core;
|
|
|
|
|
|
|
|
/* print pci info */
|
|
|
|
pci_read_config_byte(pci_dev, PCI_CLASS_REVISION, &dev->pci_rev);
|
2005-11-09 13:37:43 +08:00
|
|
|
pci_read_config_byte(pci_dev, PCI_LATENCY_TIMER, &dev->pci_lat);
|
|
|
|
printk(KERN_INFO "%s/0: found at %s, rev: %d, irq: %d, "
|
2006-06-13 06:16:52 +08:00
|
|
|
"latency: %d, mmio: 0x%llx\n", core->name,
|
2005-04-17 06:20:36 +08:00
|
|
|
pci_name(pci_dev), dev->pci_rev, pci_dev->irq,
|
2006-06-13 06:16:52 +08:00
|
|
|
dev->pci_lat,(unsigned long long)pci_resource_start(pci_dev,0));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
pci_set_master(pci_dev);
|
|
|
|
if (!pci_dma_supported(pci_dev,0xffffffff)) {
|
|
|
|
printk("%s/0: Oops: no 32bit PCI DMA ???\n",core->name);
|
|
|
|
err = -EIO;
|
|
|
|
goto fail_core;
|
|
|
|
}
|
|
|
|
|
2007-01-21 00:58:17 +08:00
|
|
|
/* Initialize VBI template */
|
|
|
|
memcpy( &cx8800_vbi_template, &cx8800_video_template,
|
|
|
|
sizeof(cx8800_vbi_template) );
|
|
|
|
strcpy(cx8800_vbi_template.name,"cx8800-vbi");
|
|
|
|
cx8800_vbi_template.type = VID_TYPE_TELETEXT|VID_TYPE_TUNER;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* initialize driver struct */
|
|
|
|
spin_lock_init(&dev->slock);
|
2007-01-21 00:58:33 +08:00
|
|
|
core->tvnorm = cx8800_video_template.current_norm;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* init video dma queues */
|
|
|
|
INIT_LIST_HEAD(&dev->vidq.active);
|
|
|
|
INIT_LIST_HEAD(&dev->vidq.queued);
|
|
|
|
dev->vidq.timeout.function = cx8800_vid_timeout;
|
|
|
|
dev->vidq.timeout.data = (unsigned long)dev;
|
|
|
|
init_timer(&dev->vidq.timeout);
|
|
|
|
cx88_risc_stopper(dev->pci,&dev->vidq.stopper,
|
|
|
|
MO_VID_DMACNTRL,0x11,0x00);
|
|
|
|
|
|
|
|
/* init vbi dma queues */
|
|
|
|
INIT_LIST_HEAD(&dev->vbiq.active);
|
|
|
|
INIT_LIST_HEAD(&dev->vbiq.queued);
|
|
|
|
dev->vbiq.timeout.function = cx8800_vbi_timeout;
|
|
|
|
dev->vbiq.timeout.data = (unsigned long)dev;
|
|
|
|
init_timer(&dev->vbiq.timeout);
|
|
|
|
cx88_risc_stopper(dev->pci,&dev->vbiq.stopper,
|
|
|
|
MO_VID_DMACNTRL,0x88,0x00);
|
|
|
|
|
|
|
|
/* get irq */
|
|
|
|
err = request_irq(pci_dev->irq, cx8800_irq,
|
2006-07-02 10:29:37 +08:00
|
|
|
IRQF_SHARED | IRQF_DISABLED, core->name, dev);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (err < 0) {
|
|
|
|
printk(KERN_ERR "%s: can't get IRQ %d\n",
|
|
|
|
core->name,pci_dev->irq);
|
|
|
|
goto fail_core;
|
|
|
|
}
|
|
|
|
cx_set(MO_PCI_INTMSK, core->pci_irqmask);
|
|
|
|
|
|
|
|
/* load and configure helper modules */
|
|
|
|
if (TUNER_ABSENT != core->tuner_type)
|
|
|
|
request_module("tuner");
|
2005-09-10 04:03:41 +08:00
|
|
|
|
2006-09-25 23:43:42 +08:00
|
|
|
if (cx88_boards[ core->board ].audio_chip == AUDIO_CHIP_WM8775)
|
|
|
|
request_module("wm8775");
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* register v4l devices */
|
|
|
|
dev->video_dev = cx88_vdev_init(core,dev->pci,
|
|
|
|
&cx8800_video_template,"video");
|
|
|
|
err = video_register_device(dev->video_dev,VFL_TYPE_GRABBER,
|
|
|
|
video_nr[core->nr]);
|
|
|
|
if (err < 0) {
|
|
|
|
printk(KERN_INFO "%s: can't register video device\n",
|
|
|
|
core->name);
|
|
|
|
goto fail_unreg;
|
|
|
|
}
|
|
|
|
printk(KERN_INFO "%s/0: registered device video%d [v4l2]\n",
|
|
|
|
core->name,dev->video_dev->minor & 0x1f);
|
|
|
|
|
|
|
|
dev->vbi_dev = cx88_vdev_init(core,dev->pci,&cx8800_vbi_template,"vbi");
|
|
|
|
err = video_register_device(dev->vbi_dev,VFL_TYPE_VBI,
|
|
|
|
vbi_nr[core->nr]);
|
|
|
|
if (err < 0) {
|
|
|
|
printk(KERN_INFO "%s/0: can't register vbi device\n",
|
|
|
|
core->name);
|
|
|
|
goto fail_unreg;
|
|
|
|
}
|
|
|
|
printk(KERN_INFO "%s/0: registered device vbi%d\n",
|
|
|
|
core->name,dev->vbi_dev->minor & 0x1f);
|
|
|
|
|
|
|
|
if (core->has_radio) {
|
|
|
|
dev->radio_dev = cx88_vdev_init(core,dev->pci,
|
|
|
|
&cx8800_radio_template,"radio");
|
|
|
|
err = video_register_device(dev->radio_dev,VFL_TYPE_RADIO,
|
|
|
|
radio_nr[core->nr]);
|
|
|
|
if (err < 0) {
|
|
|
|
printk(KERN_INFO "%s/0: can't register radio device\n",
|
|
|
|
core->name);
|
|
|
|
goto fail_unreg;
|
|
|
|
}
|
|
|
|
printk(KERN_INFO "%s/0: registered device radio%d\n",
|
|
|
|
core->name,dev->radio_dev->minor & 0x1f);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* everything worked */
|
|
|
|
list_add_tail(&dev->devlist,&cx8800_devlist);
|
|
|
|
pci_set_drvdata(pci_dev,dev);
|
|
|
|
|
|
|
|
/* initial device configuration */
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_lock(&core->lock);
|
2007-01-21 00:58:33 +08:00
|
|
|
cx88_set_tvnorm(core,core->tvnorm);
|
2006-01-10 01:25:26 +08:00
|
|
|
init_controls(core);
|
2007-01-21 00:58:29 +08:00
|
|
|
cx88_video_mux(core,0);
|
2006-02-07 16:49:14 +08:00
|
|
|
mutex_unlock(&core->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* start tvaudio thread */
|
|
|
|
if (core->tuner_type != TUNER_ABSENT)
|
|
|
|
core->kthread = kthread_run(cx88_audio_thread, core, "cx88 tvaudio");
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail_unreg:
|
|
|
|
cx8800_unregister_video(dev);
|
|
|
|
free_irq(pci_dev->irq, dev);
|
|
|
|
fail_core:
|
|
|
|
cx88_core_put(core,dev->pci);
|
|
|
|
fail_free:
|
|
|
|
kfree(dev);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __devexit cx8800_finidev(struct pci_dev *pci_dev)
|
|
|
|
{
|
2005-11-09 13:37:43 +08:00
|
|
|
struct cx8800_dev *dev = pci_get_drvdata(pci_dev);
|
2005-09-10 04:03:41 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* stop thread */
|
2005-09-10 04:03:41 +08:00
|
|
|
if (core->kthread) {
|
|
|
|
kthread_stop(core->kthread);
|
|
|
|
core->kthread = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_shutdown(core); /* FIXME */
|
2005-04-17 06:20:36 +08:00
|
|
|
pci_disable_device(pci_dev);
|
|
|
|
|
|
|
|
/* unregister stuff */
|
|
|
|
|
|
|
|
free_irq(pci_dev->irq, dev);
|
|
|
|
cx8800_unregister_video(dev);
|
|
|
|
pci_set_drvdata(pci_dev, NULL);
|
|
|
|
|
|
|
|
/* free memory */
|
|
|
|
btcx_riscmem_free(dev->pci,&dev->vidq.stopper);
|
|
|
|
list_del(&dev->devlist);
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_core_put(core,dev->pci);
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(dev);
|
|
|
|
}
|
|
|
|
|
2006-08-13 09:01:27 +08:00
|
|
|
#ifdef CONFIG_PM
|
2005-04-17 06:20:36 +08:00
|
|
|
static int cx8800_suspend(struct pci_dev *pci_dev, pm_message_t state)
|
|
|
|
{
|
2005-06-24 13:05:03 +08:00
|
|
|
struct cx8800_dev *dev = pci_get_drvdata(pci_dev);
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
|
|
|
|
|
|
|
/* stop video+vbi capture */
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
if (!list_empty(&dev->vidq.active)) {
|
|
|
|
printk("%s: suspend video\n", core->name);
|
|
|
|
stop_video_dma(dev);
|
|
|
|
del_timer(&dev->vidq.timeout);
|
|
|
|
}
|
|
|
|
if (!list_empty(&dev->vbiq.active)) {
|
|
|
|
printk("%s: suspend vbi\n", core->name);
|
|
|
|
cx8800_stop_vbi_dma(dev);
|
|
|
|
del_timer(&dev->vbiq.timeout);
|
|
|
|
}
|
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
|
|
|
|
/* FIXME -- shutdown device */
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_shutdown(core);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
pci_save_state(pci_dev);
|
|
|
|
if (0 != pci_set_power_state(pci_dev, pci_choose_state(pci_dev, state))) {
|
|
|
|
pci_disable_device(pci_dev);
|
|
|
|
dev->state.disabled = 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int cx8800_resume(struct pci_dev *pci_dev)
|
|
|
|
{
|
2005-06-24 13:05:03 +08:00
|
|
|
struct cx8800_dev *dev = pci_get_drvdata(pci_dev);
|
2005-04-17 06:20:36 +08:00
|
|
|
struct cx88_core *core = dev->core;
|
2005-09-10 04:03:55 +08:00
|
|
|
int err;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (dev->state.disabled) {
|
2005-09-10 04:03:55 +08:00
|
|
|
err=pci_enable_device(pci_dev);
|
|
|
|
if (err) {
|
|
|
|
printk(KERN_ERR "%s: can't enable device\n",
|
|
|
|
core->name);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->state.disabled = 0;
|
|
|
|
}
|
2005-09-10 04:03:55 +08:00
|
|
|
err= pci_set_power_state(pci_dev, PCI_D0);
|
|
|
|
if (err) {
|
|
|
|
printk(KERN_ERR "%s: can't enable device\n",
|
|
|
|
core->name);
|
|
|
|
|
|
|
|
pci_disable_device(pci_dev);
|
|
|
|
dev->state.disabled = 1;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
pci_restore_state(pci_dev);
|
|
|
|
|
|
|
|
/* FIXME: re-initialize hardware */
|
2005-09-10 04:03:41 +08:00
|
|
|
cx88_reset(core);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* restart video+vbi capture */
|
|
|
|
spin_lock(&dev->slock);
|
|
|
|
if (!list_empty(&dev->vidq.active)) {
|
|
|
|
printk("%s: resume video\n", core->name);
|
|
|
|
restart_video_queue(dev,&dev->vidq);
|
|
|
|
}
|
|
|
|
if (!list_empty(&dev->vbiq.active)) {
|
|
|
|
printk("%s: resume vbi\n", core->name);
|
|
|
|
cx8800_restart_vbi_queue(dev,&dev->vbiq);
|
|
|
|
}
|
|
|
|
spin_unlock(&dev->slock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2006-08-13 09:01:27 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
|
2005-05-01 23:59:29 +08:00
|
|
|
static struct pci_device_id cx8800_pci_tbl[] = {
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
.vendor = 0x14f1,
|
|
|
|
.device = 0x8800,
|
2005-06-24 13:05:03 +08:00
|
|
|
.subvendor = PCI_ANY_ID,
|
|
|
|
.subdevice = PCI_ANY_ID,
|
2005-04-17 06:20:36 +08:00
|
|
|
},{
|
|
|
|
/* --- end of list --- */
|
|
|
|
}
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(pci, cx8800_pci_tbl);
|
|
|
|
|
|
|
|
static struct pci_driver cx8800_pci_driver = {
|
2005-06-24 13:05:03 +08:00
|
|
|
.name = "cx8800",
|
|
|
|
.id_table = cx8800_pci_tbl,
|
|
|
|
.probe = cx8800_initdev,
|
|
|
|
.remove = __devexit_p(cx8800_finidev),
|
2006-08-13 09:01:27 +08:00
|
|
|
#ifdef CONFIG_PM
|
2005-04-17 06:20:36 +08:00
|
|
|
.suspend = cx8800_suspend,
|
|
|
|
.resume = cx8800_resume,
|
2006-08-13 09:01:27 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int cx8800_init(void)
|
|
|
|
{
|
|
|
|
printk(KERN_INFO "cx2388x v4l2 driver version %d.%d.%d loaded\n",
|
|
|
|
(CX88_VERSION_CODE >> 16) & 0xff,
|
|
|
|
(CX88_VERSION_CODE >> 8) & 0xff,
|
|
|
|
CX88_VERSION_CODE & 0xff);
|
|
|
|
#ifdef SNAPSHOT
|
|
|
|
printk(KERN_INFO "cx2388x: snapshot date %04d-%02d-%02d\n",
|
|
|
|
SNAPSHOT/10000, (SNAPSHOT/100)%100, SNAPSHOT%100);
|
|
|
|
#endif
|
|
|
|
return pci_register_driver(&cx8800_pci_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cx8800_fini(void)
|
|
|
|
{
|
|
|
|
pci_unregister_driver(&cx8800_pci_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(cx8800_init);
|
|
|
|
module_exit(cx8800_fini);
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------- */
|
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* c-basic-offset: 8
|
|
|
|
* End:
|
2005-06-24 13:05:03 +08:00
|
|
|
* kate: eol "unix"; indent-width 3; remove-trailing-space on; replace-trailing-space-save on; tab-width 8; replace-tabs off; space-indent off; mixed-indent off
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|