staging: use videobuf2 framework for drivers/staging/dt3155v4l driver

This patch transforms drivers/staging/dt3155v4l driver to use videobuf2 framework.
Tested and works with "xawtv -f".

Either streaming API or read method should be selected during kernel configuration.

If both are selected into the driver (not possible without another patching),
either due to my misunderstanding or problems in xawtv (or both), I get kernel panic
after some start/stop of xawtv (not strictly reproducible).

Signed-off-by: Marin Mitov <mitov@issp.bas.bg>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
Marin Mitov 2011-05-28 21:45:27 +03:00 committed by Greg Kroah-Hartman
parent 7a1763320b
commit 8ded351ad1
3 changed files with 186 additions and 336 deletions

View File

@ -1,7 +1,7 @@
config VIDEO_DT3155 config VIDEO_DT3155
tristate "DT3155 frame grabber, Video4Linux interface" tristate "DT3155 frame grabber, Video4Linux interface"
depends on PCI && VIDEO_DEV && VIDEO_V4L2 depends on PCI && VIDEO_DEV && VIDEO_V4L2
select VIDEOBUF_DMA_CONTIG select VIDEOBUF2_DMA_CONTIG
default n default n
---help--- ---help---
Enables dt3155 device driver for the DataTranslation DT3155 frame grabber. Enables dt3155 device driver for the DataTranslation DT3155 frame grabber.
@ -18,3 +18,11 @@ config DT3155_CCIR
---help--- ---help---
Select it for CCIR/50Hz (European region), Select it for CCIR/50Hz (European region),
or leave it unselected for RS-170/60Hz (North America). or leave it unselected for RS-170/60Hz (North America).
config DT3155_STREAMING
bool "Selects streaming capture method"
depends on VIDEO_DT3155
default y
---help---
Select it if you want to use streaming of memory mapped buffers
or leave it unselected if you want to use read method (one copy more).

View File

