]> git.karo-electronics.de Git - mv-sheeva.git/blobdiff - drivers/staging/solo6x10/solo6010-p2m.c
Merge tag 'v2.6.38' of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6
[mv-sheeva.git] / drivers / staging / solo6x10 / solo6010-p2m.c
index 7ed3ed4b8f7eeda57f5be2bf5f2083c1b45915eb..956dea09348a34e30a8c9ad2412c947bf40142ff 100644 (file)
  */
 
 #include <linux/kernel.h>
+#include <linux/scatterlist.h>
 
 #include "solo6010.h"
 
-// #define SOLO_TEST_P2M
+/* #define SOLO_TEST_P2M */
 
 int solo_p2m_dma(struct solo6010_dev *solo_dev, u8 id, int wr,
                 void *sys_addr, u32 ext_addr, u32 size)
@@ -30,8 +31,9 @@ int solo_p2m_dma(struct solo6010_dev *solo_dev, u8 id, int wr,
        int ret;
 
        WARN_ON(!size);
-       WARN_ON(id >= SOLO_NR_P2M);
-       if (!size || id >= SOLO_NR_P2M)
+       BUG_ON(id >= SOLO_NR_P2M);
+
+       if (!size)
                return -EINVAL;
 
        dma_addr = pci_map_single(solo_dev->pdev, sys_addr, size,
@@ -47,42 +49,137 @@ 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,
                   dma_addr_t dma_addr, u32 ext_addr, u32 size)
+{
+       struct p2m_desc *desc = kzalloc(sizeof(*desc) * 2, GFP_DMA);
+       int ret;
+
+       if (desc == NULL)
+               return -ENOMEM;
+
+       solo_p2m_push_desc(&desc[1], wr, dma_addr, ext_addr, size, 0, 0);
+       ret = solo_p2m_dma_desc(solo_dev, id, desc, 2);
+       kfree(desc);
+
+       return ret;
+}
+
+void solo_p2m_push_desc(struct p2m_desc *desc, int wr, dma_addr_t dma_addr,
+                       u32 ext_addr, u32 size, int repeat, u32 ext_size)
+{
+       desc->ta = dma_addr;
+       desc->fa = ext_addr;
+
+       desc->ext = SOLO_P2M_COPY_SIZE(size >> 2);
+       desc->ctrl = SOLO_P2M_BURST_SIZE(SOLO_P2M_BURST_256) |
+               (wr ? SOLO_P2M_WRITE : 0) | SOLO_P2M_TRANS_ON;
+
+       /* Ext size only matters when we're repeating */
+       if (repeat) {
+               desc->ext |= SOLO_P2M_EXT_INC(ext_size >> 2);
+               desc->ctrl |=  SOLO_P2M_PCI_INC(size >> 2) |
+                       SOLO_P2M_REPEAT(repeat);
+       }
+}
+
+int solo_p2m_dma_desc(struct solo6010_dev *solo_dev, u8 id,
+                     struct p2m_desc *desc, int desc_count)
 {
        struct solo_p2m_dev *p2m_dev;
-       unsigned int timeout = 0;
+       unsigned int timeout;
+       int ret = 0;
+       u32 config = 0;
+       dma_addr_t desc_dma = 0;
 
-       WARN_ON(!size);
-       WARN_ON(id >= SOLO_NR_P2M);
-       if (!size || id >= SOLO_NR_P2M)
-               return -EINVAL;
+       BUG_ON(id >= SOLO_NR_P2M);
+       BUG_ON(!desc_count || desc_count > SOLO_NR_P2M_DESC);
 
        p2m_dev = &solo_dev->p2m_dev[id];
 
-       down(&p2m_dev->sem);
+       mutex_lock(&p2m_dev->mutex);
+
+       solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0);
 
-start_dma:
        INIT_COMPLETION(p2m_dev->completion);
        p2m_dev->error = 0;
-       solo_reg_write(solo_dev, SOLO_P2M_TAR_ADR(id), dma_addr);
-       solo_reg_write(solo_dev, SOLO_P2M_EXT_ADR(id), ext_addr);
-       solo_reg_write(solo_dev, SOLO_P2M_EXT_CFG(id),
-                      SOLO_P2M_COPY_SIZE(size >> 2));
-       solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id),
-                      SOLO_P2M_BURST_SIZE(SOLO_P2M_BURST_256) |
-                      (wr ? SOLO_P2M_WRITE : 0) | SOLO_P2M_TRANS_ON);
 
