[media] add blackfin capture bridge driver
This is a v4l2 bridge driver for Blackfin video capture device, support ppi and eppi interface. Signed-off-by: Scott Jiang <scott.jiang.linux@gmail.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
This commit is contained in:
parent
f877ed9780
commit
63b1a90da9
8 changed files with 1457 additions and 0 deletions
|
@ -871,6 +871,8 @@ source "drivers/media/video/davinci/Kconfig"
|
|||
|
||||
source "drivers/media/video/omap/Kconfig"
|
||||
|
||||
source "drivers/media/video/blackfin/Kconfig"
|
||||
|
||||
config VIDEO_SH_VOU
|
||||
tristate "SuperH VOU video output driver"
|
||||
depends on VIDEO_DEV && ARCH_SHMOBILE
|
||||
|
|
|
@ -187,6 +187,8 @@ obj-$(CONFIG_VIDEO_SAMSUNG_S5P_TV) += s5p-tv/
|
|||
|
||||
obj-$(CONFIG_VIDEO_SAMSUNG_S5P_G2D) += s5p-g2d/
|
||||
|
||||
obj-$(CONFIG_BLACKFIN) += blackfin/
|
||||
|
||||
obj-$(CONFIG_ARCH_DAVINCI) += davinci/
|
||||
|
||||
obj-$(CONFIG_VIDEO_SH_VOU) += sh_vou.o
|
||||
|
|
10
drivers/media/video/blackfin/Kconfig
Normal file
10
drivers/media/video/blackfin/Kconfig
Normal file
|
@ -0,0 +1,10 @@
|
|||
config VIDEO_BLACKFIN_CAPTURE
|
||||
tristate "Blackfin Video Capture Driver"
|
||||
depends on VIDEO_V4L2 && BLACKFIN && I2C
|
||||
select VIDEOBUF2_DMA_CONTIG
|
||||
help
|
||||
V4L2 bridge driver for Blackfin video capture device.
|
||||
Choose PPI or EPPI as its interface.
|
||||
|
||||
To compile this driver as a module, choose M here: the
|
||||
module will be called bfin_video_capture.
|
2
drivers/media/video/blackfin/Makefile
Normal file
2
drivers/media/video/blackfin/Makefile
Normal file
|
@ -0,0 +1,2 @@
|
|||
bfin_video_capture-objs := bfin_capture.o ppi.o
|
||||
obj-$(CONFIG_VIDEO_BLACKFIN_CAPTURE) += bfin_video_capture.o
|
1059
drivers/media/video/blackfin/bfin_capture.c
Normal file
1059
drivers/media/video/blackfin/bfin_capture.c
Normal file
File diff suppressed because it is too large
Load diff
271
drivers/media/video/blackfin/ppi.c
Normal file
271
drivers/media/video/blackfin/ppi.c
Normal file
|
@ -0,0 +1,271 @@
|
|||
/*
|
||||
* ppi.c Analog Devices Parallel Peripheral Interface driver
|
||||
*
|
||||
* Copyright (c) 2011 Analog Devices Inc.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* 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/slab.h>
|
||||
|
||||
#include <asm/bfin_ppi.h>
|
||||
#include <asm/blackfin.h>
|
||||
#include <asm/cacheflush.h>
|
||||
#include <asm/dma.h>
|
||||
#include <asm/portmux.h>
|
||||
|
||||
#include <media/blackfin/ppi.h>
|
||||
|
||||
static int ppi_attach_irq(struct ppi_if *ppi, irq_handler_t handler);
|
||||
static void ppi_detach_irq(struct ppi_if *ppi);
|
||||
static int ppi_start(struct ppi_if *ppi);
|
||||
static int ppi_stop(struct ppi_if *ppi);
|
||||
static int ppi_set_params(struct ppi_if *ppi, struct ppi_params *params);
|
||||
static void ppi_update_addr(struct ppi_if *ppi, unsigned long addr);
|
||||
|
||||
static const struct ppi_ops ppi_ops = {
|
||||
.attach_irq = ppi_attach_irq,
|
||||
.detach_irq = ppi_detach_irq,
|
||||
.start = ppi_start,
|
||||
.stop = ppi_stop,
|
||||
.set_params = ppi_set_params,
|
||||
.update_addr = ppi_update_addr,
|
||||
};
|
||||
|
||||
static irqreturn_t ppi_irq_err(int irq, void *dev_id)
|
||||
{
|
||||
struct ppi_if *ppi = dev_id;
|
||||
const struct ppi_info *info = ppi->info;
|
||||
|
||||
switch (info->type) {
|
||||
case PPI_TYPE_PPI:
|
||||
{
|
||||
struct bfin_ppi_regs *reg = info->base;
|
||||
unsigned short status;
|
||||
|
||||
/* register on bf561 is cleared when read
|
||||
* others are W1C
|
||||
*/
|
||||
status = bfin_read16(®->status);
|
||||
bfin_write16(®->status, 0xff00);
|
||||
break;
|
||||
}
|
||||
case PPI_TYPE_EPPI:
|
||||
{
|
||||
struct bfin_eppi_regs *reg = info->base;
|
||||
bfin_write16(®->status, 0xffff);
|
||||
break;
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int ppi_attach_irq(struct ppi_if *ppi, irq_handler_t handler)
|
||||
{
|
||||
const struct ppi_info *info = ppi->info;
|
||||
int ret;
|
||||
|
||||
ret = request_dma(info->dma_ch, "PPI_DMA");
|
||||
|
||||
if (ret) {
|
||||
pr_err("Unable to allocate DMA channel for PPI\n");
|
||||
return ret;
|
||||
}
|
||||
set_dma_callback(info->dma_ch, handler, ppi);
|
||||
|
||||
if (ppi->err_int) {
|
||||
ret = request_irq(info->irq_err, ppi_irq_err, 0, "PPI ERROR", ppi);
|
||||
if (ret) {
|
||||
pr_err("Unable to allocate IRQ for PPI\n");
|
||||
free_dma(info->dma_ch);
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void ppi_detach_irq(struct ppi_if *ppi)
|
||||
{
|
||||
const struct ppi_info *info = ppi->info;
|
||||
|
||||
if (ppi->err_int)
|
||||
free_irq(info->irq_err, ppi);
|
||||
free_dma(info->dma_ch);
|
||||
}
|
||||
|
||||
static int ppi_start(struct ppi_if *ppi)
|
||||
{
|
||||
const struct ppi_info *info = ppi->info;
|
||||
|
||||
/* enable DMA */
|
||||
enable_dma(info->dma_ch);
|
||||
|
||||
/* enable PPI */
|
||||
ppi->ppi_control |= PORT_EN;
|
||||
switch (info->type) {
|
||||
case PPI_TYPE_PPI:
|
||||
{
|
||||
struct bfin_ppi_regs *reg = info->base;
|
||||
bfin_write16(®->control, ppi->ppi_control);
|
||||
break;
|
||||
}
|
||||
case PPI_TYPE_EPPI:
|
||||
{
|
||||
struct bfin_eppi_regs *reg = info->base;
|
||||
bfin_write32(®->control, ppi->ppi_control);
|
||||
break;
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
SSYNC();
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ppi_stop(struct ppi_if *ppi)
|
||||
{
|
||||
const struct ppi_info *info = ppi->info;
|
||||
|
||||
/* disable PPI */
|
||||
ppi->ppi_control &= ~PORT_EN;
|
||||
switch (info->type) {
|
||||
case PPI_TYPE_PPI:
|
||||
{
|
||||
struct bfin_ppi_regs *reg = info->base;
|
||||
bfin_write16(®->control, ppi->ppi_control);
|
||||
break;
|
||||
}
|
||||
case PPI_TYPE_EPPI:
|
||||
{
|
||||
struct bfin_eppi_regs *reg = info->base;
|
||||
bfin_write32(®->control, ppi->ppi_control);
|
||||
break;
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* disable DMA */
|
||||
clear_dma_irqstat(info->dma_ch);
|
||||
disable_dma(info->dma_ch);
|
||||
|
||||
SSYNC();
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ppi_set_params(struct ppi_if *ppi, struct ppi_params *params)
|
||||
{
|
||||
const struct ppi_info *info = ppi->info;
|
||||
int dma32 = 0;
|
||||
int dma_config, bytes_per_line, lines_per_frame;
|
||||
|
||||
bytes_per_line = params->width * params->bpp / 8;
|
||||
lines_per_frame = params->height;
|
||||
if (params->int_mask == 0xFFFFFFFF)
|
||||
ppi->err_int = false;
|
||||
else
|
||||
ppi->err_int = true;
|
||||
|
||||
dma_config = (DMA_FLOW_STOP | WNR | RESTART | DMA2D | DI_EN);
|
||||
ppi->ppi_control = params->ppi_control & ~PORT_EN;
|
||||
switch (info->type) {
|
||||
case PPI_TYPE_PPI:
|
||||
{
|
||||
struct bfin_ppi_regs *reg = info->base;
|
||||
|
||||
if (params->ppi_control & DMA32)
|
||||
dma32 = 1;
|
||||
|
||||
bfin_write16(®->control, ppi->ppi_control);
|
||||
bfin_write16(®->count, bytes_per_line - 1);
|
||||
bfin_write16(®->frame, lines_per_frame);
|
||||
break;
|
||||
}
|
||||
case PPI_TYPE_EPPI:
|
||||
{
|
||||
struct bfin_eppi_regs *reg = info->base;
|
||||
|
||||
if ((params->ppi_control & PACK_EN)
|
||||
|| (params->ppi_control & 0x38000) > DLEN_16)
|
||||
dma32 = 1;
|
||||
|
||||
bfin_write32(®->control, ppi->ppi_control);
|
||||
bfin_write16(®->line, bytes_per_line + params->blank_clocks);
|
||||
bfin_write16(®->frame, lines_per_frame);
|
||||
bfin_write16(®->hdelay, 0);
|
||||
bfin_write16(®->vdelay, 0);
|
||||
bfin_write16(®->hcount, bytes_per_line);
|
||||
bfin_write16(®->vcount, lines_per_frame);
|
||||
break;
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (dma32) {
|
||||
dma_config |= WDSIZE_32;
|
||||
set_dma_x_count(info->dma_ch, bytes_per_line >> 2);
|
||||
set_dma_x_modify(info->dma_ch, 4);
|
||||
set_dma_y_modify(info->dma_ch, 4);
|
||||
} else {
|
||||
dma_config |= WDSIZE_16;
|
||||
set_dma_x_count(info->dma_ch, bytes_per_line >> 1);
|
||||
set_dma_x_modify(info->dma_ch, 2);
|
||||
set_dma_y_modify(info->dma_ch, 2);
|
||||
}
|
||||
set_dma_y_count(info->dma_ch, lines_per_frame);
|
||||
set_dma_config(info->dma_ch, dma_config);
|
||||
|
||||
SSYNC();
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void ppi_update_addr(struct ppi_if *ppi, unsigned long addr)
|
||||
{
|
||||
set_dma_start_addr(ppi->info->dma_ch, addr);
|
||||
}
|
||||
|
||||
struct ppi_if *ppi_create_instance(const struct ppi_info *info)
|
||||
{
|
||||
struct ppi_if *ppi;
|
||||
|
||||
if (!info || !info->pin_req)
|
||||
return NULL;
|
||||
|
||||
if (peripheral_request_list(info->pin_req, KBUILD_MODNAME)) {
|
||||
pr_err("request peripheral failed\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
ppi = kzalloc(sizeof(*ppi), GFP_KERNEL);
|
||||
if (!ppi) {
|
||||
peripheral_free_list(info->pin_req);
|
||||
pr_err("unable to allocate memory for ppi handle\n");
|
||||
return NULL;
|
||||
}
|
||||
ppi->ops = &ppi_ops;
|
||||
ppi->info = info;
|
||||
|
||||
pr_info("ppi probe success\n");
|
||||
return ppi;
|
||||
}
|
||||
|
||||
void ppi_delete_instance(struct ppi_if *ppi)
|
||||
{
|
||||
peripheral_free_list(ppi->info->pin_req);
|
||||
kfree(ppi);
|
||||
}
|
37
include/media/blackfin/bfin_capture.h
Normal file
37
include/media/blackfin/bfin_capture.h
Normal file
|
@ -0,0 +1,37 @@
|
|||
#ifndef _BFIN_CAPTURE_H_
|
||||
#define _BFIN_CAPTURE_H_
|
||||
|
||||
#include <linux/i2c.h>
|
||||
|
||||
struct v4l2_input;
|
||||
struct ppi_info;
|
||||
|
||||
struct bcap_route {
|
||||
u32 input;
|
||||
u32 output;
|
||||
};
|
||||
|
||||
struct bfin_capture_config {
|
||||
/* card name */
|
||||
char *card_name;
|
||||
/* inputs available at the sub device */
|
||||
struct v4l2_input *inputs;
|
||||
/* number of inputs supported */
|
||||
int num_inputs;
|
||||
/* routing information for each input */
|
||||
struct bcap_route *routes;
|
||||
/* i2c bus adapter no */
|
||||
int i2c_adapter_id;
|
||||
/* i2c subdevice board info */
|
||||
struct i2c_board_info board_info;
|
||||
/* ppi board info */
|
||||
const struct ppi_info *ppi_info;
|
||||
/* ppi control */
|
||||
unsigned long ppi_control;
|
||||
/* ppi interrupt mask */
|
||||
u32 int_mask;
|
||||
/* horizontal blanking clocks */
|
||||
int blank_clocks;
|
||||
};
|
||||
|
||||
#endif
|
74
include/media/blackfin/ppi.h
Normal file
74
include/media/blackfin/ppi.h
Normal file
|
@ -0,0 +1,74 @@
|
|||
/*
|
||||
* Analog Devices PPI header file
|
||||
*
|
||||
* Copyright (c) 2011 Analog Devices Inc.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License version 2 as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#ifndef _PPI_H_
|
||||
#define _PPI_H_
|
||||
|
||||
#include <linux/interrupt.h>
|
||||
|
||||
#ifdef EPPI_EN
|
||||
#define PORT_EN EPPI_EN
|
||||
#define DMA32 0
|
||||
#define PACK_EN PACKEN
|
||||
#endif
|
||||
|
||||
struct ppi_if;
|
||||
|
||||
struct ppi_params {
|
||||
int width;
|
||||
int height;
|
||||
int bpp;
|
||||
unsigned long ppi_control;
|
||||
u32 int_mask;
|
||||
int blank_clocks;
|
||||
};
|
||||
|
||||
struct ppi_ops {
|
||||
int (*attach_irq)(struct ppi_if *ppi, irq_handler_t handler);
|
||||
void (*detach_irq)(struct ppi_if *ppi);
|
||||
int (*start)(struct ppi_if *ppi);
|
||||
int (*stop)(struct ppi_if *ppi);
|
||||
int (*set_params)(struct ppi_if *ppi, struct ppi_params *params);
|
||||
void (*update_addr)(struct ppi_if *ppi, unsigned long addr);
|
||||
};
|
||||
|
||||
enum ppi_type {
|
||||
PPI_TYPE_PPI,
|
||||
PPI_TYPE_EPPI,
|
||||
};
|
||||
|
||||
struct ppi_info {
|
||||
enum ppi_type type;
|
||||
int dma_ch;
|
||||
int irq_err;
|
||||
void __iomem *base;
|
||||
const unsigned short *pin_req;
|
||||
};
|
||||
|
||||
struct ppi_if {
|
||||
unsigned long ppi_control;
|
||||
const struct ppi_ops *ops;
|
||||
const struct ppi_info *info;
|
||||
bool err_int;
|
||||
void *priv;
|
||||
};
|
||||
|
||||
struct ppi_if *ppi_create_instance(const struct ppi_info *info);
|
||||
void ppi_delete_instance(struct ppi_if *ppi);
|
||||
#endif
|
Loading…
Reference in a new issue