@ -24,7 +24,7 @@
#include <linux/kthread.h> #include <linux/kthread.h>
#include <media/v4l2-dev.h> #include <media/v4l2-dev.h>
#include <media/v4l2-ioctl.h> #include <media/v4l2-ioctl.h>
#include <media/videobuf-dma-contig.h> #include <media/videobuf2-dma-contig.h>
#include "dt3155v4l.h" #include "dt3155v4l.h"
@ -38,6 +38,12 @@
#define DT3155_BUF_SIZE (768 * 576) #define DT3155_BUF_SIZE (768 * 576)
#ifdef CONFIG_DT3155_STREAMING
#define DT3155_CAPTURE_METHOD V4L2_CAP_STREAMING
#else
#define DT3155_CAPTURE_METHOD V4L2_CAP_READWRITE
#endif
/* global initializers (for all boards) */ /* global initializers (for all boards) */
#ifdef CONFIG_DT3155_CCIR #ifdef CONFIG_DT3155_CCIR
static const u8 csr2_init = VT_50HZ; static const u8 csr2_init = VT_50HZ;
@ -197,14 +203,14 @@ static int wait_i2c_reg(void __iomem *addr)
static int static int
dt3155_start_acq(struct dt3155_priv *pd) dt3155_start_acq(struct dt3155_priv *pd)
{ {
struct videobuf_buffer *vb = pd->curr_buf; struct vb2_buffer *vb = pd->curr_buf;
dma_addr_t dma_addr; dma_addr_t dma_addr;
dma_addr = videobuf_to_dma_contig(vb); dma_addr = vb2_dma_contig_plane_paddr(vb, 0);
iowrite32(dma_addr, pd->regs + EVEN_DMA_START); iowrite32(dma_addr, pd->regs + EVEN_DMA_START);
iowrite32(dma_addr + vb->width, pd->regs + ODD_DMA_START); iowrite32(dma_addr + img_width, pd->regs + ODD_DMA_START);
iowrite32(vb->width, pd->regs + EVEN_DMA_STRIDE); iowrite32(img_width, pd->regs + EVEN_DMA_STRIDE);
iowrite32(vb->width, pd->regs + ODD_DMA_STRIDE); iowrite32(img_width, pd->regs + ODD_DMA_STRIDE);
/* enable interrupts, clear all irq flags */ /* enable interrupts, clear all irq flags */
iowrite32(FLD_START_EN | FLD_END_ODD_EN | FLD_START | iowrite32(FLD_START_EN | FLD_END_ODD_EN | FLD_START |
FLD_END_EVEN | FLD_END_ODD, pd->regs + INT_CSR); FLD_END_EVEN | FLD_END_ODD, pd->regs + INT_CSR);
@ -221,95 +227,110 @@ dt3155_start_acq(struct dt3155_priv *pd)
return 0; /* success */ return 0; /* success */
} }
/*
* driver-specific callbacks (vb2_ops)
*/
static int static int
dt3155_stop_acq(struct dt3155_priv *pd) dt3155_queue_setup(struct vb2_queue *q, unsigned int *num_buffers,
unsigned int *num_planes, unsigned long sizes[],
void *alloc_ctxs[])
{ {
int tmp; struct dt3155_priv *pd = vb2_get_drv_priv(q);
void *ret;
/* stop the board */ if (*num_buffers == 0)
wait_i2c_reg(pd->regs); *num_buffers = 1;
write_i2c_reg(pd->regs, CSR2, pd->csr2); *num_planes = 1;
sizes[0] = img_width * img_height;
/* disable all irqs, clear all irq flags */ if (pd->q->alloc_ctx[0])
iowrite32(FLD_START | FLD_END_EVEN | FLD_END_ODD, pd->regs + INT_CSR); return 0;
write_i2c_reg(pd->regs, EVEN_CSR, CSR_ERROR | CSR_DONE); ret = vb2_dma_contig_init_ctx(&pd->pdev->dev);
write_i2c_reg(pd->regs, ODD_CSR, CSR_ERROR | CSR_DONE); if (IS_ERR(ret))
tmp = ioread32(pd->regs + CSR1) & (FLD_CRPT_EVEN | FLD_CRPT_ODD); return PTR_ERR(ret);
if (tmp) pd->q->alloc_ctx[0] = ret;
printk(KERN_ERR "dt3155: corrupted field %u\n", tmp);
iowrite32(FIFO_EN | SRST | FLD_CRPT_ODD | FLD_CRPT_EVEN |
FLD_DN_ODD | FLD_DN_EVEN | CAP_CONT_EVEN | CAP_CONT_ODD,
pd->regs + CSR1);
return 0; return 0;
} }
/* Locking: Caller holds q->vb_lock */ static void
static int dt3155_wait_prepare(struct vb2_queue *q)
dt3155_buf_setup(struct videobuf_queue *q, unsigned int *count,
unsigned int *size)
{ {
*size = img_width * img_height; struct dt3155_priv *pd = vb2_get_drv_priv(q);
mutex_unlock(pd->vdev->lock);
}
static void
dt3155_wait_finish(struct vb2_queue *q)
{
struct dt3155_priv *pd = vb2_get_drv_priv(q);
mutex_lock(pd->vdev->lock);
}
static int
dt3155_buf_prepare(struct vb2_buffer *vb)
{
vb2_set_plane_payload(vb, 0, img_width * img_height);
return 0; return 0;
} }
/* Locking: Caller holds q->vb_lock */
static int static int
dt3155_buf_prepare(struct videobuf_queue *q, struct videobuf_buffer *vb, dt3155_start_streaming(struct vb2_queue *q)
enum v4l2_field field)
{ {
int ret = 0; return 0;
}
vb->width = img_width;
vb->height = img_height; static int
vb->size = img_width * img_height; dt3155_stop_streaming(struct vb2_queue *q)
vb->field = field; {
if (vb->state == VIDEOBUF_NEEDS_INIT) struct dt3155_priv *pd = vb2_get_drv_priv(q);
ret = videobuf_iolock(q, vb, NULL); struct vb2_buffer *vb;
if (ret) {
vb->state = VIDEOBUF_ERROR; spin_lock_irq(&pd->lock);
printk(KERN_ERR "ERROR: videobuf_iolock() failed\n"); while (!list_empty(&pd->dmaq)) {
videobuf_dma_contig_free(q, vb); /* FIXME: needed? */ vb = list_first_entry(&pd->dmaq, typeof(*vb), done_entry);
} else list_del(&vb->done_entry);
vb->state = VIDEOBUF_PREPARED; vb2_buffer_done(vb, VB2_BUF_STATE_ERROR);
return ret; }
spin_unlock_irq(&pd->lock);
msleep(45); /* irq hendler will stop the hardware */
return 0;
} }
/* Locking: Caller holds q->vb_lock & q->irqlock */
static void static void
dt3155_buf_queue(struct videobuf_queue *q, struct videobuf_buffer *vb) dt3155_buf_queue(struct vb2_buffer *vb)
{ {
struct dt3155_priv *pd = q->priv_data; struct dt3155_priv *pd = vb2_get_drv_priv(vb->vb2_queue);
if (vb->state != VIDEOBUF_NEEDS_INIT) { /* pd->q->streaming = 1 when dt3155_buf_queue() is invoked */
vb->state = VIDEOBUF_QUEUED; spin_lock_irq(&pd->lock);
list_add_tail(&vb->queue, &pd->dmaq); if (pd->curr_buf)
wake_up_interruptible_sync(&pd->do_dma); list_add_tail(&vb->done_entry, &pd->dmaq);
} else else {
vb->state = VIDEOBUF_ERROR; pd->curr_buf = vb;
dt3155_start_acq(pd);
}
spin_unlock_irq(&pd->lock);
} }
/*
* end driver-specific callbacks
*/
/* Locking: Caller holds q->vb_lock */ const struct vb2_ops q_ops = {
static void .queue_setup = dt3155_queue_setup,
dt3155_buf_release(struct videobuf_queue *q, struct videobuf_buffer *vb) .wait_prepare = dt3155_wait_prepare,
{ .wait_finish = dt3155_wait_finish,
if (vb->state == VIDEOBUF_ACTIVE)
videobuf_waiton(q, vb, 0, 0); /* FIXME: cannot be interrupted */
videobuf_dma_contig_free(q, vb);
vb->state = VIDEOBUF_NEEDS_INIT;
}
static struct videobuf_queue_ops vbq_ops = {
.buf_setup = dt3155_buf_setup,
.buf_prepare = dt3155_buf_prepare, .buf_prepare = dt3155_buf_prepare,
.start_streaming = dt3155_start_streaming,
.stop_streaming = dt3155_stop_streaming,
.buf_queue = dt3155_buf_queue, .buf_queue = dt3155_buf_queue,
.buf_release = dt3155_buf_release,
}; };
static irqreturn_t static irqreturn_t
dt3155_irq_handler_even(int irq, void *dev_id) dt3155_irq_handler_even(int irq, void *dev_id)
{ {
struct dt3155_priv *ipd = dev_id; struct dt3155_priv *ipd = dev_id;
struct videobuf_buffer *ivb; struct vb2_buffer *ivb;
dma_addr_t dma_addr; dma_addr_t dma_addr;
u32 tmp; u32 tmp;
@ -341,33 +362,22 @@ dt3155_irq_handler_even(int irq, void *dev_id)
} }
spin_lock(&ipd->lock); spin_lock(&ipd->lock);
if (ipd->curr_buf && ipd->curr_buf->state == VIDEOBUF_ACTIVE) { if (ipd->curr_buf) {
if (waitqueue_active(&ipd->curr_buf->done)) { do_gettimeofday(&ipd->curr_buf->v4l2_buf.timestamp);
do_gettimeofday(&ipd->curr_buf->ts); ipd->curr_buf->v4l2_buf.sequence = (ipd->field_count) >> 1;
ipd->curr_buf->field_count = ipd->field_count; vb2_buffer_done(ipd->curr_buf, VB2_BUF_STATE_DONE);
ipd->curr_buf->state = VIDEOBUF_DONE;
wake_up(&ipd->curr_buf->done);
} else {
ivb = ipd->curr_buf;
goto load_dma;
} }
} else
if (!ipd->q->streaming || list_empty(&ipd->dmaq))
goto stop_dma; goto stop_dma;
if (list_empty(&ipd->dmaq)) ivb = list_first_entry(&ipd->dmaq, typeof(*ivb), done_entry);
goto stop_dma; list_del(&ivb->done_entry);
ivb = list_first_entry(&ipd->dmaq, typeof(*ivb), queue);
list_del(&ivb->queue);
if (ivb->state == VIDEOBUF_QUEUED) {
ivb->state = VIDEOBUF_ACTIVE;
ipd->curr_buf = ivb; ipd->curr_buf = ivb;
} else dma_addr = vb2_dma_contig_plane_paddr(ivb, 0);
goto stop_dma;
load_dma:
dma_addr = videobuf_to_dma_contig(ivb);
iowrite32(dma_addr, ipd->regs + EVEN_DMA_START); iowrite32(dma_addr, ipd->regs + EVEN_DMA_START);
iowrite32(dma_addr + ivb->width, ipd->regs + ODD_DMA_START); iowrite32(dma_addr + img_width, ipd->regs + ODD_DMA_START);
iowrite32(ivb->width, ipd->regs + EVEN_DMA_STRIDE); iowrite32(img_width, ipd->regs + EVEN_DMA_STRIDE);
iowrite32(ivb->width, ipd->regs + ODD_DMA_STRIDE); iowrite32(img_width, ipd->regs + ODD_DMA_STRIDE);
mmiowb(); mmiowb();
/* enable interrupts, clear all irq flags */ /* enable interrupts, clear all irq flags */
iowrite32(FLD_START_EN | FLD_END_ODD_EN | FLD_START | iowrite32(FLD_START_EN | FLD_END_ODD_EN | FLD_START |
@ -379,68 +389,40 @@ stop_dma:
ipd->curr_buf = NULL; ipd->curr_buf = NULL;
/* stop the board */ /* stop the board */
write_i2c_reg_nowait(ipd->regs, CSR2, ipd->csr2); write_i2c_reg_nowait(ipd->regs, CSR2, ipd->csr2);
iowrite32(FIFO_EN | SRST | FLD_CRPT_ODD | FLD_CRPT_EVEN |
FLD_DN_ODD | FLD_DN_EVEN, ipd->regs + CSR1);
/* disable interrupts, clear all irq flags */ /* disable interrupts, clear all irq flags */
iowrite32(FLD_START | FLD_END_EVEN | FLD_END_ODD, ipd->regs + INT_CSR); iowrite32(FLD_START | FLD_END_EVEN | FLD_END_ODD, ipd->regs + INT_CSR);
spin_unlock(&ipd->lock); spin_unlock(&ipd->lock);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static int
dt3155_threadfn(void *arg)
{
struct dt3155_priv *pd = arg;
struct videobuf_buffer *vb;
unsigned long flags;
while (1) {
wait_event_interruptible(pd->do_dma,
kthread_should_stop() || !list_empty(&pd->dmaq));
if (kthread_should_stop())
break;
spin_lock_irqsave(&pd->lock, flags);
if (pd->curr_buf) /* dma is active */
goto done;
if (list_empty(&pd->dmaq)) /* no empty biffers */
goto done;
vb = list_first_entry(&pd->dmaq, typeof(*vb), queue);
list_del(&vb->queue);
if (vb->state == VIDEOBUF_QUEUED) {
vb->state = VIDEOBUF_ACTIVE;
pd->curr_buf = vb;
spin_unlock_irqrestore(&pd->lock, flags);
/* start dma */
dt3155_start_acq(pd);
continue;
} else
printk(KERN_DEBUG "%s(): This is a BUG\n", __func__);
done:
spin_unlock_irqrestore(&pd->lock, flags);
}
return 0;
}
static int static int
dt3155_open(struct file *filp) dt3155_open(struct file *filp)
{ {
int ret = 0; int ret = 0;
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
printk(KERN_INFO "dt3155: open(): minor: %i\n", pd->vdev->minor); printk(KERN_INFO "dt3155: open(): minor: %i, users: %i\n",
pd->vdev->minor, pd->users);
if (mutex_lock_interruptible(&pd->mux) == -EINTR)
return -ERESTARTSYS;
if (!pd->users) { if (!pd->users) {
pd->vidq = kzalloc(sizeof(*pd->vidq), GFP_KERNEL); pd->q = kzalloc(sizeof(*pd->q), GFP_KERNEL);
if (!pd->vidq) { if (!pd->q) {
printk(KERN_ERR "dt3155: error: alloc queue\n"); printk(KERN_ERR "dt3155: error: alloc queue\n");
ret = -ENOMEM; ret = -ENOMEM;
goto err_alloc_queue; goto err_alloc_queue;
} }
videobuf_queue_dma_contig_init(pd->vidq, &vbq_ops, pd->q->type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
&pd->pdev->dev, &pd->lock, pd->q->io_modes = VB2_READ | VB2_MMAP;
V4L2_BUF_TYPE_VIDEO_CAPTURE, V4L2_FIELD_NONE, pd->q->ops = &q_ops;
sizeof(struct videobuf_buffer), pd, NULL); pd->q->mem_ops = &vb2_dma_contig_memops;
pd->q->drv_priv = pd;
pd->curr_buf = NULL;
pd->field_count = 0;
vb2_queue_init(pd->q); /* cannot fail */
INIT_LIST_HEAD(&pd->dmaq);
spin_lock_init(&pd->lock);
/* disable all irqs, clear all irq flags */ /* disable all irqs, clear all irq flags */
iowrite32(FLD_START | FLD_END_EVEN | FLD_END_ODD, iowrite32(FLD_START | FLD_END_EVEN | FLD_END_ODD,
pd->regs + INT_CSR); pd->regs + INT_CSR);
@ -451,26 +433,13 @@ dt3155_open(struct file *filp)
printk(KERN_ERR "dt3155: error: request_irq\n"); printk(KERN_ERR "dt3155: error: request_irq\n");
goto err_request_irq; goto err_request_irq;
} }
pd->curr_buf = NULL;
pd->thread = kthread_run(dt3155_threadfn, pd,
"dt3155_thread_%i", pd->vdev->minor);
if (IS_ERR(pd->thread)) {
printk(KERN_ERR "dt3155: kthread_run() failed\n");
ret = PTR_ERR(pd->thread);
goto err_thread;
}
pd->field_count = 0;
} }
pd->users++; pd->users++;
goto done; return 0; /* success */
err_thread:
free_irq(pd->pdev->irq, pd);
err_request_irq: err_request_irq:
kfree(pd->vidq); kfree(pd->q);
pd->vidq = NULL; pd->q = NULL;
err_alloc_queue: err_alloc_queue:
done:
mutex_unlock(&pd->mux);
return ret; return ret;
} }
@ -478,61 +447,29 @@ static int
dt3155_release(struct file *filp) dt3155_release(struct file *filp)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_buffer *tmp;
unsigned long flags;
int ret = 0;
printk(KERN_INFO "dt3155: release(): minor: %i\n", pd->vdev->minor); printk(KERN_INFO "dt3155: release(): minor: %i, users: %i\n",
pd->vdev->minor, pd->users - 1);
if (mutex_lock_interruptible(&pd->mux) == -EINTR)
return -ERESTARTSYS;
pd->users--; pd->users--;
BUG_ON(pd->users < 0); BUG_ON(pd->users < 0);
if (pd->acq_fp == filp) {
spin_lock_irqsave(&pd->lock, flags);
INIT_LIST_HEAD(&pd->dmaq); /* queue is emptied */
tmp = pd->curr_buf;
spin_unlock_irqrestore(&pd->lock, flags);
if (tmp)
videobuf_waiton(pd->vidq, tmp, 0, 1); /* block, interruptible */
dt3155_stop_acq(pd);
videobuf_stop(pd->vidq);
pd->acq_fp = NULL;
pd->streaming = 0;
}
if (!pd->users) { if (!pd->users) {
kthread_stop(pd->thread); vb2_queue_release(pd->q);
free_irq(pd->pdev->irq, pd); free_irq(pd->pdev->irq, pd);
kfree(pd->vidq); if (pd->q->alloc_ctx[0])
pd->vidq = NULL; vb2_dma_contig_cleanup_ctx(pd->q->alloc_ctx[0]);
kfree(pd->q);
pd->q = NULL;
} }
mutex_unlock(&pd->mux); return 0;
return ret;
} }
static ssize_t static ssize_t
dt3155_read(struct file *filp, char __user *user, size_t size, loff_t *loff) dt3155_read(struct file *filp, char __user *user, size_t size, loff_t *loff)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
int ret;
if (mutex_lock_interruptible(&pd->mux) == -EINTR) return vb2_read(pd->q, user, size, loff, filp->f_flags & O_NONBLOCK);
return -ERESTARTSYS;
if (!pd->acq_fp) {
pd->acq_fp = filp;
pd->streaming = 0;
} else if (pd->acq_fp != filp) {
ret = -EBUSY;
goto done;
} else if (pd->streaming == 1) {
ret = -EINVAL;
goto done;
}
ret = videobuf_read_stream(pd->vidq, user, size, loff, 0,
filp->f_flags & O_NONBLOCK);
done:
mutex_unlock(&pd->mux);
return ret;
} }
static unsigned int static unsigned int
@ -540,7 +477,7 @@ dt3155_poll(struct file *filp, struct poll_table_struct *polltbl)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
return videobuf_poll_stream(filp, pd->vidq, polltbl); return vb2_poll(pd->q, filp, polltbl);
} }
static int static int
@ -548,7 +485,7 @@ dt3155_mmap(struct file *filp, struct vm_area_struct *vma)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
return videobuf_mmap_mapper(pd->vidq, vma); return vb2_mmap(pd->q, vma);
} }
static const struct v4l2_file_operations dt3155_fops = { static const struct v4l2_file_operations dt3155_fops = {
@ -565,46 +502,16 @@ static int
dt3155_ioc_streamon(struct file *filp, void *p, enum v4l2_buf_type type) dt3155_ioc_streamon(struct file *filp, void *p, enum v4l2_buf_type type)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
int ret = -ERESTARTSYS;
if (mutex_lock_interruptible(&pd->mux) == -EINTR) return vb2_streamon(pd->q, type);
return ret;
if (!pd->acq_fp) {
ret = videobuf_streamon(pd->vidq);
if (ret)
goto unlock;
pd->acq_fp = filp;
pd->streaming = 1;
wake_up_interruptible_sync(&pd->do_dma);
} else if (pd->acq_fp == filp) {
pd->streaming = 1;
ret = videobuf_streamon(pd->vidq);
if (!ret)
wake_up_interruptible_sync(&pd->do_dma);
} else
ret = -EBUSY;
unlock:
mutex_unlock(&pd->mux);
return ret;
} }
static int static int
dt3155_ioc_streamoff(struct file *filp, void *p, enum v4l2_buf_type type) dt3155_ioc_streamoff(struct file *filp, void *p, enum v4l2_buf_type type)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_buffer *tmp;
unsigned long flags;
int ret;
ret = videobuf_streamoff(pd->vidq); return vb2_streamoff(pd->q, type);
if (ret)
return ret;
spin_lock_irqsave(&pd->lock, flags);
tmp = pd->curr_buf;
spin_unlock_irqrestore(&pd->lock, flags);
if (tmp)
videobuf_waiton(pd->vidq, tmp, 0, 1); /* block, interruptible */
return ret;
} }
static int static int
@ -618,8 +525,7 @@ dt3155_ioc_querycap(struct file *filp, void *p, struct v4l2_capability *cap)
cap->version = cap->version =
KERNEL_VERSION(DT3155_VER_MAJ, DT3155_VER_MIN, DT3155_VER_EXT); KERNEL_VERSION(DT3155_VER_MAJ, DT3155_VER_MIN, DT3155_VER_EXT);
cap->capabilities = V4L2_CAP_VIDEO_CAPTURE | cap->capabilities = V4L2_CAP_VIDEO_CAPTURE |
V4L2_CAP_STREAMING | DT3155_CAPTURE_METHOD;
V4L2_CAP_READWRITE;
return 0; return 0;
} }
@ -667,93 +573,39 @@ dt3155_ioc_try_fmt_vid_cap(struct file *filp, void *p, struct v4l2_format *f)
static int static int
dt3155_ioc_s_fmt_vid_cap(struct file *filp, void *p, struct v4l2_format *f) dt3155_ioc_s_fmt_vid_cap(struct file *filp, void *p, struct v4l2_format *f)
{ {
struct dt3155_priv *pd = video_drvdata(filp); return dt3155_ioc_g_fmt_vid_cap(filp, p, f);
int ret = -ERESTARTSYS;
if (mutex_lock_interruptible(&pd->mux) == -EINTR)
return ret;
if (!pd->acq_fp) {
pd->acq_fp = filp;
pd->streaming = 0;
} else if (pd->acq_fp != filp) {
ret = -EBUSY;
goto done;
}
/* FIXME: we don't change the format for now
if (pd->vidq->streaming || pd->vidq->reading || pd->curr_buff) {
ret = -EBUSY;
goto done;
}
*/
ret = dt3155_ioc_g_fmt_vid_cap(filp, p, f);
done:
mutex_unlock(&pd->mux);
return ret;
} }
static int static int
dt3155_ioc_reqbufs(struct file *filp, void *p, struct v4l2_requestbuffers *b) dt3155_ioc_reqbufs(struct file *filp, void *p, struct v4l2_requestbuffers *b)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_queue *q = pd->vidq;
int ret = -ERESTARTSYS;
if (b->memory != V4L2_MEMORY_MMAP) return vb2_reqbufs(pd->q, b);
return -EINVAL;
if (mutex_lock_interruptible(&pd->mux) == -EINTR)
return ret;
if (!pd->acq_fp)
pd->acq_fp = filp;
else if (pd->acq_fp != filp) {
ret = -EBUSY;
goto done;
}
pd->streaming = 1;
ret = 0;
done:
mutex_unlock(&pd->mux);
if (ret)
return ret;
if (b->count)
ret = videobuf_reqbufs(q, b);
else { /* FIXME: is it necessary? */
printk(KERN_DEBUG "dt3155: request to free buffers\n");
/* ret = videobuf_mmap_free(q); */
ret = dt3155_ioc_streamoff(filp, p,
V4L2_BUF_TYPE_VIDEO_CAPTURE);
}
return ret;
} }
static int static int
dt3155_ioc_querybuf(struct file *filp, void *p, struct v4l2_buffer *b) dt3155_ioc_querybuf(struct file *filp, void *p, struct v4l2_buffer *b)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_queue *q = pd->vidq;
return videobuf_querybuf(q, b); return vb2_querybuf(pd->q, b);
} }
static int static int
dt3155_ioc_qbuf(struct file *filp, void *p, struct v4l2_buffer *b) dt3155_ioc_qbuf(struct file *filp, void *p, struct v4l2_buffer *b)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_queue *q = pd->vidq;
int ret;
ret = videobuf_qbuf(q, b); return vb2_qbuf(pd->q, b);
if (ret)
return ret;
return videobuf_querybuf(q, b);
} }
static int static int
dt3155_ioc_dqbuf(struct file *filp, void *p, struct v4l2_buffer *b) dt3155_ioc_dqbuf(struct file *filp, void *p, struct v4l2_buffer *b)
{ {
struct dt3155_priv *pd = video_drvdata(filp); struct dt3155_priv *pd = video_drvdata(filp);
struct videobuf_queue *q = pd->vidq;
return videobuf_dqbuf(q, b, filp->f_flags & O_NONBLOCK); return vb2_dqbuf(pd->q, b, filp->f_flags & O_NONBLOCK);
} }
static int static int
@ -880,21 +732,21 @@ static const struct v4l2_ioctl_ops dt3155_ioctl_ops = {
}; };
static int __devinit static int __devinit
dt3155_init_board(struct pci_dev *dev) dt3155_init_board(struct pci_dev *pdev)
{ {
struct dt3155_priv *pd = pci_get_drvdata(dev); struct dt3155_priv *pd = pci_get_drvdata(pdev);
void *buf_cpu; void *buf_cpu;
dma_addr_t buf_dma; dma_addr_t buf_dma;
int i; int i;
u8 tmp; u8 tmp;
pci_set_master(dev); /* dt3155 needs it */ pci_set_master(pdev); /* dt3155 needs it */
/* resetting the adapter */ /* resetting the adapter */
iowrite32(FLD_CRPT_ODD | FLD_CRPT_EVEN | FLD_DN_ODD | FLD_DN_EVEN, iowrite32(FLD_CRPT_ODD | FLD_CRPT_EVEN | FLD_DN_ODD | FLD_DN_EVEN,
pd->regs + CSR1); pd->regs + CSR1);
mmiowb(); mmiowb();
msleep(10); msleep(20);
/* initializing adaper registers */ /* initializing adaper registers */
iowrite32(FIFO_EN | SRST, pd->regs + CSR1); iowrite32(FIFO_EN | SRST, pd->regs + CSR1);
@ -949,7 +801,7 @@ dt3155_init_board(struct pci_dev *dev)
write_i2c_reg(pd->regs, AD_CMD, VIDEO_CNL_1 | SYNC_CNL_1 | SYNC_LVL_3); write_i2c_reg(pd->regs, AD_CMD, VIDEO_CNL_1 | SYNC_CNL_1 | SYNC_LVL_3);
/* allocate memory, and initialize the DMA machine */ /* allocate memory, and initialize the DMA machine */
buf_cpu = dma_alloc_coherent(&dev->dev, DT3155_BUF_SIZE, &buf_dma, buf_cpu = dma_alloc_coherent(&pdev->dev, DT3155_BUF_SIZE, &buf_dma,
GFP_KERNEL); GFP_KERNEL);
if (!buf_cpu) { if (!buf_cpu) {
printk(KERN_ERR "dt3155: dma_alloc_coherent " printk(KERN_ERR "dt3155: dma_alloc_coherent "
@ -975,7 +827,7 @@ dt3155_init_board(struct pci_dev *dev)
iowrite32(FIFO_EN | SRST | FLD_DN_EVEN | FLD_DN_ODD, pd->regs + CSR1); iowrite32(FIFO_EN | SRST | FLD_DN_EVEN | FLD_DN_ODD, pd->regs + CSR1);
/* deallocate memory */ /* deallocate memory */
dma_free_coherent(&dev->dev, DT3155_BUF_SIZE, buf_cpu, buf_dma); dma_free_coherent(&pdev->dev, DT3155_BUF_SIZE, buf_cpu, buf_dma);
if (tmp & BUSY_EVEN) { if (tmp & BUSY_EVEN) {
printk(KERN_ERR "dt3155: BUSY_EVEN not cleared\n"); printk(KERN_ERR "dt3155: BUSY_EVEN not cleared\n");
return -EIO; return -EIO;
@ -996,7 +848,7 @@ static struct video_device dt3155_vdev = {
/* same as in drivers/base/dma-coherent.c */ /* same as in drivers/base/dma-coherent.c */
struct dma_coherent_mem { struct dma_coherent_mem {
void *virt_base; void *virt_base;
u32 device_base; dma_addr_t device_base;
int size; int size;
int flags; int flags;
unsigned long *bitmap; unsigned long *bitmap;
@ -1058,18 +910,18 @@ dt3155_free_coherent(struct device *dev)
} }
static int __devinit static int __devinit
dt3155_probe(struct pci_dev *dev, const struct pci_device_id *id) dt3155_probe(struct pci_dev *pdev, const struct pci_device_id *id)
{ {
int err; int err;
struct dt3155_priv *pd; struct dt3155_priv *pd;
printk(KERN_INFO "dt3155: probe()\n"); printk(KERN_INFO "dt3155: probe()\n");
err = dma_set_mask(&dev->dev, DMA_BIT_MASK(32)); err = dma_set_mask(&pdev->dev, DMA_BIT_MASK(32));
if (err) { if (err) {
printk(KERN_ERR "dt3155: cannot set dma_mask\n"); printk(KERN_ERR "dt3155: cannot set dma_mask\n");
return -ENODEV; return -ENODEV;
} }
err = dma_set_coherent_mask(&dev->dev, DMA_BIT_MASK(32)); err = dma_set_coherent_mask(&pdev->dev, DMA_BIT_MASK(32));
if (err) { if (err) {
printk(KERN_ERR "dt3155: cannot set dma_coherent_mask\n"); printk(KERN_ERR "dt3155: cannot set dma_coherent_mask\n");
return -ENODEV; return -ENODEV;
@ -1085,31 +937,31 @@ dt3155_probe(struct pci_dev *dev, const struct pci_device_id *id)
goto err_video_device_alloc; goto err_video_device_alloc;
} }
*pd->vdev = dt3155_vdev; *pd->vdev = dt3155_vdev;
pci_set_drvdata(dev, pd); /* for use in dt3155_remove() */ pci_set_drvdata(pdev, pd); /* for use in dt3155_remove() */
video_set_drvdata(pd->vdev, pd); /* for use in video_fops */ video_set_drvdata(pd->vdev, pd); /* for use in video_fops */
pd->users = 0; pd->users = 0;
pd->acq_fp = NULL; pd->pdev = pdev;
pd->pdev = dev;
INIT_LIST_HEAD(&pd->dmaq); INIT_LIST_HEAD(&pd->dmaq);
init_waitqueue_head(&pd->do_dma);
mutex_init(&pd->mux); mutex_init(&pd->mux);
pd->vdev->lock = &pd->mux; /* for locking v4l2_file_operations */
spin_lock_init(&pd->lock);
pd->csr2 = csr2_init; pd->csr2 = csr2_init;
pd->config = config_init; pd->config = config_init;
err = pci_enable_device(pd->pdev); err = pci_enable_device(pdev);
if (err) { if (err) {
printk(KERN_ERR "dt3155: pci_dev not enabled\n"); printk(KERN_ERR "dt3155: pci_dev not enabled\n");
goto err_enable_dev; goto err_enable_dev;
} }
err = pci_request_region(pd->pdev, 0, pci_name(pd->pdev)); err = pci_request_region(pdev, 0, pci_name(pdev));
if (err) if (err)
goto err_req_region; goto err_req_region;
pd->regs = pci_iomap(pd->pdev, 0, pci_resource_len(pd->pdev, 0)); pd->regs = pci_iomap(pdev, 0, pci_resource_len(pd->pdev, 0));
if (!pd->regs) { if (!pd->regs) {
err = -ENOMEM; err = -ENOMEM;
printk(KERN_ERR "dt3155: pci_iomap failed\n"); printk(KERN_ERR "dt3155: pci_iomap failed\n");
goto err_pci_iomap; goto err_pci_iomap;
} }
err = dt3155_init_board(pd->pdev); err = dt3155_init_board(pdev);
if (err) { if (err) {
printk(KERN_ERR "dt3155: dt3155_init_board failed\n"); printk(KERN_ERR "dt3155: dt3155_init_board failed\n");
goto err_init_board; goto err_init_board;
@ -1119,7 +971,7 @@ dt3155_probe(struct pci_dev *dev, const struct pci_device_id *id)
printk(KERN_ERR "dt3155: Cannot register video device\n"); printk(KERN_ERR "dt3155: Cannot register video device\n");
goto err_init_board; goto err_init_board;
} }
err = dt3155_alloc_coherent(&dev->dev, DT3155_CHUNK_SIZE, err = dt3155_alloc_coherent(&pdev->dev, DT3155_CHUNK_SIZE,
DMA_MEMORY_MAP); DMA_MEMORY_MAP);
if (err) if (err)
printk(KERN_INFO "dt3155: preallocated 8 buffers\n"); printk(KERN_INFO "dt3155: preallocated 8 buffers\n");
@ -1127,11 +979,11 @@ dt3155_probe(struct pci_dev *dev, const struct pci_device_id *id)
return 0; /* success */ return 0; /* success */
err_init_board: err_init_board:
pci_iounmap(pd->pdev, pd->regs); pci_iounmap(pdev, pd->regs);
err_pci_iomap: err_pci_iomap:
pci_release_region(pd->pdev, 0); pci_release_region(pdev, 0);
err_req_region: err_req_region:
pci_disable_device(pd->pdev); pci_disable_device(pdev);
err_enable_dev: err_enable_dev:
video_device_release(pd->vdev); video_device_release(pd->vdev);
err_video_device_alloc: err_video_device_alloc:
@ -1140,16 +992,16 @@ err_video_device_alloc:
} }
static void __devexit static void __devexit
dt3155_remove(struct pci_dev *dev) dt3155_remove(struct pci_dev *pdev)
{ {
struct dt3155_priv *pd = pci_get_drvdata(dev); struct dt3155_priv *pd = pci_get_drvdata(pdev);
printk(KERN_INFO "dt3155: remove()\n"); printk(KERN_INFO "dt3155: remove()\n");
dt3155_free_coherent(&dev->dev); dt3155_free_coherent(&pdev->dev);
video_unregister_device(pd->vdev); video_unregister_device(pd->vdev);
pci_iounmap(dev, pd->regs); pci_iounmap(pdev, pd->regs);
pci_release_region(pd->pdev, 0); pci_release_region(pdev, 0);
pci_disable_device(pd->pdev); pci_disable_device(pdev);
/* /*
* video_device_release() is invoked automatically * video_device_release() is invoked automatically
* see: struct video_device dt3155_vdev * see: struct video_device dt3155_vdev

View File

@ -179,18 +179,13 @@ struct dt3155_stats {
* struct dt3155_priv - private data structure * struct dt3155_priv - private data structure
* *
* @vdev: pointer to video_device structure * @vdev: pointer to video_device structure
* @acq_fp pointer to filp that starts acquisition
* @streaming streaming is negotiated
* @pdev: pointer to pci_dev structure * @pdev: pointer to pci_dev structure
* @vidq pointer to videobuf_queue structure * @q pointer to vb2_queue structure
* @curr_buf: pointer to curren buffer * @curr_buf: pointer to curren buffer
* @thread pointer to worker thraed
* @irq_handler: irq handler for the driver
* @qt_ops local copy of dma-contig qtype_ops
* @dmaq queue for dma buffers
* @do_dma wait queue of the kernel thread
* @mux: mutex to protect the instance * @mux: mutex to protect the instance
* @lock spinlock for videobuf queues * @irq_handler: irq handler for the driver
* @dmaq queue for dma buffers
* @lock spinlock for dma queue
* @field_count fields counter * @field_count fields counter
* @stats: statistics structure * @stats: statistics structure
* @users open count * @users open count
@ -200,17 +195,12 @@ struct dt3155_stats {
*/ */
struct dt3155_priv { struct dt3155_priv {
struct video_device *vdev; struct video_device *vdev;
struct file *acq_fp;
int streaming;
struct pci_dev *pdev; struct pci_dev *pdev;
struct videobuf_queue *vidq; struct vb2_queue *q;
struct videobuf_buffer *curr_buf; struct vb2_buffer *curr_buf;
struct task_struct *thread;
irq_handler_t irq_handler;
struct videobuf_qtype_ops qt_ops;
struct list_head dmaq;
wait_queue_head_t do_dma;
struct mutex mux; struct mutex mux;
irq_handler_t irq_handler;
struct list_head dmaq;
spinlock_t lock; spinlock_t lock;
unsigned int field_count; unsigned int field_count;
struct dt3155_stats stats; struct dt3155_stats stats;