+       /* Enable the descriptors */
+       config = solo_reg_read(solo_dev, SOLO_P2M_CONFIG(id));
+       desc_dma = pci_map_single(solo_dev->pdev, desc,
+                                 desc_count * sizeof(*desc),
+                                 PCI_DMA_TODEVICE);
+       solo_reg_write(solo_dev, SOLO_P2M_DES_ADR(id), desc_dma);
+       solo_reg_write(solo_dev, SOLO_P2M_DESC_ID(id), desc_count - 1);
+       solo_reg_write(solo_dev, SOLO_P2M_CONFIG(id), config |
+                      SOLO_P2M_DESC_MODE);
+
+       /* Should have all descriptors completed from one interrupt */
        timeout = wait_for_completion_timeout(&p2m_dev->completion, HZ);
 
        solo_reg_write(solo_dev, SOLO_P2M_CONTROL(id), 0);
 
-       /* XXX Really looks to me like we will get stuck here if a
-        * real PCI P2M error occurs */
+       /* 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)
-               goto start_dma;
+               ret = -EIO;
+       else if (timeout == 0)
+               ret = -EAGAIN;
+
+       mutex_unlock(&p2m_dev->mutex);
+
+       WARN_ON_ONCE(ret);
 
-       up(&p2m_dev->sem);
+       return ret;
+}
+
+int solo_p2m_dma_sg(struct solo6010_dev *solo_dev, u8 id,
+                   struct p2m_desc *pdesc, int wr,
+                   struct scatterlist *sg, u32 sg_off,
+                   u32 ext_addr, u32 size)
+{
+       int i;
+       int idx;
+
+       BUG_ON(id >= SOLO_NR_P2M);
+
+       if (WARN_ON_ONCE(!size))
+               return -EINVAL;
+
+       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)) {
+               struct p2m_desc *desc = &pdesc[idx];
+               u32 sg_len = sg_dma_len(sg);
+               u32 len;
 
-       return (timeout == 0) ? -EAGAIN : 0;
+               if (sg_off >= sg_len) {
+                       sg_off -= sg_len;
+                       continue;
+               }
+
+               sg_len -= sg_off;
+               len = min(sg_len, size);
+
+               solo_p2m_push_desc(desc, wr, sg_dma_address(sg) + sg_off,
+                                  ext_addr, len, 0, 0);
+
+               size -= len;
+               ext_addr += len;
+               idx++;
+
+               sg_off = 0;
+       }
+
+       WARN_ON_ONCE(size || i >= SOLO_NR_P2M_DESC);
+
+       return solo_p2m_dma_desc(solo_dev, id, pdesc, idx);
 }
 
 #ifdef SOLO_TEST_P2M
@@ -147,13 +244,16 @@ static void run_p2m_test(struct solo6010_dev *solo_dev)
        return;
 }
 #else
-#define run_p2m_test(__solo)   do{}while(0)
+#define run_p2m_test(__solo)   do {} while (0)
 #endif
 
 void solo_p2m_isr(struct solo6010_dev *solo_dev, int id)
 {
+       struct solo_p2m_dev *p2m_dev = &solo_dev->p2m_dev[id];
+
        solo_reg_write(solo_dev, SOLO_IRQ_STAT, SOLO_IRQ_P2M(id));
-       complete(&solo_dev->p2m_dev[id].completion);
+
+       complete(&p2m_dev->completion);
 }
 
 void solo_p2m_error_isr(struct solo6010_dev *solo_dev, u32 status)
@@ -188,16 +288,14 @@ int solo_p2m_init(struct solo6010_dev *solo_dev)
        for (i = 0; i < SOLO_NR_P2M; i++) {
                p2m_dev = &solo_dev->p2m_dev[i];
 
-               sema_init(&p2m_dev->sem, 1);
+               mutex_init(&p2m_dev->mutex);
                init_completion(&p2m_dev->completion);
 
-               solo_reg_write(solo_dev, SOLO_P2M_DES_ADR(i),
-                              __pa(p2m_dev->desc));
-
                solo_reg_write(solo_dev, SOLO_P2M_CONTROL(i), 0);
                solo_reg_write(solo_dev, SOLO_P2M_CONFIG(i),
                               SOLO_P2M_CSC_16BIT_565 |
-                              SOLO_P2M_DMA_INTERVAL(0) |
+                              SOLO_P2M_DMA_INTERVAL(3) |
+                              SOLO_P2M_DESC_INTR_OPT |
                               SOLO_P2M_PCI_MASTER_MODE);
                solo6010_irq_on(solo_dev, SOLO_IRQ_P2M(i));
        }