Hi Hans, On 18 January 2016 at 10:02, Hans Verkuil <hverkuil@xxxxxxxxx> wrote: > Hi Ezequiel, > > Thanks for working on this! Do you know where I can get a board tw686x board? > I always like to have hardware to test the driver, if at all possible. > No, I don't know. I have one to spare, and I could send it to you. > See below for a review of this driver. > > On 12/27/2015 03:26 AM, Ezequiel Garcia wrote: >> This commit introduces the support for the Techwell TW686x video >> capture IC. This hardware supports a few DMA modes, including >> scatter-gather and frame (contiguous). >> >> This commit supports only DMA frame (contiguous) mode. >> DMA scatter-gather mode support may be added in the future. >> >> Currently supported chips: >> - TW6864 (4 video channels), >> - TW6865 (4 video channels, not tested, second generation chip), >> - TW6868 (8 video channels but only 4 first channels using >> built-in video decoder are supported, not tested), >> - TW6869 (8 video channels, second generation chip). >> >> Cc: Krzysztof Hałasa <khalasa@xxxxxxx> >> Signed-off-by: Ezequiel Garcia <ezequiel@xxxxxxxxxxxxxxxxxxxx> >> --- >> This patchset superseeds Krzysztof's original patch: >> https://patchwork.linuxtv.org/patch/30448/ >> >> Tested on a custom TW6869-based capture card. >> Latest v4l2-compliance test pass: >> >> $ /usr/bin/v4l2-compliance -f -s >> Driver Info: >> Driver name : tw686x >> Card type : tw6869 >> Bus info : PCI:0000:04:00.0 >> Driver version: 4.4.0 >> Capabilities : 0x85200001 >> Video Capture >> Read/Write >> Streaming >> Extended Pix Format >> Device Capabilities >> Device Caps : 0x05200001 >> Video Capture >> Read/Write >> Streaming >> Extended Pix Format >> >> Compliance test for device /dev/video0 (not using libv4l2): >> >> [..] >> Test input 0: >> >> Streaming ioctls: >> test read/write: OK >> test MMAP: OK >> test USERPTR: OK (Not Supported) >> test DMABUF: Cannot test, specify --expbuf-device >> >> Stream using all formats: >> test MMAP for Format UYVY, Frame Size 360x240: >> Stride 720, Field Interlaced: OK >> test MMAP for Format UYVY, Frame Size 720x480: >> Stride 1440, Field Interlaced: OK >> test MMAP for Format RGBP, Frame Size 360x240: >> Stride 720, Field Interlaced: OK >> test MMAP for Format RGBP, Frame Size 720x480: >> Stride 1440, Field Interlaced: OK >> test MMAP for Format YUYV, Frame Size 360x240: >> Stride 720, Field Interlaced: OK >> test MMAP for Format YUYV, Frame Size 720x480: >> Stride 1440, Field Interlaced: OK >> >> Total: 114, Succeeded: 114, Failed: 0, Warnings: 0 >> --- >> MAINTAINERS | 8 + >> drivers/media/pci/Kconfig | 1 + >> drivers/media/pci/Makefile | 1 + >> drivers/media/pci/tw686x/Kconfig | 18 + >> drivers/media/pci/tw686x/Makefile | 3 + >> drivers/media/pci/tw686x/tw686x-audio.c | 379 +++++++++++++ >> drivers/media/pci/tw686x/tw686x-core.c | 356 ++++++++++++ >> drivers/media/pci/tw686x/tw686x-regs.h | 118 ++++ >> drivers/media/pci/tw686x/tw686x-video.c | 953 ++++++++++++++++++++++++++++++++ >> drivers/media/pci/tw686x/tw686x.h | 189 +++++++ >> 10 files changed, 2026 insertions(+) >> create mode 100644 drivers/media/pci/tw686x/Kconfig >> create mode 100644 drivers/media/pci/tw686x/Makefile >> create mode 100644 drivers/media/pci/tw686x/tw686x-audio.c >> create mode 100644 drivers/media/pci/tw686x/tw686x-core.c >> create mode 100644 drivers/media/pci/tw686x/tw686x-regs.h >> create mode 100644 drivers/media/pci/tw686x/tw686x-video.c >> create mode 100644 drivers/media/pci/tw686x/tw686x.h >> >> diff --git a/MAINTAINERS b/MAINTAINERS >> index 4635e1d14612..9308f8242cd3 100644 >> --- a/MAINTAINERS >> +++ b/MAINTAINERS >> @@ -10846,6 +10846,14 @@ W: https://linuxtv.org >> S: Odd Fixes >> F: drivers/media/pci/tw68/ >> >> +TW686X VIDEO4LINUX DRIVER >> +M: Ezequiel Garcia <ezequiel@xxxxxxxxxxxxxxxxxxxx> >> +L: linux-media@xxxxxxxxxxxxxxx >> +T: git git://linuxtv.org/media_tree.git >> +W: http://linuxtv.org >> +S: Maintained >> +F: drivers/media/pci/tw686x/ >> + >> TPM DEVICE DRIVER >> M: Peter Huewe <peterhuewe@xxxxxx> >> M: Marcel Selhorst <tpmdd@xxxxxxxxxxxx> >> diff --git a/drivers/media/pci/Kconfig b/drivers/media/pci/Kconfig >> index 48a611bc3e18..4f6467fbaeb4 100644 >> --- a/drivers/media/pci/Kconfig >> +++ b/drivers/media/pci/Kconfig >> @@ -14,6 +14,7 @@ source "drivers/media/pci/meye/Kconfig" >> source "drivers/media/pci/solo6x10/Kconfig" >> source "drivers/media/pci/sta2x11/Kconfig" >> source "drivers/media/pci/tw68/Kconfig" >> +source "drivers/media/pci/tw686x/Kconfig" >> source "drivers/media/pci/zoran/Kconfig" >> endif >> >> diff --git a/drivers/media/pci/Makefile b/drivers/media/pci/Makefile >> index 5f8aacb8b9b8..2e54c36441f7 100644 >> --- a/drivers/media/pci/Makefile >> +++ b/drivers/media/pci/Makefile >> @@ -25,6 +25,7 @@ obj-$(CONFIG_VIDEO_BT848) += bt8xx/ >> obj-$(CONFIG_VIDEO_SAA7134) += saa7134/ >> obj-$(CONFIG_VIDEO_SAA7164) += saa7164/ >> obj-$(CONFIG_VIDEO_TW68) += tw68/ >> +obj-$(CONFIG_VIDEO_TW686X) += tw686x/ >> obj-$(CONFIG_VIDEO_DT3155) += dt3155/ >> obj-$(CONFIG_VIDEO_MEYE) += meye/ >> obj-$(CONFIG_STA2X11_VIP) += sta2x11/ >> diff --git a/drivers/media/pci/tw686x/Kconfig b/drivers/media/pci/tw686x/Kconfig >> new file mode 100644 >> index 000000000000..bd73d3dd25b4 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/Kconfig >> @@ -0,0 +1,18 @@ >> +config VIDEO_TW686X >> + tristate "Intersil/Techwell TW686x video capture cards" >> + depends on PCI && VIDEO_DEV && VIDEO_V4L2 && SND >> + depends on HAS_DMA >> + select VIDEOBUF2_DMA_CONTIG >> + select SND_PCM >> + help >> + Support for Intersil/Techwell TW686x-based frame grabber cards. >> + >> + Currently supported chips: >> + - TW6864 (4 video channels), >> + - TW6865 (4 video channels, not tested, second generation chip), >> + - TW6868 (8 video channels but only 4 first channels using >> + built-in video decoder are supported, not tested), >> + - TW6869 (8 video channels, second generation chip). >> + >> + To compile this driver as a module, choose M here: the module >> + will be named tw686x. >> diff --git a/drivers/media/pci/tw686x/Makefile b/drivers/media/pci/tw686x/Makefile >> new file mode 100644 >> index 000000000000..99819542b733 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/Makefile >> @@ -0,0 +1,3 @@ >> +tw686x-objs := tw686x-core.o tw686x-video.o tw686x-audio.o >> + >> +obj-$(CONFIG_VIDEO_TW686X) += tw686x.o >> diff --git a/drivers/media/pci/tw686x/tw686x-audio.c b/drivers/media/pci/tw686x/tw686x-audio.c >> new file mode 100644 >> index 000000000000..2295d2d1a945 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/tw686x-audio.c >> @@ -0,0 +1,379 @@ >> +/* >> + * Copyright (C) 2015 VanguardiaSur - www.vanguardiasur.com.ar >> + * >> + * Based on the audio support from the tw6869 driver: >> + * Copyright 2015 www.starterkit.ru <info@xxxxxxxxxxxxx> >> + * >> + * Based on: >> + * Driver for Intersil|Techwell TW6869 based DVR cards >> + * (c) 2011-12 liran <jli11@xxxxxxxxxxxx> [Intersil|Techwell China] >> + * >> + * This program is free software; you can redistribute it and/or modify it >> + * under the terms of version 2 of the GNU General Public License >> + * as published by the Free Software Foundation. >> + */ >> + >> +#include <linux/types.h> >> +#include <linux/kernel.h> >> +#include <linux/module.h> >> +#include <linux/init.h> >> +#include <linux/kmod.h> >> +#include <linux/mutex.h> >> +#include <linux/pci.h> >> +#include <linux/delay.h> >> + >> +#include <sound/core.h> >> +#include <sound/initval.h> >> +#include <sound/pcm.h> >> +#include <sound/control.h> >> +#include "tw686x.h" >> +#include "tw686x-regs.h" >> + >> +#define AUDIO_CHANNEL_OFFSET 8 >> + >> +void tw686x_audio_irq(struct tw686x_dev *dev, unsigned long requests, >> + unsigned int pb_status) >> +{ >> + unsigned long flags; >> + unsigned int ch, pb; >> + >> + for_each_set_bit(ch, &requests, max_channels(dev)) { >> + > > Remove this empty line. > Sure. >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ch]; >> + struct tw686x_audio_buf *done = NULL; >> + struct tw686x_audio_buf *next = NULL; >> + struct tw686x_dma_desc *desc; >> + >> + pb = !!(pb_status & BIT(AUDIO_CHANNEL_OFFSET + ch)); >> + >> + spin_lock_irqsave(&ac->lock, flags); >> + >> + /* Sanity check */ >> + if (!ac->ss || !ac->curr_bufs[0] || !ac->curr_bufs[1]) { >> + spin_unlock_irqrestore(&ac->lock, flags); >> + continue; >> + } >> + >> + if (!list_empty(&ac->buf_list)) { >> + next = list_first_entry(&ac->buf_list, >> + struct tw686x_audio_buf, list); >> + list_move_tail(&next->list, &ac->buf_list); >> + done = ac->curr_bufs[!pb]; >> + ac->curr_bufs[pb] = next; >> + } >> + spin_unlock_irqrestore(&ac->lock, flags); >> + >> + desc = &ac->dma_descs[pb]; >> + if (done && next && desc->virt) { >> + memcpy(done->virt, desc->virt, desc->size); >> + ac->ptr = done->dma - ac->buf[0].dma; >> + snd_pcm_period_elapsed(ac->ss); >> + } >> + } >> +} >> + >> +static int tw686x_pcm_hw_params(struct snd_pcm_substream *ss, >> + struct snd_pcm_hw_params *hw_params) >> +{ >> + return snd_pcm_lib_malloc_pages(ss, params_buffer_bytes(hw_params)); >> +} >> + >> +static int tw686x_pcm_hw_free(struct snd_pcm_substream *ss) >> +{ >> + return snd_pcm_lib_free_pages(ss); >> +} >> + >> +static const struct snd_pcm_hardware tw686x_capture_hw = { >> + .info = (SNDRV_PCM_INFO_MMAP | >> + SNDRV_PCM_INFO_INTERLEAVED | >> + SNDRV_PCM_INFO_BLOCK_TRANSFER | >> + SNDRV_PCM_INFO_MMAP_VALID), >> + .formats = SNDRV_PCM_FMTBIT_S16_LE, >> + .rates = SNDRV_PCM_RATE_8000_48000, >> + .rate_min = 8000, >> + .rate_max = 48000, >> + .channels_min = 1, >> + .channels_max = 1, >> + .buffer_bytes_max = TW686X_AUDIO_PAGE_MAX * TW686X_PAGE_SIZE, >> + .period_bytes_min = TW686X_PAGE_SIZE, >> + .period_bytes_max = TW686X_PAGE_SIZE, >> + .periods_min = 2, >> + .periods_max = TW686X_AUDIO_PAGE_MAX, >> +}; >> + >> +static int tw686x_pcm_open(struct snd_pcm_substream *ss) >> +{ >> + struct tw686x_dev *dev = snd_pcm_substream_chip(ss); >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ss->number]; >> + struct snd_pcm_runtime *rt = ss->runtime; >> + int err; >> + >> + ac->ss = ss; >> + rt->hw = tw686x_capture_hw; >> + >> + err = snd_pcm_hw_constraint_integer(rt, SNDRV_PCM_HW_PARAM_PERIODS); >> + if (err < 0) >> + return err; >> + >> + return 0; >> +} >> + >> +static int tw686x_pcm_close(struct snd_pcm_substream *ss) >> +{ >> + struct tw686x_dev *dev = snd_pcm_substream_chip(ss); >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ss->number]; >> + >> + ac->ss = NULL; >> + return 0; >> +} >> + >> +static int tw686x_pcm_prepare(struct snd_pcm_substream *ss) >> +{ >> + struct tw686x_dev *dev = snd_pcm_substream_chip(ss); >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ss->number]; >> + struct snd_pcm_runtime *rt = ss->runtime; >> + unsigned int period_size = snd_pcm_lib_period_bytes(ss); >> + struct tw686x_audio_buf *p_buf, *b_buf; >> + unsigned long flags; >> + int i; >> + >> + spin_lock_irqsave(&dev->lock, flags); >> + tw686x_disable_channel(dev, AUDIO_CHANNEL_OFFSET + ac->ch); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + if (dev->audio_rate != rt->rate) { >> + u32 reg; >> + >> + dev->audio_rate = rt->rate; >> + reg = ((125000000 / rt->rate) << 16) + >> + ((125000000 % rt->rate) << 16) / rt->rate; >> + >> + reg_write(dev, AUDIO_CONTROL2, reg); >> + } >> + >> + if ((period_size != TW686X_PAGE_SIZE) || >> + (rt->periods < 2) || (rt->periods > TW686X_AUDIO_PAGE_MAX)) { > > There is no need for the extra parenthesis around simple comparisons. > OK. >> + return -EINVAL; >> + } >> + >> + spin_lock_irqsave(&ac->lock, flags); >> + INIT_LIST_HEAD(&ac->buf_list); >> + >> + for (i = 0; i < rt->periods; i++) { >> + ac->buf[i].dma = rt->dma_addr + period_size * i; >> + ac->buf[i].virt = rt->dma_area + period_size * i; >> + INIT_LIST_HEAD(&ac->buf[i].list); >> + list_add_tail(&ac->buf[i].list, &ac->buf_list); >> + } >> + >> + p_buf = list_first_entry(&ac->buf_list, struct tw686x_audio_buf, list); >> + list_move_tail(&p_buf->list, &ac->buf_list); >> + >> + b_buf = list_first_entry(&ac->buf_list, struct tw686x_audio_buf, list); >> + list_move_tail(&b_buf->list, &ac->buf_list); >> + >> + ac->curr_bufs[0] = p_buf; >> + ac->curr_bufs[1] = b_buf; >> + ac->ptr = 0; >> + spin_unlock_irqrestore(&ac->lock, flags); >> + >> + return 0; >> +} >> + >> +static int tw686x_pcm_trigger(struct snd_pcm_substream *ss, int cmd) >> +{ >> + struct tw686x_dev *dev = snd_pcm_substream_chip(ss); >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ss->number]; >> + unsigned long flags; >> + int err = 0; >> + >> + switch (cmd) { >> + case SNDRV_PCM_TRIGGER_START: >> + if (ac->curr_bufs[0] && ac->curr_bufs[1]) { >> + spin_lock_irqsave(&dev->lock, flags); >> + tw686x_enable_channel(dev, AUDIO_CHANNEL_OFFSET + ac->ch); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + mod_timer(&dev->dma_delay_timer, >> + jiffies + msecs_to_jiffies(100)); >> + } else { >> + err = -EIO; >> + } >> + break; >> + case SNDRV_PCM_TRIGGER_STOP: >> + spin_lock_irqsave(&dev->lock, flags); >> + tw686x_disable_channel(dev, AUDIO_CHANNEL_OFFSET + ac->ch); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + spin_lock_irqsave(&ac->lock, flags); >> + ac->curr_bufs[0] = NULL; >> + ac->curr_bufs[1] = NULL; >> + spin_unlock_irqrestore(&ac->lock, flags); >> + break; >> + default: >> + err = -EINVAL; >> + } >> + return err; >> +} >> + >> +static snd_pcm_uframes_t tw686x_pcm_pointer(struct snd_pcm_substream *ss) >> +{ >> + struct tw686x_dev *dev = snd_pcm_substream_chip(ss); >> + struct tw686x_audio_channel *ac = &dev->audio_channels[ss->number]; >> + >> + return bytes_to_frames(ss->runtime, ac->ptr); >> +} >> + >> +static struct snd_pcm_ops tw686x_pcm_ops = { >> + .open = tw686x_pcm_open, >> + .close = tw686x_pcm_close, >> + .ioctl = snd_pcm_lib_ioctl, >> + .hw_params = tw686x_pcm_hw_params, >> + .hw_free = tw686x_pcm_hw_free, >> + .prepare = tw686x_pcm_prepare, >> + .trigger = tw686x_pcm_trigger, >> + .pointer = tw686x_pcm_pointer, >> +}; >> + >> +static int tw686x_snd_pcm_init(struct tw686x_dev *dev) >> +{ >> + struct snd_card *card = dev->snd_card; >> + struct snd_pcm *pcm; >> + struct snd_pcm_substream *ss; >> + unsigned int i; >> + int err; >> + >> + err = snd_pcm_new(card, card->driver, 0, 0, max_channels(dev), &pcm); >> + if (err < 0) >> + return err; >> + >> + snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &tw686x_pcm_ops); >> + snd_pcm_chip(pcm) = dev; >> + pcm->info_flags = 0; >> + strcpy(pcm->name, "tw686x PCM"); > > strlcpy > OK. >> + >> + for (i = 0, ss = pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream; >> + ss; ss = ss->next, i++) >> + sprintf(ss->name, "vch%u audio", i); > > snprintf > OK. >> + >> + return snd_pcm_lib_preallocate_pages_for_all(pcm, >> + SNDRV_DMA_TYPE_DEV, >> + snd_dma_pci_data(dev->pci_dev), >> + TW686X_AUDIO_PAGE_MAX * TW686X_PAGE_SIZE, >> + TW686X_AUDIO_PAGE_MAX * TW686X_PAGE_SIZE); >> +} >> + >> +static void tw686x_audio_dma_free(struct tw686x_dev *dev, >> + struct tw686x_audio_channel *ac) >> +{ >> + int pb; >> + >> + for (pb = 0; pb < 2; pb++) { >> + if (!ac->dma_descs[pb].virt) >> + continue; >> + pci_free_consistent(dev->pci_dev, ac->dma_descs[pb].size, >> + ac->dma_descs[pb].virt, >> + ac->dma_descs[pb].phys); >> + ac->dma_descs[pb].virt = NULL; >> + } >> +} >> + >> +static int tw686x_audio_dma_alloc(struct tw686x_dev *dev, >> + struct tw686x_audio_channel *ac) >> +{ >> + int pb; >> + >> + for (pb = 0; pb < 2; pb++) { >> + u32 reg = pb ? ADMA_B_ADDR[ac->ch] : ADMA_P_ADDR[ac->ch]; >> + void *virt; >> + >> + virt = pci_alloc_consistent(dev->pci_dev, TW686X_PAGE_SIZE, >> + &ac->dma_descs[pb].phys); >> + if (!virt) { >> + dev_err(&dev->pci_dev->dev, >> + "dma%d: unable to allocate audio DMA %s-buffer\n", >> + ac->ch, pb ? "B" : "P"); >> + return -ENOMEM; >> + } >> + ac->dma_descs[pb].virt = virt; >> + ac->dma_descs[pb].size = TW686X_PAGE_SIZE; >> + reg_write(dev, reg, ac->dma_descs[pb].phys); >> + } >> + return 0; >> +} >> + >> +void tw686x_audio_free(struct tw686x_dev *dev) >> +{ >> + unsigned long flags; >> + u32 dma_ch_mask; >> + u32 dma_cmd; >> + >> + spin_lock_irqsave(&dev->lock, flags); >> + dma_cmd = reg_read(dev, DMA_CMD); >> + dma_ch_mask = reg_read(dev, DMA_CHANNEL_ENABLE); >> + reg_write(dev, DMA_CMD, dma_cmd & ~0xff00); >> + reg_write(dev, DMA_CHANNEL_ENABLE, dma_ch_mask & ~0xff00); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + if (!dev->snd_card) >> + return; >> + snd_card_free(dev->snd_card); >> + dev->snd_card = NULL; >> +} >> + >> +int tw686x_audio_init(struct tw686x_dev *dev) >> +{ >> + struct pci_dev *pci_dev = dev->pci_dev; >> + struct snd_card *card; >> + int err, ch; >> + >> + /* >> + * AUDIO_CONTROL1 >> + * DMA byte length [31:19] = 4096 (i.e. ALSA period) >> + * External audio enable [0] = enabled >> + */ >> + reg_write(dev, AUDIO_CONTROL1, 0x80000001); >> + >> + err = snd_card_new(&pci_dev->dev, SNDRV_DEFAULT_IDX1, >> + SNDRV_DEFAULT_STR1, >> + THIS_MODULE, 0, &card); >> + if (err < 0) >> + return err; >> + >> + dev->snd_card = card; >> + strcpy(card->driver, "tw686x"); >> + strcpy(card->shortname, "tw686x"); >> + sprintf(card->longname, "%s", pci_name(pci_dev)); > > Use strlcpy and snprintf. > OK. >> + snd_card_set_dev(card, &pci_dev->dev); >> + >> + for (ch = 0; ch < max_channels(dev); ch++) { >> + struct tw686x_audio_channel *ac; >> + >> + ac = &dev->audio_channels[ch]; >> + spin_lock_init(&ac->lock); >> + ac->dev = dev; >> + ac->ch = ch; >> + >> + err = tw686x_audio_dma_alloc(dev, ac); >> + if (err < 0) >> + goto err_cleanup; >> + } >> + >> + err = tw686x_snd_pcm_init(dev); >> + if (err < 0) >> + goto err_cleanup; >> + >> + err = snd_card_register(card); >> + if (!err) >> + return 0; >> + >> +err_cleanup: >> + for (ch = 0; ch < max_channels(dev); ch++) { >> + if (!dev->audio_channels[ch].dev) >> + continue; >> + tw686x_audio_dma_free(dev, &dev->audio_channels[ch]); >> + } >> + snd_card_free(card); >> + dev->snd_card = NULL; >> + return err; >> +} >> diff --git a/drivers/media/pci/tw686x/tw686x-core.c b/drivers/media/pci/tw686x/tw686x-core.c >> new file mode 100644 >> index 000000000000..8774facb29d5 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/tw686x-core.c >> @@ -0,0 +1,356 @@ >> +/* >> + * Copyright (C) 2015 VanguardiaSur - www.vanguardiasur.com.ar >> + * >> + * Based on original driver by Krzysztof Hałasa: >> + * Copyright (C) 2015 Industrial Research Institute for Automation >> + * and Measurements PIAP >> + * >> + * This program is free software; you can redistribute it and/or modify it >> + * under the terms of version 2 of the GNU General Public License >> + * as published by the Free Software Foundation. >> + */ >> + >> +#include <linux/init.h> >> +#include <linux/interrupt.h> >> +#include <linux/delay.h> >> +#include <linux/kernel.h> >> +#include <linux/module.h> >> +#include <linux/pci_ids.h> >> +#include <linux/slab.h> >> +#include <linux/timer.h> >> + >> +#include "tw686x.h" >> +#include "tw686x-regs.h" >> + >> +static u32 dma_interval = 0x00098968; >> +module_param(dma_interval, int, 0444); >> +MODULE_PARM_DESC(dma_interval, "Minimum time span for DMA interrupting host"); >> + >> +/* >> + * The purpose of this awful hack is to avoid >> + * enabling the DMA channels "too fast" which seems >> + * to make some TW686x devices _very_ angry and freeze the CPU. >> + */ >> +static void tw686x_dma_delay(unsigned long data) >> +{ >> + struct tw686x_dev *dev = (struct tw686x_dev *) data; > > Weird indentation, spaces instead of tab. Remove space after cast. > Ouch, I'll fix it. >> + unsigned long flags; >> + >> + spin_lock_irqsave(&dev->lock, flags); >> + >> + reg_write(dev, DMA_CHANNEL_ENABLE, dev->pending_dma_en); >> + reg_write(dev, DMA_CMD, dev->pending_dma_cmd); >> + dev->pending_dma_en = 0; >> + dev->pending_dma_cmd = 0; >> + >> + spin_unlock_irqrestore(&dev->lock, flags); >> +} >> + >> +static void tw686x_reset_channels(struct tw686x_dev *dev, unsigned int ch_mask) >> +{ >> + u32 dma_en, dma_cmd; >> + >> + dma_en = reg_read(dev, DMA_CHANNEL_ENABLE); >> + dma_cmd = reg_read(dev, DMA_CMD); >> + >> + /* >> + * Save pending register status, the timer will >> + * restore them. >> + */ >> + dev->pending_dma_en |= dma_en; >> + dev->pending_dma_cmd |= dma_cmd; >> + >> + /* Disable the reset channels */ >> + reg_write(dev, DMA_CHANNEL_ENABLE, dma_en & ~ch_mask); >> + >> + if ((dma_en & ~ch_mask) == 0) { >> + dev_dbg(&dev->pci_dev->dev, "reset: stopping DMA\n"); >> + dma_cmd &= ~BIT(31); >> + } >> + reg_write(dev, DMA_CMD, dma_cmd & ~ch_mask); >> +} >> + >> +static irqreturn_t tw686x_irq(int irq, void *dev_id) >> +{ >> + struct tw686x_dev *dev = (struct tw686x_dev *)dev_id; >> + unsigned int video_requests, audio_requests, reset_ch; >> + u32 fifo_status, fifo_signal, fifo_ov, fifo_bad, fifo_errors; >> + u32 int_status, dma_en, video_en, pb_status; >> + unsigned long flags; >> + >> + int_status = reg_read(dev, INT_STATUS); /* cleared on read */ >> + fifo_status = reg_read(dev, VIDEO_FIFO_STATUS); >> + >> + /* INT_STATUS does not include FIFO_STATUS errors! */ >> + if (!int_status && !TW686X_FIFO_ERROR(fifo_status)) >> + return IRQ_NONE; >> + >> + if (int_status & BIT(17)) { > > Either use a define for BIT(17) or add a comment so I know what bit 17 is. > Sure, I'll define it. >> + dev_dbg(&dev->pci_dev->dev, >> + "DMA timeout. Resetting DMA for all channels\n"); >> + reset_ch = ~0; >> + goto reset_channels; >> + } >> + >> + spin_lock_irqsave(&dev->lock, flags); >> + dma_en = reg_read(dev, DMA_CHANNEL_ENABLE); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + video_en = dma_en & 0xff; >> + fifo_signal = ~(fifo_status & 0xff) & video_en; >> + fifo_ov = fifo_status >> 24; >> + fifo_bad = fifo_status >> 16; >> + >> + /* Mask of channels with signal and FIFO errors */ >> + fifo_errors = fifo_signal & (fifo_ov | fifo_bad); >> + >> + reset_ch = 0; >> + pb_status = reg_read(dev, PB_STATUS); >> + >> + /* Coalesce video frame/error events */ >> + video_requests = (int_status & video_en) | fifo_errors; >> + audio_requests = (int_status & dma_en) >> 8; >> + >> + if (video_requests) >> + tw686x_video_irq(dev, video_requests, pb_status, >> + fifo_status, &reset_ch); >> + if (audio_requests) >> + tw686x_audio_irq(dev, audio_requests, pb_status); >> + >> +reset_channels: >> + if (reset_ch) { >> + spin_lock_irqsave(&dev->lock, flags); >> + tw686x_reset_channels(dev, reset_ch); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + mod_timer(&dev->dma_delay_timer, >> + jiffies + msecs_to_jiffies(100)); >> + } >> + >> + return IRQ_HANDLED; >> +} >> + >> +static void tw686x_dev_release(struct v4l2_device *v4l2_dev) >> +{ >> + struct tw686x_dev *dev = container_of(v4l2_dev, struct tw686x_dev, >> + v4l2_dev); >> + unsigned int ch; >> + >> + for (ch = 0; ch < max_channels(dev); ch++) >> + v4l2_ctrl_handler_free(&dev->video_channels[ch].ctrl_handler); >> + >> + v4l2_device_unregister(&dev->v4l2_dev); >> + >> + kfree(dev->audio_channels); >> + kfree(dev->video_channels); >> + kfree(dev); >> +} >> + >> +static int tw686x_probe(struct pci_dev *pci_dev, >> + const struct pci_device_id *pci_id) >> +{ >> + struct tw686x_dev *dev; >> + int err; >> + >> + dev = kzalloc(sizeof(*dev), GFP_KERNEL); >> + if (!dev) >> + return -ENOMEM; >> + dev->type = pci_id->driver_data; >> + sprintf(dev->name, "tw%04X", pci_dev->device); >> + >> + dev->video_channels = kcalloc(max_channels(dev), >> + sizeof(*dev->video_channels), GFP_KERNEL); >> + if (!dev->video_channels) { >> + err = -ENOMEM; >> + goto free_dev; >> + } >> + >> + dev->audio_channels = kcalloc(max_channels(dev), >> + sizeof(*dev->audio_channels), GFP_KERNEL); >> + if (!dev->audio_channels) { >> + err = -ENOMEM; >> + goto free_video; >> + } >> + >> + pr_info("%s: PCI %s, IRQ %d, MMIO 0x%lx\n", dev->name, >> + pci_name(pci_dev), pci_dev->irq, >> + (unsigned long)pci_resource_start(pci_dev, 0)); >> + >> + dev->pci_dev = pci_dev; >> + if (pci_enable_device(pci_dev)) { >> + err = -EIO; >> + goto free_audio; >> + } >> + >> + pci_set_master(pci_dev); >> + err = pci_set_dma_mask(pci_dev, DMA_BIT_MASK(32)); >> + if (err) { >> + dev_err(&pci_dev->dev, "32-bit PCI DMA not supported\n"); >> + err = -EIO; >> + goto disable_pci; >> + } >> + >> + err = pci_request_regions(pci_dev, dev->name); >> + if (err) { >> + dev_err(&pci_dev->dev, "unable to request PCI region\n"); >> + goto disable_pci; >> + } >> + >> + dev->mmio = pci_ioremap_bar(pci_dev, 0); >> + if (!dev->mmio) { >> + dev_err(&pci_dev->dev, "unable to remap PCI region\n"); >> + err = -ENOMEM; >> + goto free_region; >> + } >> + >> + /* Reset all subsystems */ >> + reg_write(dev, SYS_SOFT_RST, 0x0F); > > The codingstyle for the kernel is to use lower case hex values, so 0x0f. > > Please change this throughout the source code. > Sure. >> + mdelay(1); >> + >> + reg_write(dev, SRST[0], 0x3F); >> + if (max_channels(dev) > 4) >> + reg_write(dev, SRST[1], 0x3F); >> + >> + /* Disable the DMA engine */ >> + reg_write(dev, DMA_CMD, 0); >> + reg_write(dev, DMA_CHANNEL_ENABLE, 0); >> + >> + /* Enable DMA FIFO overflow and pointer check */ >> + reg_write(dev, DMA_CONFIG, 0xFFFFFF04); >> + reg_write(dev, DMA_CHANNEL_TIMEOUT, 0x140C8584); >> + reg_write(dev, DMA_TIMER_INTERVAL, dma_interval); >> + >> + spin_lock_init(&dev->lock); >> + >> + err = request_irq(pci_dev->irq, tw686x_irq, IRQF_SHARED, >> + dev->name, dev); >> + if (err < 0) { >> + dev_err(&pci_dev->dev, "unable to request interrupt\n"); >> + goto iounmap; >> + } >> + >> + setup_timer(&dev->dma_delay_timer, >> + tw686x_dma_delay, (unsigned long) dev); >> + >> + /* >> + * This must be set right before initializing v4l2_dev. >> + * It's used to release resources after the last handle >> + * held is released. >> + */ >> + dev->v4l2_dev.release = tw686x_dev_release; >> + err = tw686x_video_init(dev); >> + if (err) { >> + dev_err(&pci_dev->dev, "can't register video\n"); >> + goto free_irq; >> + } >> + >> + err = tw686x_audio_init(dev); >> + if (err) { >> + dev_warn(&pci_dev->dev, "can't register audio\n"); >> + } >> + >> + pci_set_drvdata(pci_dev, dev); >> + return 0; >> + >> +free_irq: >> + free_irq(pci_dev->irq, dev); >> +iounmap: >> + pci_iounmap(pci_dev, dev->mmio); >> +free_region: >> + pci_release_regions(pci_dev); >> +disable_pci: >> + pci_disable_device(pci_dev); >> +free_audio: >> + kfree(dev->audio_channels); >> +free_video: >> + kfree(dev->video_channels); >> +free_dev: >> + kfree(dev); >> + return err; >> +} >> + >> +static void tw686x_remove(struct pci_dev *pci_dev) >> +{ >> + struct tw686x_dev *dev = pci_get_drvdata(pci_dev); >> + unsigned long flags; >> + >> + /* This guarantees the IRQ handler is no longer running, >> + * which means we can kiss good-bye some resources. >> + */ >> + free_irq(pci_dev->irq, dev); >> + >> + tw686x_video_free(dev); >> + tw686x_audio_free(dev); >> + del_timer_sync(&dev->dma_delay_timer); >> + >> + pci_iounmap(pci_dev, dev->mmio); >> + pci_release_regions(pci_dev); >> + pci_disable_device(pci_dev); >> + >> + /* >> + * This allows to detect device is not here, >> + * and will be used by vb2_ops. The lock is really >> + * important here. >> + */ >> + spin_lock_irqsave(&dev->lock, flags); >> + dev->pci_dev = NULL; >> + spin_unlock_irqrestore(&dev->lock, flags); > > As you sure this is needed? Normally you can only come here if the module > is removed, which isn't possible as long as userspace is using it. And if > the module is removed, then vb2 shouldn't be called at all. > > The only exception would be if this is a hot-pluggable device, which is > quite unlikely for a PCI device. I don't believe any of the pci drivers > support that. > A previous version of the driver didn't have that. However, under certain stress testing it was observed that the PCIe link goes down. I still have the traces for that: [..] [21833.389031] pciehp 0000:13:01.0:pcie24: pcie_isr: intr_loc 100 [21833.389035] pciehp 0000:13:01.0:pcie24: Data Link Layer State change [21833.389038] pciehp 0000:13:01.0:pcie24: slot(1-5): Link Down event [21833.389076] pciehp 0000:13:01.0:pcie24: Disabling domain:bus:device=0000:14:00 [21833.389078] pciehp 0000:13:01.0:pcie24: pciehp_unconfigure_device: domain:bus:dev = 0000:14:00 [21833.389103] TW686x 0000:14:00.0: removing [21833.416557] TW686x 0000:14:00.0: removed [..] I have no idea why the link goes down (hardware issue?), but it's better to handle it gracefully :) > Dubious code. > >> + >> + /* >> + * This calls tw686x_dev_release if it's the last reference. >> + * therwise, release is posponed until there are no users left. > > Typo: Otherwise ... postponed > Right. >> + */ >> + v4l2_device_put(&dev->v4l2_dev); > > Again, spaces instead of tabs. > Will fix. >> +} >> + >> +/* >> + * On TW6864 and TW6868, all channels share the pair of video DMA SG tables, >> + * with 10-bit start_idx and end_idx determining start and end of frame buffer >> + * for particular channel. >> + * TW6868 with all its 8 channels would be problematic (only 127 SG entries per >> + * channel) but we support only 4 channels on this chip anyway (the first >> + * 4 channels are driven with internal video decoder, the other 4 would require >> + * an external TW286x part). >> + * >> + * On TW6865 and TW6869, each channel has its own DMA SG table, with indexes >> + * starting with 0. Both chips have complete sets of internal video decoders >> + * (respectively 4 or 8-channel). >> + * >> + * All chips have separate SG tables for two video frames. >> + */ >> + >> +/* driver_data is number of A/V channels */ >> +static const struct pci_device_id tw686x_pci_tbl[] = { >> + { >> + PCI_DEVICE(PCI_VENDOR_ID_TECHWELL, 0x6864), >> + .driver_data = 4 >> + }, >> + { >> + PCI_DEVICE(PCI_VENDOR_ID_TECHWELL, 0x6865), /* not tested */ >> + .driver_data = 4 | TYPE_SECOND_GEN >> + }, >> + /* >> + * TW6868 supports 8 A/V channels with an external TW2865 chip; >> + * not supported by the driver. >> + */ >> + { >> + PCI_DEVICE(PCI_VENDOR_ID_TECHWELL, 0x6868), /* not tested */ >> + .driver_data = 4 >> + }, >> + { >> + PCI_DEVICE(PCI_VENDOR_ID_TECHWELL, 0x6869), >> + .driver_data = 8 | TYPE_SECOND_GEN}, >> + {} >> +}; >> +MODULE_DEVICE_TABLE(pci, tw686x_pci_tbl); >> + >> +static struct pci_driver tw686x_pci_driver = { >> + .name = "tw686x", >> + .id_table = tw686x_pci_tbl, >> + .probe = tw686x_probe, >> + .remove = tw686x_remove, >> +}; >> +module_pci_driver(tw686x_pci_driver); >> + >> +MODULE_DESCRIPTION("Driver for video frame grabber cards based on Intersil/Techwell TW686[4589]"); >> +MODULE_AUTHOR("Ezequiel Garcia <ezequiel@xxxxxxxxxxxxxxxxxxxx>"); >> +MODULE_LICENSE("GPL v2"); >> diff --git a/drivers/media/pci/tw686x/tw686x-regs.h b/drivers/media/pci/tw686x/tw686x-regs.h >> new file mode 100644 >> index 000000000000..92c331ceb065 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/tw686x-regs.h >> @@ -0,0 +1,118 @@ >> +/* DMA controller registers */ >> +#define REG8_1(a0) ((const u16[8]){a0, a0 + 1, a0 + 2, a0 + 3, \ >> + a0 + 4, a0 + 5, a0 + 6, a0 + 7}) >> +#define REG8_2(a0) ((const u16[8]){a0, a0 + 2, a0 + 4, a0 + 6, \ >> + a0 + 8, a0 + 0xA, a0 + 0xC, a0 + 0xE}) >> +#define REG8_8(a0) ((const u16[8]){a0, a0 + 8, a0 + 0x10, a0 + 0x18, \ >> + a0 + 0x20, a0 + 0x28, a0 + 0x30, a0 + 0x38}) >> +#define INT_STATUS 0x00 >> +#define PB_STATUS 0x01 >> +#define DMA_CMD 0x02 >> +#define VIDEO_FIFO_STATUS 0x03 >> +#define VIDEO_CHANNEL_ID 0x04 >> +#define VIDEO_PARSER_STATUS 0x05 >> +#define SYS_SOFT_RST 0x06 >> +#define DMA_PAGE_TABLE0_ADDR ((const u16[8]){0x08, 0xD0, 0xD2, 0xD4, \ >> + 0xD6, 0xD8, 0xDA, 0xDC}) >> +#define DMA_PAGE_TABLE1_ADDR ((const u16[8]){0x09, 0xD1, 0xD3, 0xD5, \ >> + 0xD7, 0xD9, 0xDB, 0xDD}) >> +#define DMA_CHANNEL_ENABLE 0x0a >> +#define DMA_CONFIG 0x0b >> +#define DMA_TIMER_INTERVAL 0x0c >> +#define DMA_CHANNEL_TIMEOUT 0x0d >> +#define VDMA_CHANNEL_CONFIG REG8_1(0x10) >> +#define ADMA_P_ADDR REG8_2(0x18) >> +#define ADMA_B_ADDR REG8_2(0x19) >> +#define DMA10_P_ADDR 0x28 /* ??? */ >> +#define DMA10_B_ADDR 0x29 >> +#define VIDEO_CONTROL1 0x2A >> +#define VIDEO_CONTROL2 0x2B >> +#define AUDIO_CONTROL1 0x2C >> +#define AUDIO_CONTROL2 0x2D >> +#define PHASE_REF 0x2E >> +#define GPIO_REG 0x2F >> +#define INTL_HBAR_CTRL REG8_1(0x30) >> +#define AUDIO_CONTROL3 0x38 >> +#define VIDEO_FIELD_CTRL REG8_1(0x39) >> +#define HSCALER_CTRL REG8_1(0x42) >> +#define VIDEO_SIZE REG8_1(0x4A) >> +#define VIDEO_SIZE_F2 REG8_1(0x52) >> +#define MD_CONF REG8_1(0x60) >> +#define MD_INIT REG8_1(0x68) >> +#define MD_MAP0 REG8_1(0x70) >> +#define VDMA_P_ADDR REG8_8(0x80) /* not used in DMA SG mode */ >> +#define VDMA_WHP REG8_8(0x81) >> +#define VDMA_B_ADDR REG8_8(0x82) >> +#define VDMA_F2_P_ADDR REG8_8(0x84) >> +#define VDMA_F2_WHP REG8_8(0x85) >> +#define VDMA_F2_B_ADDR REG8_8(0x86) >> +#define EP_REG_ADDR 0xFE >> +#define EP_REG_DATA 0xFF >> + >> +/* Video decoder registers */ >> +#define VDREG8(a0) ((const u16[8]){ \ >> + a0 + 0x000, a0 + 0x010, a0 + 0x020, a0 + 0x030, \ >> + a0 + 0x100, a0 + 0x110, a0 + 0x120, a0 + 0x130}) >> +#define VIDSTAT VDREG8(0x100) >> +#define BRIGHT VDREG8(0x101) >> +#define CONTRAST VDREG8(0x102) >> +#define SHARPNESS VDREG8(0x103) >> +#define SAT_U VDREG8(0x104) >> +#define SAT_V VDREG8(0x105) >> +#define HUE VDREG8(0x106) >> +#define CROP_HI VDREG8(0x107) >> +#define VDELAY_LO VDREG8(0x108) >> +#define VACTIVE_LO VDREG8(0x109) >> +#define HDELAY_LO VDREG8(0x10A) >> +#define HACTIVE_LO VDREG8(0x10B) >> +#define MVSN VDREG8(0x10C) >> +#define STATUS2 VDREG8(0x10C) >> +#define SDT VDREG8(0x10E) >> +#define SDT_EN VDREG8(0x10F) >> + >> +#define VSCALE_LO VDREG8(0x144) >> +#define SCALE_HI VDREG8(0x145) >> +#define HSCALE_LO VDREG8(0x146) >> +#define F2CROP_HI VDREG8(0x147) >> +#define F2VDELAY_LO VDREG8(0x148) >> +#define F2VACTIVE_LO VDREG8(0x149) >> +#define F2HDELAY_LO VDREG8(0x14A) >> +#define F2HACTIVE_LO VDREG8(0x14B) >> +#define F2VSCALE_LO VDREG8(0x14C) >> +#define F2SCALE_HI VDREG8(0x14D) >> +#define F2HSCALE_LO VDREG8(0x14E) >> +#define F2CNT VDREG8(0x14F) >> + >> +#define VDREG2(a0) ((const u16[2]){a0, a0 + 0x100}) >> +#define SRST VDREG2(0x180) >> +#define ACNTL VDREG2(0x181) >> +#define ACNTL2 VDREG2(0x182) >> +#define CNTRL1 VDREG2(0x183) >> +#define CKHY VDREG2(0x184) >> +#define SHCOR VDREG2(0x185) >> +#define CORING VDREG2(0x186) >> +#define CLMPG VDREG2(0x187) >> +#define IAGC VDREG2(0x188) >> +#define VCTRL1 VDREG2(0x18F) >> +#define MISC1 VDREG2(0x194) >> +#define LOOP VDREG2(0x195) >> +#define MISC2 VDREG2(0x196) >> + >> +#define CLMD VDREG2(0x197) >> +#define ANPWRDOWN VDREG2(0x1ce) >> +#define AIGAIN ((const u16[8]){0x1D0, 0x1D1, 0x1D2, 0x1D3, \ >> + 0x2D0, 0x2D1, 0x2D2, 0x2D3}) >> + >> +#define SYS_MODE_DMA_SHIFT 13 >> +#define TW686X_VIDSTAT_HLOCK BIT(6) >> +#define TW686X_VIDSTAT_VDLOSS BIT(7) >> + >> +#define TW686X_STD_NTSC_M 0 >> +#define TW686X_STD_PAL 1 >> +#define TW686X_STD_SECAM 2 >> +#define TW686X_STD_NTSC_443 3 >> +#define TW686X_STD_PAL_M 4 >> +#define TW686X_STD_PAL_CN 5 >> +#define TW686X_STD_PAL_60 6 >> + >> +#define TW686X_FIFO_ERROR(x) (x & ~(0xFF)) >> diff --git a/drivers/media/pci/tw686x/tw686x-video.c b/drivers/media/pci/tw686x/tw686x-video.c >> new file mode 100644 >> index 000000000000..7287a9de7a1a >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/tw686x-video.c >> @@ -0,0 +1,953 @@ >> +/* >> + * Copyright (C) 2015 VanguardiaSur - www.vanguardiasur.com.ar >> + * >> + * Based on original driver by Krzysztof Hałasa: >> + * Copyright (C) 2015 Industrial Research Institute for Automation >> + * and Measurements PIAP >> + * >> + * This program is free software; you can redistribute it and/or modify it >> + * under the terms of version 2 of the GNU General Public License >> + * as published by the Free Software Foundation. >> + */ >> + >> +#include <linux/init.h> >> +#include <linux/delay.h> >> +#include <linux/list.h> >> +#include <linux/module.h> >> +#include <linux/kernel.h> >> +#include <linux/slab.h> >> +#include <media/v4l2-common.h> >> +#include <media/v4l2-event.h> >> +#include <media/videobuf2-vmalloc.h> >> +#include "tw686x.h" >> +#include "tw686x-regs.h" >> + >> +#define TW686X_INPUTS_PER_CH 4 >> +#define TW686X_VIDEO_WIDTH 720 >> +#define TW686X_VIDEO_HEIGHT(id) ((id & V4L2_STD_625_50) ? 576 : 480) >> + >> +static const struct tw686x_format formats[] = { >> + { >> + .name = "4:2:2 packed, UYVY", >> + .fourcc = V4L2_PIX_FMT_UYVY, >> + .mode = 0, >> + .depth = 16, >> + }, { >> + .name = "16 bpp RGB", >> + .fourcc = V4L2_PIX_FMT_RGB565, >> + .mode = 5, >> + .depth = 16, >> + }, { >> + .name = "4:2:2 packed, YUYV", > > Drop name: these days this is filled in by the v4l2 core to ensure > consistent naming. > OK. >> + .fourcc = V4L2_PIX_FMT_YUYV, >> + .mode = 6, >> + .depth = 16, >> + } >> +}; >> + >> +static unsigned int tw686x_fields_map(v4l2_std_id std, unsigned int fps) >> +{ >> + unsigned int map[15] = { >> + 0x00000000, 0x00000001, 0x00004001, 0x00104001, 0x00404041, >> + 0x01041041, 0x01104411, 0x01111111, 0x04444445, 0x04511445, >> + 0x05145145, 0x05151515, 0x05515455, 0x05551555, 0x05555555 >> + }; >> + >> + unsigned int std_625_50[26] = { >> + 0, 1, 1, 2, 3, 3, 4, 4, 5, 5, 6, 7, 7, >> + 8, 8, 9, 10, 10, 11, 11, 12, 13, 13, 14, 14, 0 >> + }; >> + >> + unsigned int std_525_60[31] = { >> + 0, 1, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, >> + 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13, 14, 0, 0 >> + }; > > This can all be static const. > OK. >> + >> + unsigned int i = >> + (std & V4L2_STD_625_50) ? std_625_50[fps] : std_525_60[fps]; >> + >> + return map[i]; >> +} >> + >> +static void tw686x_set_framerate(struct tw686x_video_channel *vc, >> + unsigned int fps) >> +{ >> + unsigned int map; >> + >> + if (vc->fps == fps) >> + return; >> + >> + map = tw686x_fields_map(vc->video_standard, fps) << 1; >> + map |= map << 1; >> + if (map > 0) >> + map |= BIT(31); >> + reg_write(vc->dev, VIDEO_FIELD_CTRL[vc->ch], map); >> + vc->fps = fps; >> +} >> + >> +static const struct tw686x_format *format_by_fourcc(unsigned fourcc) >> +{ >> + unsigned cnt; >> + >> + for (cnt = 0; cnt < ARRAY_SIZE(formats); cnt++) >> + if (formats[cnt].fourcc == fourcc) >> + return &formats[cnt]; >> + return NULL; >> +} >> + >> +static int tw686x_queue_setup(struct vb2_queue *vq, >> + unsigned int *nbuffers, unsigned int *nplanes, >> + unsigned int sizes[], void *alloc_ctxs[]) >> +{ >> + struct tw686x_video_channel *vc = vb2_get_drv_priv(vq); >> + unsigned int szimage = >> + (vc->width * vc->height * vc->format->depth) >> 3; >> + >> + /* >> + * Let's request at least three buffers: two for the >> + * DMA engine and one for userspace. >> + */ >> + if (vq->num_buffers + *nbuffers < 3) >> + *nbuffers = 3 - vq->num_buffers; >> + >> + if (*nplanes) { >> + if (*nplanes != 1 || sizes[0] < szimage) >> + return -EINVAL; >> + return 0; >> + } >> + >> + sizes[0] = szimage; >> + *nplanes = 1; >> + return 0; >> +} >> + >> +static void tw686x_buf_queue(struct vb2_buffer *vb) >> +{ >> + struct tw686x_video_channel *vc = vb2_get_drv_priv(vb->vb2_queue); >> + struct tw686x_dev *dev = vc->dev; >> + struct pci_dev *pci_dev; >> + unsigned long flags; >> + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); >> + struct tw686x_v4l2_buf *buf = >> + container_of(vbuf, struct tw686x_v4l2_buf, vb); >> + >> + /* Check device presence */ >> + spin_lock_irqsave(&dev->lock, flags); >> + pci_dev = dev->pci_dev; >> + spin_unlock_irqrestore(&dev->lock, flags); >> + if (!pci_dev) { >> + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR); >> + return; >> + } >> + >> + spin_lock_irqsave(&vc->qlock, flags); >> + list_add_tail(&buf->list, &vc->vidq_queued); >> + spin_unlock_irqrestore(&vc->qlock, flags); >> +} >> + >> +/* >> + * We can call this even when alloc_dma failed for the given channel >> + */ >> +static void tw686x_free_dma(struct tw686x_video_channel *vc, unsigned int pb) >> +{ >> + struct tw686x_dma_desc *desc = &vc->dma_descs[pb]; >> + struct tw686x_dev *dev = vc->dev; >> + struct pci_dev *pci_dev; >> + unsigned long flags; >> + >> + /* Check device presence. Shouldn't really happen! */ >> + spin_lock_irqsave(&dev->lock, flags); >> + pci_dev = dev->pci_dev; >> + spin_unlock_irqrestore(&dev->lock, flags); >> + if (!pci_dev) { >> + WARN(1, "trying to deallocate on missing device\n"); >> + return; >> + } > > Dubious code, as mentioned earlier. > >> + >> + if (desc->virt) { >> + pci_free_consistent(dev->pci_dev, desc->size, >> + desc->virt, desc->phys); >> + desc->virt = NULL; >> + } >> +} >> + >> +static int tw686x_alloc_dma(struct tw686x_video_channel *vc, unsigned int pb) >> +{ >> + struct tw686x_dev *dev = vc->dev; >> + u32 reg = pb ? VDMA_B_ADDR[vc->ch] : VDMA_P_ADDR[vc->ch]; >> + unsigned int len; >> + void *virt; >> + >> + WARN(vc->dma_descs[pb].virt, >> + "Allocating buffer but previous still here\n"); >> + >> + len = (vc->width * vc->height * vc->format->depth) >> 3; >> + virt = pci_alloc_consistent(dev->pci_dev, len, >> + &vc->dma_descs[pb].phys); >> + if (!virt) { >> + v4l2_err(&dev->v4l2_dev, "dma%d: unable to allocate %s-buffer\n", >> + vc->ch, pb ? "B" : "P"); >> + return -ENOMEM; >> + } >> + vc->dma_descs[pb].size = len; >> + vc->dma_descs[pb].virt = virt; >> + reg_write(dev, reg, vc->dma_descs[pb].phys); >> + >> + return 0; >> +} >> + >> +static void tw686x_buffer_refill(struct tw686x_video_channel *vc, unsigned int pb) >> +{ >> + struct tw686x_v4l2_buf *buf; >> + >> + while (!list_empty(&vc->vidq_queued)) { >> + >> + buf = list_first_entry(&vc->vidq_queued, >> + struct tw686x_v4l2_buf, list); >> + list_del(&buf->list); >> + >> + vc->curr_bufs[pb] = buf; >> + return; >> + } >> + vc->curr_bufs[pb] = NULL; >> +} >> + >> +static void tw686x_clear_queue(struct tw686x_video_channel *vc) >> +{ >> + unsigned int pb; >> + >> + while (!list_empty(&vc->vidq_queued)) { >> + struct tw686x_v4l2_buf *buf; >> + >> + buf = list_first_entry(&vc->vidq_queued, >> + struct tw686x_v4l2_buf, list); >> + list_del(&buf->list); >> + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR); >> + } >> + >> + for (pb = 0; pb < 2; pb++) { >> + if (vc->curr_bufs[pb]) >> + vb2_buffer_done(&vc->curr_bufs[pb]->vb.vb2_buf, >> + VB2_BUF_STATE_ERROR); >> + vc->curr_bufs[pb] = NULL; >> + } >> +} >> + >> +static int tw686x_start_streaming(struct vb2_queue *vq, unsigned int count) >> +{ >> + struct tw686x_video_channel *vc = vb2_get_drv_priv(vq); >> + struct tw686x_dev *dev = vc->dev; >> + struct pci_dev *pci_dev; >> + unsigned long flags; >> + int pb; >> + >> + /* Check device presence */ >> + spin_lock_irqsave(&dev->lock, flags); >> + pci_dev = dev->pci_dev; >> + spin_unlock_irqrestore(&dev->lock, flags); >> + if (!pci_dev) >> + return -ENODEV; >> + >> + spin_lock_irqsave(&vc->qlock, flags); >> + >> + /* Sanity check */ >> + if (!vc->dma_descs[0].virt || !vc->dma_descs[1].virt) { >> + spin_unlock_irqrestore(&vc->qlock, flags); >> + v4l2_err(&dev->v4l2_dev, "video%d: refusing to start without DMA buffers\n", >> + vc->num); >> + return -ENOMEM; > > If start_streaming fails with an error, then you need to call vb2_buffer_done > with status VB2_BUF_STATE_QUEUED for all buffers. > Ah, OK. Will fix it. >> + } >> + >> + for (pb = 0; pb < 2; pb++) >> + tw686x_buffer_refill(vc, pb); >> + spin_unlock_irqrestore(&vc->qlock, flags); >> + >> + vc->sequence = 0; >> + vc->pb = 0; >> + >> + spin_lock_irqsave(&dev->lock, flags); >> + tw686x_enable_channel(dev, vc->ch); >> + spin_unlock_irqrestore(&dev->lock, flags); >> + >> + mod_timer(&dev->dma_delay_timer, jiffies + msecs_to_jiffies(100)); >> + >> + return 0; >> +} >> + >> +static void tw686x_stop_streaming(struct vb2_queue *vq) >> +{ >> + struct tw686x_video_channel *vc = vb2_get_drv_priv(vq); >> + struct tw686x_dev *dev = vc->dev; >> + struct pci_dev *pci_dev; >> + unsigned long flags; >> + >> + /* Check device presence */ >> + spin_lock_irqsave(&dev->lock, flags); >> + pci_dev = dev->pci_dev; >> + spin_unlock_irqrestore(&dev->lock, flags); >> + if (pci_dev) >> + tw686x_disable_channel(dev, vc->ch); >> + >> + spin_lock_irqsave(&vc->qlock, flags); >> + tw686x_clear_queue(vc); >> + spin_unlock_irqrestore(&vc->qlock, flags); >> +} >> + >> +static int tw686x_buf_prepare(struct vb2_buffer *vb) >> +{ >> + struct tw686x_video_channel *vc = vb2_get_drv_priv(vb->vb2_queue); >> + unsigned int size = >> + (vc->width * vc->height * vc->format->depth) >> 3; >> + >> + if (vb2_plane_size(vb, 0) < size) >> + return -EINVAL; >> + vb2_set_plane_payload(vb, 0, size); >> + return 0; >> +} >> + >> +static struct vb2_ops tw686x_video_qops = { >> + .queue_setup = tw686x_queue_setup, >> + .buf_queue = tw686x_buf_queue, >> + .buf_prepare = tw686x_buf_prepare, >> + .start_streaming = tw686x_start_streaming, >> + .stop_streaming = tw686x_stop_streaming, >> + .wait_prepare = vb2_ops_wait_prepare, >> + .wait_finish = vb2_ops_wait_finish, >> +}; >> + >> +static int tw686x_s_ctrl(struct v4l2_ctrl *ctrl) >> +{ >> + struct tw686x_video_channel *vc; >> + struct tw686x_dev *dev; >> + unsigned ch; >> + >> + vc = container_of(ctrl->handler, struct tw686x_video_channel, >> + ctrl_handler); >> + dev = vc->dev; >> + ch = vc->ch; >> + >> + switch (ctrl->id) { >> + case V4L2_CID_BRIGHTNESS: >> + reg_write(dev, BRIGHT[ch], ctrl->val & 0xFF); >> + return 0; >> + >> + case V4L2_CID_CONTRAST: >> + reg_write(dev, CONTRAST[ch], ctrl->val); >> + return 0; >> + >> + case V4L2_CID_SATURATION: >> + reg_write(dev, SAT_U[ch], ctrl->val); >> + reg_write(dev, SAT_V[ch], ctrl->val); >> + return 0; >> + >> + case V4L2_CID_HUE: >> + reg_write(dev, HUE[ch], ctrl->val & 0xFF); >> + return 0; >> + } >> + >> + return -EINVAL; >> +} >> + >> +static const struct v4l2_ctrl_ops ctrl_ops = { >> + .s_ctrl = tw686x_s_ctrl, >> +}; >> + >> +static int tw686x_g_fmt_vid_cap(struct file *file, void *priv, >> + struct v4l2_format *f) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + >> + f->fmt.pix.width = vc->width; >> + f->fmt.pix.height = vc->height; >> + f->fmt.pix.field = V4L2_FIELD_INTERLACED; >> + f->fmt.pix.pixelformat = vc->format->fourcc; >> + f->fmt.pix.colorspace = V4L2_COLORSPACE_SMPTE170M; >> + f->fmt.pix.bytesperline = (f->fmt.pix.width * vc->format->depth) / 8; >> + f->fmt.pix.sizeimage = f->fmt.pix.height * f->fmt.pix.bytesperline; >> + return 0; >> +} >> + >> +static int tw686x_try_fmt_vid_cap(struct file *file, void *priv, >> + struct v4l2_format *f) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + unsigned int video_height = TW686X_VIDEO_HEIGHT(vc->video_standard); >> + const struct tw686x_format *format; >> + >> + format = format_by_fourcc(f->fmt.pix.pixelformat); >> + if (!format) { >> + format = &formats[0]; >> + f->fmt.pix.pixelformat = format->fourcc; >> + } >> + >> + if (f->fmt.pix.width <= TW686X_VIDEO_WIDTH / 2) >> + f->fmt.pix.width = TW686X_VIDEO_WIDTH / 2; >> + else >> + f->fmt.pix.width = TW686X_VIDEO_WIDTH; >> + >> + if (f->fmt.pix.height <= video_height / 2) >> + f->fmt.pix.height = video_height / 2; >> + else >> + f->fmt.pix.height = video_height; >> + >> + f->fmt.pix.bytesperline = (f->fmt.pix.width * format->depth) / 8; >> + f->fmt.pix.sizeimage = f->fmt.pix.height * f->fmt.pix.bytesperline; >> + f->fmt.pix.colorspace = V4L2_COLORSPACE_SMPTE170M; >> + f->fmt.pix.field = V4L2_FIELD_INTERLACED; >> + >> + return 0; >> +} >> + >> +static int tw686x_s_fmt_vid_cap(struct file *file, void *priv, >> + struct v4l2_format *f) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + u32 val, width, line_width, height; >> + unsigned long bitsperframe; >> + int err, pb; >> + >> + if (vb2_is_busy(&vc->vidq)) >> + return -EBUSY; >> + >> + bitsperframe = vc->width * vc->height * vc->format->depth; >> + err = tw686x_try_fmt_vid_cap(file, priv, f); >> + if (err) >> + return err; >> + >> + vc->format = format_by_fourcc(f->fmt.pix.pixelformat); >> + vc->width = f->fmt.pix.width; >> + vc->height = f->fmt.pix.height; >> + >> + /* We need new DMA buffers if the framesize has changed */ >> + if (bitsperframe != vc->width * vc->height * vc->format->depth) { >> + > > Remove empty line. > OK. >> + for (pb = 0; pb < 2; pb++) >> + tw686x_free_dma(vc, pb); >> + >> + for (pb = 0; pb < 2; pb++) { >> + err = tw686x_alloc_dma(vc, pb); >> + if (err) { >> + if (pb > 0) >> + tw686x_free_dma(vc, 0); >> + return err; >> + } >> + } >> + } >> + >> + val = reg_read(vc->dev, VDMA_CHANNEL_CONFIG[vc->ch]); >> + >> + if (vc->width <= TW686X_VIDEO_WIDTH / 2) >> + val |= BIT(23); >> + else >> + val &= ~BIT(23); >> + >> + if (vc->height <= TW686X_VIDEO_HEIGHT(vc->video_standard) / 2) >> + val |= BIT(24); >> + else >> + val &= ~BIT(24); >> + >> + val &= ~(0x7 << 20); >> + val |= vc->format->mode << 20; >> + reg_write(vc->dev, VDMA_CHANNEL_CONFIG[vc->ch], val); >> + >> + /* Program the DMA frame size */ >> + width = (vc->width * 2) & 0x7ff; >> + height = vc->height / 2; >> + line_width = (vc->width * 2) & 0x7ff; >> + val = (height << 22) | (line_width << 11) | width; >> + reg_write(vc->dev, VDMA_WHP[vc->ch], val); >> + return 0; >> +} >> + >> +static int tw686x_querycap(struct file *file, void *priv, >> + struct v4l2_capability *cap) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + struct tw686x_dev *dev = vc->dev; >> + >> + strcpy(cap->driver, "tw686x"); >> + strcpy(cap->card, dev->name); >> + sprintf(cap->bus_info, "PCI:%s", pci_name(dev->pci_dev)); > > strlcpy and snprintf > Will fix. >> + cap->device_caps = V4L2_CAP_VIDEO_CAPTURE | V4L2_CAP_STREAMING | >> + V4L2_CAP_READWRITE; >> + cap->capabilities = cap->device_caps | V4L2_CAP_DEVICE_CAPS; >> + return 0; >> +} >> + >> +static int tw686x_s_std(struct file *file, void *priv, v4l2_std_id id) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + struct v4l2_format f; >> + u32 val, ret; >> + >> + if (vc->video_standard == id) >> + return 0; >> + >> + if (vb2_is_busy(&vc->vidq)) >> + return -EBUSY; >> + >> + if (id & V4L2_STD_NTSC) >> + val = 0; >> + else if (id & V4L2_STD_PAL) >> + val = 1; >> + else if (id & V4L2_STD_SECAM) >> + val = 2; >> + else if (id & V4L2_STD_NTSC_443) >> + val = 3; >> + else if (id & V4L2_STD_PAL_M) >> + val = 4; >> + else if (id & V4L2_STD_PAL_Nc) >> + val = 5; >> + else if (id & V4L2_STD_PAL_60) >> + val = 6; >> + else >> + return -EINVAL; >> + >> + vc->video_standard = id; >> + reg_write(vc->dev, SDT[vc->ch], val); >> + >> + val = reg_read(vc->dev, VIDEO_CONTROL1); >> + if (id & V4L2_STD_625_50) >> + val |= (1 << (SYS_MODE_DMA_SHIFT + vc->ch)); >> + else >> + val &= ~(1 << (SYS_MODE_DMA_SHIFT + vc->ch)); >> + reg_write(vc->dev, VIDEO_CONTROL1, val); >> + >> + /* >> + * Adjust format after V4L2_STD_525_60/V4L2_STD_625_50 change, >> + * calling g_fmt and s_fmt will sanitize the height >> + * according to the standard. >> + */ >> + ret = tw686x_g_fmt_vid_cap(file, priv, &f); >> + if (!ret) >> + tw686x_s_fmt_vid_cap(file, priv, &f); >> + return 0; >> +} >> + >> +static int tw686x_querystd(struct file *file, void *priv, v4l2_std_id *std) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + struct tw686x_dev *dev = vc->dev; >> + unsigned int old_std, detected_std = 0; >> + unsigned long end; >> + >> + if (vb2_is_streaming(&vc->vidq)) >> + return -EBUSY; >> + >> + /* Enable and start standard detection */ >> + old_std = reg_read(dev, SDT[vc->ch]); >> + reg_write(dev, SDT[vc->ch], 0x7); >> + reg_write(dev, SDT_EN[vc->ch], 0xff); >> + >> + end = jiffies + msecs_to_jiffies(500); >> + while (time_is_after_jiffies(end)) { >> + >> + detected_std = reg_read(dev, SDT[vc->ch]); >> + if (!(detected_std & BIT(7))) >> + break; >> + msleep(100); >> + } >> + reg_write(dev, SDT[vc->ch], old_std); >> + >> + /* Exit if still busy */ >> + if (detected_std & BIT(7)) >> + return 0; >> + >> + detected_std = (detected_std >> 4) & 0x7; >> + switch (detected_std) { >> + case TW686X_STD_NTSC_M: >> + *std &= V4L2_STD_NTSC; >> + break; >> + case TW686X_STD_NTSC_443: >> + *std &= V4L2_STD_NTSC_443; >> + break; >> + case TW686X_STD_PAL_M: >> + *std &= V4L2_STD_PAL_M; >> + break; >> + case TW686X_STD_PAL_60: >> + *std &= V4L2_STD_PAL_60; >> + break; >> + case TW686X_STD_PAL: >> + *std &= V4L2_STD_PAL; >> + break; >> + case TW686X_STD_PAL_CN: >> + *std &= V4L2_STD_PAL_Nc; >> + break; >> + case TW686X_STD_SECAM: >> + *std &= V4L2_STD_SECAM; >> + break; >> + default: >> + *std = 0; >> + } >> + return 0; >> +} >> + >> +static int tw686x_g_std(struct file *file, void *priv, v4l2_std_id *id) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + >> + *id = vc->video_standard; >> + return 0; >> +} >> + >> +static int tw686x_enum_fmt_vid_cap(struct file *file, void *priv, >> + struct v4l2_fmtdesc *f) >> +{ >> + if (f->index >= ARRAY_SIZE(formats)) >> + return -EINVAL; >> + >> + strlcpy(f->description, formats[f->index].name, sizeof(f->description)); > > Drop this line, this is now done by the v4l2 core. > OK. >> + f->pixelformat = formats[f->index].fourcc; >> + return 0; >> +} >> + >> +static int tw686x_g_parm(struct file *file, void *priv, >> + struct v4l2_streamparm *sp) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + struct v4l2_captureparm *cp = &sp->parm.capture; >> + >> + if (sp->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) >> + return -EINVAL; >> + sp->parm.capture.readbuffers = 3; >> + >> + cp->capability = V4L2_CAP_TIMEPERFRAME; >> + cp->timeperframe.numerator = 1; >> + cp->timeperframe.denominator = vc->fps; >> + return 0; >> +} >> + >> +static int tw686x_s_parm(struct file *file, void *priv, >> + struct v4l2_streamparm *sp) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + struct v4l2_captureparm *cp = &sp->parm.capture; >> + unsigned int denominator = cp->timeperframe.denominator; >> + unsigned int numerator = cp->timeperframe.numerator; >> + unsigned int fps; >> + >> + if (vb2_is_busy(&vc->vidq)) >> + return -EBUSY; >> + >> + fps = (!numerator || !denominator) ? 0 : denominator / numerator; >> + if (vc->video_standard & V4L2_STD_625_50) >> + fps = (!fps || fps > 25) ? 25 : fps; >> + else >> + fps = (!fps || fps > 30) ? 30 : fps; >> + tw686x_set_framerate(vc, fps); >> + >> + return tw686x_g_parm(file, priv, sp); >> +} > > Why do you need to implement g/s_parm? This is not typically done for SDTV > capture devices since you cannot set it. And the v4l2 core will provide a > default g_parm implementation that does the right thing. I think you can > just delete these two functions. > Hm, I think it's a left over from the code this driver is based. I haven't really used this ioctl. >> + >> +static int tw686x_s_input(struct file *file, void *priv, unsigned int i) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + u32 val; >> + >> + if (i >= TW686X_INPUTS_PER_CH) >> + return -EINVAL; >> + > > I would add this here: > > if (i == vc->input) > return 0; > Ah, right. >> + /* >> + * Not sure we are able to support on the fly input change >> + */ >> + if (vb2_is_busy(&vc->vidq)) >> + return -EBUSY; >> + >> + vc->input = i; >> + >> + val = reg_read(vc->dev, VDMA_CHANNEL_CONFIG[vc->ch]); >> + val &= ~(0x3 << 30); >> + val |= i << 30; >> + reg_write(vc->dev, VDMA_CHANNEL_CONFIG[vc->ch], val); >> + return 0; >> +} >> + >> +static int tw686x_g_input(struct file *file, void *priv, unsigned int *i) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + >> + *i = vc->input; >> + return 0; >> +} >> + >> +static int tw686x_enum_input(struct file *file, void *priv, >> + struct v4l2_input *i) >> +{ >> + struct tw686x_video_channel *vc = video_drvdata(file); >> + unsigned int vidstat; >> + >> + if (i->index >= TW686X_INPUTS_PER_CH) >> + return -EINVAL; >> + >> + sprintf(i->name, "Composite%d", i->index); >> + i->type = V4L2_INPUT_TYPE_CAMERA; >> + i->std = vc->device->tvnorms; >> + i->capabilities = V4L2_IN_CAP_STD; >> + >> + vidstat = reg_read(vc->dev, VIDSTAT[vc->ch]); >> + i->status = 0; >> + if (vidstat & TW686X_VIDSTAT_VDLOSS) >> + i->status |= V4L2_IN_ST_NO_SIGNAL; >> + if (!(vidstat & TW686X_VIDSTAT_HLOCK)) >> + i->status |= V4L2_IN_ST_NO_H_LOCK; >> + >> + return 0; >> +} >> + >> +const struct v4l2_file_operations tw686x_video_fops = { >> + .owner = THIS_MODULE, >> + .open = v4l2_fh_open, >> + .unlocked_ioctl = video_ioctl2, >> + .release = vb2_fop_release, >> + .poll = vb2_fop_poll, >> + .read = vb2_fop_read, >> + .mmap = vb2_fop_mmap, >> +}; >> + >> +const struct v4l2_ioctl_ops tw686x_video_ioctl_ops = { >> + .vidioc_querycap = tw686x_querycap, >> + .vidioc_g_fmt_vid_cap = tw686x_g_fmt_vid_cap, >> + .vidioc_s_fmt_vid_cap = tw686x_s_fmt_vid_cap, >> + .vidioc_enum_fmt_vid_cap = tw686x_enum_fmt_vid_cap, >> + .vidioc_try_fmt_vid_cap = tw686x_try_fmt_vid_cap, >> + >> + .vidioc_querystd = tw686x_querystd, >> + .vidioc_g_std = tw686x_g_std, >> + .vidioc_s_std = tw686x_s_std, >> + >> + .vidioc_g_parm = tw686x_g_parm, >> + .vidioc_s_parm = tw686x_s_parm, >> + >> + .vidioc_enum_input = tw686x_enum_input, >> + .vidioc_g_input = tw686x_g_input, >> + .vidioc_s_input = tw686x_s_input, >> + >> + .vidioc_reqbufs = vb2_ioctl_reqbufs, >> + .vidioc_querybuf = vb2_ioctl_querybuf, >> + .vidioc_qbuf = vb2_ioctl_qbuf, >> + .vidioc_dqbuf = vb2_ioctl_dqbuf, >> + .vidioc_create_bufs = vb2_ioctl_create_bufs, >> + .vidioc_streamon = vb2_ioctl_streamon, >> + .vidioc_streamoff = vb2_ioctl_streamoff, >> + >> + .vidioc_log_status = v4l2_ctrl_log_status, >> + .vidioc_subscribe_event = v4l2_ctrl_subscribe_event, >> + .vidioc_unsubscribe_event = v4l2_event_unsubscribe, >> +}; >> + >> +static void tw686x_buffer_copy(struct tw686x_video_channel *vc, >> + unsigned int pb, struct vb2_v4l2_buffer *vb) >> +{ >> + struct tw686x_dma_desc *desc = &vc->dma_descs[pb]; >> + struct vb2_buffer *vb2_buf = &vb->vb2_buf; >> + >> + vb->field = V4L2_FIELD_INTERLACED; >> + vb->sequence = vc->sequence++; >> + >> + memcpy(vb2_plane_vaddr(vb2_buf, 0), desc->virt, desc->size); >> + vb2_buf->timestamp = ktime_get_ns(); >> + vb2_buffer_done(vb2_buf, VB2_BUF_STATE_DONE); >> +} >> + >> +void tw686x_video_irq(struct tw686x_dev *dev, unsigned long requests, >> + unsigned int pb_status, unsigned int fifo_status, >> + unsigned int *reset_ch) >> +{ >> + struct tw686x_video_channel *vc; >> + struct vb2_v4l2_buffer *vb; >> + unsigned long flags; >> + unsigned int ch, pb; >> + >> + for_each_set_bit(ch, &requests, max_channels(dev)) { >> + > > Remove empty line. > OK. >> + vc = &dev->video_channels[ch]; >> + >> + /* >> + * This can either be a blue frame (with signal-lost bit set) >> + * or a good frame (with signal-lost bit clear). If we have just >> + * got signal, then this channel needs resetting. >> + */ >> + if (vc->no_signal && !(fifo_status & BIT(ch))) { >> + v4l2_printk(KERN_DEBUG, &dev->v4l2_dev, >> + "video%d: signal recovered\n", vc->num); >> + vc->no_signal = false; >> + *reset_ch |= BIT(ch); >> + vc->pb = 0; >> + continue; >> + } >> + vc->no_signal = !!(fifo_status & BIT(ch)); >> + >> + /* Check FIFO errors only if there's signal */ >> + if (!vc->no_signal) { >> + u32 fifo_ov, fifo_bad; >> + >> + fifo_ov = (fifo_status >> 24) & BIT(ch); >> + fifo_bad = (fifo_status >> 16) & BIT(ch); >> + if (fifo_ov || fifo_bad) { >> + /* Mark this channel for reset */ >> + v4l2_printk(KERN_DEBUG, &dev->v4l2_dev, "video%d: FIFO error\n", vc->num); >> + *reset_ch |= BIT(ch); >> + vc->pb = 0; >> + continue; >> + } >> + } >> + >> + pb = !!(pb_status & BIT(ch)); >> + if (vc->pb != pb) { >> + /* Mark this channel for reset */ >> + v4l2_printk(KERN_DEBUG, &dev->v4l2_dev, >> + "video%d: unexpected p-b buffer!\n", vc->num); >> + *reset_ch |= BIT(ch); >> + vc->pb = 0; >> + continue; >> + } >> + >> + /* handle video stream */ >> + spin_lock_irqsave(&vc->qlock, flags); >> + if (vc->curr_bufs[pb]) { >> + vb = &vc->curr_bufs[pb]->vb; >> + tw686x_buffer_copy(vc, pb, vb); > > You have to copy the data? It's not possible the program the DMA so that > it DMAs into the buffer itself? That's quite unusual for a PCI device. > Yes, it's possible and I spent an enormous amount of time trying to make it work (originally using scatter-gather mode, and then with frame mode). However, despite my many efforts it always stucked (sooner or later in the tests) into a hard machine freeze. There are two apparent sources for the freeze: (1) To make the above work you need to program the registers so the chip DMAs into a new buffer each time the current DMA buffer is completed. (2) Also, when a signal error is detected and/or signal is lost and recovered, the DMA channels are re-programmed as well. It was only when all the registers write got removed and minimized to the bare minimum (registers are written before streaming starts and then stay mostly untouched) that I got a stable driver working fine for several weeks. The ugly delay timer is meant to mitigate (2). And the buffer copy is to workaround (1). Chip and board vendors couldn't provide any explanation for this behavior. I have two different boards (one with 1-chip, one with 2-chips and a PCIe switch), and the issues are present on both. In any case, the vendor's Windows driver does the similar buffer-copy. I understand that on some platforms this implementation could be too costly (it's completely cheap on any modern x86), and I intend to provide some option to provide "frame DMA-to-buffer" and "scatter-gather DMA". However, I wanted to get this basic version merged first. (Sorry, I should have included all this in the cover letter since it was pretty obvious you would ask :) >> + } >> + vc->pb = !pb; >> + tw686x_buffer_refill(vc, pb); >> + spin_unlock_irqrestore(&vc->qlock, flags); >> + } >> +} >> + >> +void tw686x_video_free(struct tw686x_dev *dev) >> +{ >> + unsigned int ch, pb; >> + >> + for (ch = 0; ch < max_channels(dev); ch++) { >> + struct tw686x_video_channel *vc = &dev->video_channels[ch]; >> + >> + if (vc->device) >> + video_unregister_device(vc->device); >> + >> + for (pb = 0; pb < 2; pb++) >> + tw686x_free_dma(vc, pb); >> + } >> +} >> + >> +int tw686x_video_init(struct tw686x_dev *dev) >> +{ >> + unsigned int ch, val, pb; >> + int err; >> + >> + err = v4l2_device_register(&dev->pci_dev->dev, &dev->v4l2_dev); >> + if (err) >> + return err; >> + >> + for (ch = 0; ch < max_channels(dev); ch++) { >> + struct tw686x_video_channel *vc = &dev->video_channels[ch]; >> + struct video_device *vdev; >> + >> + mutex_init(&vc->vb_mutex); >> + spin_lock_init(&vc->qlock); >> + INIT_LIST_HEAD(&vc->vidq_queued); >> + >> + vc->dev = dev; >> + vc->ch = ch; >> + >> + /* default settings */ >> + vc->format = &formats[0]; >> + vc->video_standard = V4L2_STD_NTSC; >> + vc->width = TW686X_VIDEO_WIDTH; >> + vc->height = TW686X_VIDEO_HEIGHT(vc->video_standard); >> + vc->input = 0; >> + >> + reg_write(vc->dev, SDT[ch], 0); >> + tw686x_set_framerate(vc, 30); >> + >> + reg_write(dev, VDELAY_LO[ch], 0x14); >> + reg_write(dev, HACTIVE_LO[ch], 0xD0); >> + reg_write(dev, VIDEO_SIZE[ch], 0); >> + >> + for (pb = 0; pb < 2; pb++) { >> + err = tw686x_alloc_dma(vc, pb); >> + if (err) >> + goto error; >> + } >> + >> + vdev = video_device_alloc(); >> + if (!vdev) { >> + v4l2_err(&dev->v4l2_dev, "dma%d: unable to allocate device\n", ch); >> + err = -ENOMEM; >> + goto error; >> + } >> + >> + vc->vidq.io_modes = VB2_READ | VB2_MMAP | VB2_DMABUF; >> + vc->vidq.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; >> + vc->vidq.drv_priv = vc; >> + vc->vidq.buf_struct_size = sizeof(struct tw686x_v4l2_buf); >> + vc->vidq.ops = &tw686x_video_qops; >> + vc->vidq.mem_ops = &vb2_vmalloc_memops; >> + vc->vidq.timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_MONOTONIC; >> + vc->vidq.min_buffers_needed = 2; >> + vc->vidq.lock = &vc->vb_mutex; >> + >> + err = vb2_queue_init(&vc->vidq); >> + if (err) >> + goto error; >> + >> + snprintf(vdev->name, sizeof(vdev->name), "%s video", dev->name); >> + vdev->fops = &tw686x_video_fops; >> + vdev->ioctl_ops = &tw686x_video_ioctl_ops; >> + vdev->release = video_device_release; >> + vdev->v4l2_dev = &dev->v4l2_dev; >> + vdev->queue = &vc->vidq; >> + vdev->tvnorms = V4L2_STD_525_60 | V4L2_STD_625_50; >> + vdev->minor = -1; >> + vdev->lock = &vc->vb_mutex; >> + vc->device = vdev; >> + video_set_drvdata(vdev, vc); >> + err = video_register_device(vdev, VFL_TYPE_GRABBER, -1); >> + if (err < 0) >> + goto error; > > Move this down to the end of the loop. Userspace can use the video device > as soon as it is created, so it is wise to create it only after everything > else has been configured. > OK. >> + vc->num = vdev->num; >> + >> + err = v4l2_ctrl_handler_init(&vc->ctrl_handler, 4); >> + if (err) { >> + v4l2_err(&dev->v4l2_dev, >> + "video%d: cannot init ctrl handler\n", >> + vc->num); >> + goto error; >> + } >> + vdev->ctrl_handler = &vc->ctrl_handler; >> + v4l2_ctrl_new_std(&vc->ctrl_handler, &ctrl_ops, >> + V4L2_CID_BRIGHTNESS, -128, 127, 1, 0); >> + v4l2_ctrl_new_std(&vc->ctrl_handler, &ctrl_ops, >> + V4L2_CID_CONTRAST, 0, 255, 1, 100); >> + v4l2_ctrl_new_std(&vc->ctrl_handler, &ctrl_ops, >> + V4L2_CID_SATURATION, 0, 255, 1, 128); >> + v4l2_ctrl_new_std(&vc->ctrl_handler, &ctrl_ops, >> + V4L2_CID_HUE, -128, 127, 1, 0); >> + err = vc->ctrl_handler.error; >> + if (err) >> + goto error; >> + >> + err = v4l2_ctrl_handler_setup(&vc->ctrl_handler); >> + if (err) >> + goto error; >> + } >> + >> + /* Set DMA frame mode on all channels. Only supported mode for now. */ >> + val = TW686X_DEF_PHASE_REF; >> + for (ch = 0; ch < max_channels(dev); ch++) >> + val |= TW686X_FRAME_MODE << (16 + ch * 2); >> + reg_write(dev, PHASE_REF, val); >> + >> + reg_write(dev, MISC2[0], 0xE7); >> + reg_write(dev, VCTRL1[0], 0xCC); >> + reg_write(dev, LOOP[0], 0xA5); >> + if (max_channels(dev) > 4) { >> + reg_write(dev, VCTRL1[1], 0xCC); >> + reg_write(dev, LOOP[1], 0xA5); >> + reg_write(dev, MISC2[1], 0xE7); >> + } >> + return 0; >> + >> +error: >> + tw686x_video_free(dev); >> + return err; >> +} >> diff --git a/drivers/media/pci/tw686x/tw686x.h b/drivers/media/pci/tw686x/tw686x.h >> new file mode 100644 >> index 000000000000..0ffb0be84aa2 >> --- /dev/null >> +++ b/drivers/media/pci/tw686x/tw686x.h >> @@ -0,0 +1,189 @@ >> +/* >> + * Copyright (C) 2015 VanguardiaSur - www.vanguardiasur.com.ar >> + * >> + * Copyright (C) 2015 Industrial Research Institute for Automation >> + * and Measurements PIAP >> + * Written by Krzysztof Hałasa >> + * >> + * This program is free software; you can redistribute it and/or modify it >> + * under the terms of version 2 of the GNU General Public License >> + * as published by the Free Software Foundation. >> + */ >> + >> +#include <linux/mutex.h> >> +#include <linux/pci.h> >> +#include <linux/timer.h> >> +#include <linux/videodev2.h> >> +#include <media/v4l2-common.h> >> +#include <media/v4l2-ctrls.h> >> +#include <media/v4l2-device.h> >> +#include <media/v4l2-ioctl.h> >> +#include <media/videobuf2-v4l2.h> >> +#include <sound/pcm.h> >> + >> +#include "tw686x-regs.h" >> + >> +#define TYPE_MAX_CHANNELS 0x0F >> +#define TYPE_SECOND_GEN 0x10 >> +#define TW686X_DEF_PHASE_REF 0x1518 >> + >> +#define TW686X_FIELD_MODE 0x3 >> +#define TW686X_FRAME_MODE 0x2 >> +/* 0x1 is reserved */ >> +#define TW686X_SG_MODE 0x0 >> + >> +#define TW686X_PAGE_SIZE 4096 >> +#define TW686X_AUDIO_PAGE_MAX 16 >> + >> +struct tw686x_format { >> + char *name; >> + unsigned fourcc; >> + unsigned depth; >> + unsigned mode; >> +}; >> + >> +struct tw686x_dma_desc { >> + dma_addr_t phys; >> + void *virt; >> + unsigned size; >> +}; >> + >> +struct tw686x_audio_buf { >> + dma_addr_t dma; >> + void *virt; >> + struct list_head list; >> +}; >> + >> +struct tw686x_v4l2_buf { >> + struct vb2_v4l2_buffer vb; >> + struct list_head list; >> +}; >> + >> +struct tw686x_audio_channel { >> + struct tw686x_dev *dev; >> + struct snd_pcm_substream *ss; >> + unsigned int ch; >> + struct tw686x_audio_buf *curr_bufs[2]; >> + struct tw686x_dma_desc dma_descs[2]; >> + dma_addr_t ptr; >> + >> + struct tw686x_audio_buf buf[TW686X_AUDIO_PAGE_MAX]; >> + struct list_head buf_list; >> + spinlock_t lock; >> +}; >> + >> +struct tw686x_video_channel { >> + struct tw686x_dev *dev; >> + >> + struct vb2_queue vidq; >> + struct list_head vidq_queued; >> + struct video_device *device; >> + struct tw686x_v4l2_buf *curr_bufs[2]; >> + struct tw686x_dma_desc dma_descs[2]; >> + >> + struct v4l2_ctrl_handler ctrl_handler; >> + const struct tw686x_format *format; >> + struct mutex vb_mutex; >> + spinlock_t qlock; >> + v4l2_std_id video_standard; >> + unsigned int width, height; >> + unsigned int h_halve, v_halve; >> + unsigned int ch; >> + unsigned int num; >> + unsigned int fps; >> + unsigned int input; >> + unsigned int sequence; >> + unsigned int pb; >> + bool no_signal; >> +}; >> + >> +/** >> + * struct tw686x_dev - global device status >> + * @lock: spinlock controlling access to the >> + * shared device registers (DMA enable/disable). >> + */ >> +struct tw686x_dev { >> + spinlock_t lock; >> + >> + struct v4l2_device v4l2_dev; >> + struct snd_card *snd_card; >> + >> + char name[32]; >> + unsigned int type; >> + struct pci_dev *pci_dev; >> + __u32 __iomem *mmio; >> + >> + void *alloc_ctx; >> + >> + struct tw686x_video_channel *video_channels; >> + struct tw686x_audio_channel *audio_channels; >> + >> + int audio_rate; /* per-device value */ >> + >> + struct timer_list dma_delay_timer; >> + u32 pending_dma_en; /* must be protected by lock */ >> + u32 pending_dma_cmd; /* must be protected by lock */ >> +}; >> + >> +static inline uint32_t reg_read(struct tw686x_dev *dev, unsigned reg) >> +{ >> + return readl(dev->mmio + reg); >> +} >> + >> +static inline void reg_write(struct tw686x_dev *dev, unsigned reg, >> + uint32_t value) >> +{ >> + writel(value, dev->mmio + reg); >> +} >> + >> +static inline void tw686x_disable_channel(struct tw686x_dev *dev, >> + unsigned int channel) >> +{ >> + u32 dma_en = reg_read(dev, DMA_CHANNEL_ENABLE); >> + u32 dma_cmd = reg_read(dev, DMA_CMD); >> + >> + dma_en &= ~BIT(channel); >> + dma_cmd &= ~BIT(channel); >> + >> + /* Must remove it from pending too */ >> + dev->pending_dma_en &= ~BIT(channel); >> + dev->pending_dma_cmd &= ~BIT(channel); >> + >> + /* Stop DMA if no channels are enabled */ >> + if (!dma_en) >> + dma_cmd = 0; >> + reg_write(dev, DMA_CHANNEL_ENABLE, dma_en); >> + reg_write(dev, DMA_CMD, dma_cmd); >> +} >> + >> +static inline void tw686x_enable_channel(struct tw686x_dev *dev, >> + unsigned int channel) >> +{ >> + u32 dma_en = reg_read(dev, DMA_CHANNEL_ENABLE); >> + u32 dma_cmd = reg_read(dev, DMA_CMD); >> + >> + dev->pending_dma_en |= dma_en | BIT(channel); >> + dev->pending_dma_cmd |= dma_cmd | BIT(31) | BIT(channel); >> +} > > I think these should be normal functions. I see no reason for en/disabling > channel to be inline. > OK. >> + >> +static inline unsigned max_channels(struct tw686x_dev *dev) >> +{ >> + return dev->type & TYPE_MAX_CHANNELS; /* 4 or 8 channels */ >> +} >> + >> +static inline unsigned is_second_gen(struct tw686x_dev *dev) >> +{ >> + /* each channel has its own DMA SG table */ >> + return dev->type & TYPE_SECOND_GEN; >> +} >> + >> +int tw686x_video_init(struct tw686x_dev *dev); >> +void tw686x_video_free(struct tw686x_dev *dev); >> +void tw686x_video_irq(struct tw686x_dev *dev, unsigned long requests, >> + unsigned int pb_status, unsigned int fifo_status, >> + unsigned int *reset_ch); >> + >> +int tw686x_audio_init(struct tw686x_dev *dev); >> +void tw686x_audio_free(struct tw686x_dev *dev); >> +void tw686x_audio_irq(struct tw686x_dev *dev, unsigned long requests, >> + unsigned int pb_status); >> > > Regards, > > Hans Thanks a lot for the thorough review! -- Ezequiel García, VanguardiaSur www.vanguardiasur.com.ar -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html