Commit 0e4de059 authored by Ben Collins's avatar Ben Collins

solo6x10: Implement working P2M descriptor more DMA

We have to insert 1 dead descriptor first (all zereos), and then the real
data descriptors after that.

Everything uses descriptor mode now (8800 interrupts per second for display
v4l2 is now down to ~96).
Signed-off-by: default avatarBen Collins <bcollins@bluecherry.net>
parent 319fb3ba
...@@ -15,7 +15,6 @@ TODO (general): ...@@ -15,7 +15,6 @@ TODO (general):
* implement a CID control for motion areas/thresholds * implement a CID control for motion areas/thresholds
* implement CID controls for mozaic areas * implement CID controls for mozaic areas
* allow for higher level of interval (for < 1 fps) * allow for higher level of interval (for < 1 fps)
* Get proper descriptor mode working in p2m
* sound: * sound:
- implement playback via external sound jack - implement playback via external sound jack
- implement loopback of external sound jack with incoming audio? - implement loopback of external sound jack with incoming audio?
......
...@@ -50,11 +50,14 @@ int solo_p2m_dma(struct solo6010_dev *solo_dev, u8 id, int wr, ...@@ -50,11 +50,14 @@ int solo_p2m_dma(struct solo6010_dev *solo_dev, u8 id, int wr,
int solo_p2m_dma_t(struct solo6010_dev *solo_dev, u8 id, int wr, int solo_p2m_dma_t(struct solo6010_dev *solo_dev, u8 id, int wr,
dma_addr_t dma_addr, u32 ext_addr, u32 size) dma_addr_t dma_addr, u32 ext_addr, u32 size)
{ {
struct p2m_desc desc; struct p2m_desc *desc = kzalloc(sizeof(*desc) * 2, GFP_DMA);
solo_p2m_push_desc(&desc, wr, dma_addr, ext_addr, size, 0, 0); if (desc == NULL)
return -ENOMEM;
return solo_p2m_dma_desc(solo_dev, id, &desc, 1); solo_p2m_push_desc(&desc[1], wr, dma_addr, ext_addr, size, 0, 0);
return solo_p2m_dma_desc(solo_dev, id, desc, 2);
} }
void solo_p2m_push_desc(struct p2m_desc *desc, int wr, dma_addr_t dma_addr, void solo_p2m_push_desc(struct p2m_desc *desc, int wr, dma_addr_t dma_addr,
...@@ -81,34 +84,44 @@ int solo_p2m_dma_desc(struct solo6010_dev *solo_dev, u8 id, ...@@ -81,34 +84,44 @@ int solo_p2m_dma_desc(struct solo6010_dev *solo_dev, u8 id,
struct solo_p2m_dev *p2m_dev; struct solo_p2m_dev *p2m_dev;
unsigned int timeout; unsigned int timeout;
int ret = 0; int ret = 0;
u32 config = 0;
dma_addr_t desc_dma = 0;
BUG_ON(id >= SOLO_NR_P2M); BUG_ON(id >= SOLO_NR_P2M);
BUG_ON(desc_count > SOLO_NR_P2M_DESC); BUG_ON(!desc_count || desc_count > SOLO_NR_P2M_DESC);
p2m_dev = &solo_dev->p2m_dev[id]; p2m_dev = &solo_dev->p2m_dev[id];
mutex_lock(&p2m_dev->mutex); mutex_lock(&p2m_dev->mutex);
solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0);
INIT_COMPLETION(p2m_dev->completion); INIT_COMPLETION(p2m_dev->completion);
p2m_dev->error = 0; p2m_dev->error = 0;
/* Setup the descriptor count and base address */ /* Enable the descriptors */
p2m_dev->num_descs = desc_count; config = solo_reg_read(solo_dev, SOLO_P2M_CONFIG(id));
p2m_dev->descs = desc; desc_dma = pci_map_single(solo_dev->pdev, desc,
p2m_dev->desc_idx = 0; desc_count * sizeof(*desc),
PCI_DMA_TODEVICE);
/* We plug in the first descriptor here. The isr will take solo_reg_write(solo_dev, SOLO_P2M_DES_ADR(id), desc_dma);
* over from desc[1] after this. */ solo_reg_write(solo_dev, SOLO_P2M_DESC_ID(id), desc_count - 1);
solo_reg_write(solo_dev, SOLO_P2M_TAR_ADR(id), desc[0].ta); solo_reg_write(solo_dev, SOLO_P2M_CONFIG(id), config |
solo_reg_write(solo_dev, SOLO_P2M_EXT_ADR(id), desc[0].fa); SOLO_P2M_DESC_MODE);
solo_reg_write(solo_dev, SOLO_P2M_EXT_CFG(id), desc[0].ext);
solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), desc[0].ctrl);
/* Should have all descriptors completed from one interrupt */ /* Should have all descriptors completed from one interrupt */
timeout = wait_for_completion_timeout(&p2m_dev->completion, HZ); timeout = wait_for_completion_timeout(&p2m_dev->completion, HZ);
solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0); solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0);
/* Reset back to non-descriptor mode */
solo_reg_write(solo_dev, SOLO_P2M_CONFIG(id), config);
solo_reg_write(solo_dev, SOLO_P2M_DESC_ID(id), 0);
solo_reg_write(solo_dev, SOLO_P2M_DES_ADR(id), 0);
pci_unmap_single(solo_dev->pdev, desc_dma,
desc_count * sizeof(*desc),
PCI_DMA_TODEVICE);
if (p2m_dev->error) if (p2m_dev->error)
ret = -EIO; ret = -EIO;
else if (timeout == 0) else if (timeout == 0)
...@@ -134,9 +147,12 @@ int solo_p2m_dma_sg(struct solo6010_dev *solo_dev, u8 id, ...@@ -134,9 +147,12 @@ int solo_p2m_dma_sg(struct solo6010_dev *solo_dev, u8 id,
if (WARN_ON_ONCE(!size)) if (WARN_ON_ONCE(!size))
return -EINVAL; return -EINVAL;
for (i = idx = 0; i < SOLO_NR_P2M_DESC && sg && size > 0; memset(pdesc, 0, sizeof(*pdesc));
/* Should rewrite this to handle > SOLO_NR_P2M_DESC transactions */
for (i = 0, idx = 1; idx < SOLO_NR_P2M_DESC && sg && size > 0;
i++, sg = sg_next(sg)) { i++, sg = sg_next(sg)) {
struct p2m_desc *desc = &pdesc[i]; struct p2m_desc *desc = &pdesc[idx];
u32 sg_len = sg_dma_len(sg); u32 sg_len = sg_dma_len(sg);
u32 len; u32 len;
...@@ -231,26 +247,10 @@ static void run_p2m_test(struct solo6010_dev *solo_dev) ...@@ -231,26 +247,10 @@ static void run_p2m_test(struct solo6010_dev *solo_dev)
void solo_p2m_isr(struct solo6010_dev *solo_dev, int id) void solo_p2m_isr(struct solo6010_dev *solo_dev, int id)
{ {
struct solo_p2m_dev *p2m_dev = &solo_dev->p2m_dev[id]; struct solo_p2m_dev *p2m_dev = &solo_dev->p2m_dev[id];
struct p2m_desc *desc;
solo_reg_write(solo_dev, SOLO_IRQ_STAT, SOLO_IRQ_P2M(id)); solo_reg_write(solo_dev, SOLO_IRQ_STAT, SOLO_IRQ_P2M(id));
p2m_dev->desc_idx++;
if (p2m_dev->desc_idx >= p2m_dev->num_descs) {
complete(&p2m_dev->completion); complete(&p2m_dev->completion);
return;
}
/* Reset the p2m and start the next one */
solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0);
desc = &p2m_dev->descs[p2m_dev->desc_idx];
solo_reg_write(solo_dev, SOLO_P2M_TAR_ADR(id), desc->ta);
solo_reg_write(solo_dev, SOLO_P2M_EXT_ADR(id), desc->fa);
solo_reg_write(solo_dev, SOLO_P2M_EXT_CFG(id), desc->ext);
solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), desc->ctrl);
} }
void solo_p2m_error_isr(struct solo6010_dev *solo_dev, u32 status) void solo_p2m_error_isr(struct solo6010_dev *solo_dev, u32 status)
...@@ -292,6 +292,7 @@ int solo_p2m_init(struct solo6010_dev *solo_dev) ...@@ -292,6 +292,7 @@ int solo_p2m_init(struct solo6010_dev *solo_dev)
solo_reg_write(solo_dev, SOLO_P2M_CONFIG(i), solo_reg_write(solo_dev, SOLO_P2M_CONFIG(i),
SOLO_P2M_CSC_16BIT_565 | SOLO_P2M_CSC_16BIT_565 |
SOLO_P2M_DMA_INTERVAL(3) | SOLO_P2M_DMA_INTERVAL(3) |
SOLO_P2M_DESC_INTR_OPT |
SOLO_P2M_PCI_MASTER_MODE); SOLO_P2M_PCI_MASTER_MODE);
solo6010_irq_on(solo_dev, SOLO_IRQ_P2M(i)); solo6010_irq_on(solo_dev, SOLO_IRQ_P2M(i));
} }
......
...@@ -206,7 +206,9 @@ static int solo_v4l2_set_ch(struct solo6010_dev *solo_dev, u8 ch) ...@@ -206,7 +206,9 @@ static int solo_v4l2_set_ch(struct solo6010_dev *solo_dev, u8 ch)
static void disp_reset_desc(struct solo_filehandle *fh) static void disp_reset_desc(struct solo_filehandle *fh)
{ {
fh->desc_idx = 0; /* We use desc mode, which ignores desc 0 */
memset(fh->desc, 0, sizeof(*fh->desc));
fh->desc_idx = 1;
} }
static int disp_flush_descs(struct solo_filehandle *fh) static int disp_flush_descs(struct solo_filehandle *fh)
......
...@@ -137,9 +137,6 @@ struct solo_p2m_dev { ...@@ -137,9 +137,6 @@ struct solo_p2m_dev {
struct mutex mutex; struct mutex mutex;
struct completion completion; struct completion completion;
int error; int error;
int num_descs;
int desc_idx;
struct p2m_desc *descs;
}; };
#define OSD_TEXT_MAX 30 #define OSD_TEXT_MAX 30
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment