Merge tag 'gpio-for-linus' of git://git.secretlab.ca/git/linux
[deliverable/linux.git] / drivers / ntb / ntb_transport.c
index bf7ade14c742b29d4f4cca257e662ff98176f204..e0bdfd7f9930aab4b42984b468ff87fac2868f44 100644 (file)
@@ -58,7 +58,7 @@
 #include <linux/ntb.h>
 #include "ntb_hw.h"
 
-#define NTB_TRANSPORT_VERSION  1
+#define NTB_TRANSPORT_VERSION  2
 
 static unsigned int transport_mtu = 0x401E;
 module_param(transport_mtu, uint, 0644);
@@ -78,6 +78,10 @@ struct ntb_queue_entry {
        unsigned int flags;
 };
 
+struct ntb_rx_info {
+       unsigned int entry;
+};
+
 struct ntb_transport_qp {
        struct ntb_transport *transport;
        struct ntb_device *ndev;
@@ -87,13 +91,16 @@ struct ntb_transport_qp {
        bool qp_link;
        u8 qp_num;      /* Only 64 QP's are allowed.  0-63 */
 
+       struct ntb_rx_info __iomem *rx_info;
+       struct ntb_rx_info *remote_rx_info;
+
        void (*tx_handler) (struct ntb_transport_qp *qp, void *qp_data,
                            void *data, int len);
        struct list_head tx_free_q;
        spinlock_t ntb_tx_free_q_lock;
-       void *tx_mw_begin;
-       void *tx_mw_end;
-       void *tx_offset;
+       void __iomem *tx_mw;
+       unsigned int tx_index;
+       unsigned int tx_max_entry;
        unsigned int tx_max_frame;
 
        void (*rx_handler) (struct ntb_transport_qp *qp, void *qp_data,
@@ -103,9 +110,9 @@ struct ntb_transport_qp {
        struct list_head rx_free_q;
        spinlock_t ntb_rx_pend_q_lock;
        spinlock_t ntb_rx_free_q_lock;
-       void *rx_buff_begin;
-       void *rx_buff_end;
-       void *rx_offset;
+       void *rx_buff;
+       unsigned int rx_index;
+       unsigned int rx_max_entry;
        unsigned int rx_max_frame;
 
        void (*event_handler) (void *data, int status);
@@ -159,7 +166,7 @@ enum {
 };
 
 struct ntb_payload_header {
-       u64 ver;
+       unsigned int ver;
        unsigned int len;
        unsigned int flags;
 };
@@ -212,7 +219,7 @@ static int ntb_client_remove(struct device *dev)
        return 0;
 }
 
-struct bus_type ntb_bus_type = {
+static struct bus_type ntb_bus_type = {
        .name = "ntb_bus",
        .match = ntb_match_bus,
        .probe = ntb_client_probe,
@@ -364,20 +371,18 @@ void ntb_unregister_client(struct ntb_client *drv)
 }
 EXPORT_SYMBOL_GPL(ntb_unregister_client);
 
-static int debugfs_open(struct inode *inode, struct file *filp)
-{
-       filp->private_data = inode->i_private;
-       return 0;
-}
-
 static ssize_t debugfs_read(struct file *filp, char __user *ubuf, size_t count,
                            loff_t *offp)
 {
        struct ntb_transport_qp *qp;
-       char buf[1024];
+       char *buf;
        ssize_t ret, out_offset, out_count;
 
-       out_count = 1024;
+       out_count = 600;
+
+       buf = kmalloc(out_count, GFP_KERNEL);
+       if (!buf)
+               return -ENOMEM;
 
        qp = filp->private_data;
        out_offset = 0;
@@ -396,11 +401,11 @@ static ssize_t debugfs_read(struct file *filp, char __user *ubuf, size_t count,
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
                               "rx_err_ver - \t%llu\n", qp->rx_err_ver);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "rx_buff_begin - %p\n", qp->rx_buff_begin);
+                              "rx_buff - \t%p\n", qp->rx_buff);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "rx_offset - \t%p\n", qp->rx_offset);
+                              "rx_index - \t%u\n", qp->rx_index);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "rx_buff_end - \t%p\n", qp->rx_buff_end);
+                              "rx_max_entry - \t%u\n", qp->rx_max_entry);
 
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
                               "tx_bytes - \t%llu\n", qp->tx_bytes);
@@ -409,23 +414,26 @@ static ssize_t debugfs_read(struct file *filp, char __user *ubuf, size_t count,
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
                               "tx_ring_full - \t%llu\n", qp->tx_ring_full);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "tx_mw_begin - \t%p\n", qp->tx_mw_begin);
+                              "tx_mw - \t%p\n", qp->tx_mw);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "tx_offset - \t%p\n", qp->tx_offset);
+                              "tx_index - \t%u\n", qp->tx_index);
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "tx_mw_end - \t%p\n", qp->tx_mw_end);
+                              "tx_max_entry - \t%u\n", qp->tx_max_entry);
 
        out_offset += snprintf(buf + out_offset, out_count - out_offset,
-                              "QP Link %s\n", (qp->qp_link == NTB_LINK_UP) ?
+                              "\nQP Link %s\n", (qp->qp_link == NTB_LINK_UP) ?
                               "Up" : "Down");
+       if (out_offset > out_count)
+               out_offset = out_count;
 
        ret = simple_read_from_buffer(ubuf, count, offp, buf, out_offset);
+       kfree(buf);
        return ret;
 }
 
 static const struct file_operations ntb_qp_debugfs_stats = {
        .owner = THIS_MODULE,
-       .open = debugfs_open,
+       .open = simple_open,
        .read = debugfs_read,
 };
 
@@ -464,27 +472,33 @@ static void ntb_transport_setup_qp_mw(struct ntb_transport *nt,
        struct ntb_transport_qp *qp = &nt->qps[qp_num];
        unsigned int rx_size, num_qps_mw;
        u8 mw_num = QP_TO_MW(qp_num);
-       void *offset;
+       unsigned int i;
 
-       WARN_ON(nt->mw[mw_num].virt_addr == 0);
+       WARN_ON(nt->mw[mw_num].virt_addr == NULL);
 
        if (nt->max_qps % NTB_NUM_MW && mw_num < nt->max_qps % NTB_NUM_MW)
                num_qps_mw = nt->max_qps / NTB_NUM_MW + 1;
        else
                num_qps_mw = nt->max_qps / NTB_NUM_MW;
 
-       rx_size = nt->mw[mw_num].size / num_qps_mw;
-       qp->rx_buff_begin = nt->mw[mw_num].virt_addr +
-                           (qp_num / NTB_NUM_MW * rx_size);
-       qp->rx_buff_end = qp->rx_buff_begin + rx_size;
-       qp->rx_offset = qp->rx_buff_begin;
+       rx_size = (unsigned int) nt->mw[mw_num].size / num_qps_mw;
+       qp->remote_rx_info = nt->mw[mw_num].virt_addr +
+                            (qp_num / NTB_NUM_MW * rx_size);
+       rx_size -= sizeof(struct ntb_rx_info);
+
+       qp->rx_buff = qp->remote_rx_info + sizeof(struct ntb_rx_info);
        qp->rx_max_frame = min(transport_mtu, rx_size);
+       qp->rx_max_entry = rx_size / qp->rx_max_frame;
+       qp->rx_index = 0;
+
+       qp->remote_rx_info->entry = qp->rx_max_entry;
 
        /* setup the hdr offsets with 0's */
-       for (offset = qp->rx_buff_begin + qp->rx_max_frame -
-                     sizeof(struct ntb_payload_header);
-            offset < qp->rx_buff_end; offset += qp->rx_max_frame)
+       for (i = 0; i < qp->rx_max_entry; i++) {
+               void *offset = qp->rx_buff + qp->rx_max_frame * (i + 1) -
+                              sizeof(struct ntb_payload_header);
                memset(offset, 0, sizeof(struct ntb_payload_header));
+       }
 
        qp->rx_pkts = 0;
        qp->tx_pkts = 0;
@@ -761,12 +775,15 @@ static void ntb_transport_init_queue(struct ntb_transport *nt,
        else
                num_qps_mw = nt->max_qps / NTB_NUM_MW;
 
-       tx_size = ntb_get_mw_size(qp->ndev, mw_num) / num_qps_mw;
-       qp->tx_mw_begin = ntb_get_mw_vbase(nt->ndev, mw_num) +
-                         (qp_num / NTB_NUM_MW * tx_size);
-       qp->tx_mw_end = qp->tx_mw_begin + tx_size;
-       qp->tx_offset = qp->tx_mw_begin;
+       tx_size = (unsigned int) ntb_get_mw_size(qp->ndev, mw_num) / num_qps_mw;
+       qp->rx_info = ntb_get_mw_vbase(nt->ndev, mw_num) +
+                     (qp_num / NTB_NUM_MW * tx_size);
+       tx_size -= sizeof(struct ntb_rx_info);
+
+       qp->tx_mw = qp->rx_info + sizeof(struct ntb_rx_info);
        qp->tx_max_frame = min(transport_mtu, tx_size);
+       qp->tx_max_entry = tx_size / qp->tx_max_frame;
+       qp->tx_index = 0;
 
        if (nt->debugfs_dir) {
                char debugfs_name[4];
@@ -893,25 +910,15 @@ void ntb_transport_free(void *transport)
 static void ntb_rx_copy_task(struct ntb_transport_qp *qp,
                             struct ntb_queue_entry *entry, void *offset)
 {
-
-       struct ntb_payload_header *hdr;
-
-       BUG_ON(offset < qp->rx_buff_begin ||
-              offset + qp->rx_max_frame >= qp->rx_buff_end);
-
-       hdr = offset + qp->rx_max_frame - sizeof(struct ntb_payload_header);
-       entry->len = hdr->len;
+       void *cb_data = entry->cb_data;
+       unsigned int len = entry->len;
 
        memcpy(entry->buf, offset, entry->len);
 
-       /* Ensure that the data is fully copied out before clearing the flag */
-       wmb();
-       hdr->flags = 0;
+       ntb_list_add(&qp->ntb_rx_free_q_lock, &entry->entry, &qp->rx_free_q);
 
        if (qp->rx_handler && qp->client_ready == NTB_LINK_UP)
-               qp->rx_handler(qp, qp->cb_data, entry->cb_data, entry->len);
-
-       ntb_list_add(&qp->ntb_rx_free_q_lock, &entry->entry, &qp->rx_free_q);
+               qp->rx_handler(qp, qp->cb_data, cb_data, len);
 }
 
 static int ntb_process_rxc(struct ntb_transport_qp *qp)
@@ -920,33 +927,31 @@ static int ntb_process_rxc(struct ntb_transport_qp *qp)
        struct ntb_queue_entry *entry;
        void *offset;
 
+       offset = qp->rx_buff + qp->rx_max_frame * qp->rx_index;
+       hdr = offset + qp->rx_max_frame - sizeof(struct ntb_payload_header);
+
        entry = ntb_list_rm(&qp->ntb_rx_pend_q_lock, &qp->rx_pend_q);
        if (!entry) {
-               hdr = offset + qp->rx_max_frame -
-                     sizeof(struct ntb_payload_header);
                dev_dbg(&ntb_query_pdev(qp->ndev)->dev,
-                       "no buffer - HDR ver %llu, len %d, flags %x\n",
+                       "no buffer - HDR ver %u, len %d, flags %x\n",
                        hdr->ver, hdr->len, hdr->flags);
                qp->rx_err_no_buf++;
                return -ENOMEM;
        }
 
-       offset = qp->rx_offset;
-       hdr = offset + qp->rx_max_frame - sizeof(struct ntb_payload_header);
-
        if (!(hdr->flags & DESC_DONE_FLAG)) {
                ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry,
-                                 &qp->rx_pend_q);
+                            &qp->rx_pend_q);
                qp->rx_ring_empty++;
                return -EAGAIN;
        }
 
-       if (hdr->ver != qp->rx_pkts) {
+       if (hdr->ver != (u32) qp->rx_pkts) {
                dev_dbg(&ntb_query_pdev(qp->ndev)->dev,
-                       "qp %d: version mismatch, expected %llu - got %llu\n",
+                       "qp %d: version mismatch, expected %llu - got %u\n",
                        qp->qp_num, qp->rx_pkts, hdr->ver);
                ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry,
-                                 &qp->rx_pend_q);
+                            &qp->rx_pend_q);
                qp->rx_err_ver++;
                return -EIO;
        }
@@ -955,31 +960,21 @@ static int ntb_process_rxc(struct ntb_transport_qp *qp)
                ntb_qp_link_down(qp);
 
                ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry,
-                                 &qp->rx_pend_q);
-
-               /* Ensure that the data is fully copied out before clearing the
-                * done flag
-                */
-               wmb();
-               hdr->flags = 0;
+                            &qp->rx_pend_q);
                goto out;
        }
 
        dev_dbg(&ntb_query_pdev(qp->ndev)->dev,
-               "rx offset %p, ver %llu - %d payload received, buf size %d\n",
-               qp->rx_offset, hdr->ver, hdr->len, entry->len);
+               "rx offset %u, ver %u - %d payload received, buf size %d\n",
+               qp->rx_index, hdr->ver, hdr->len, entry->len);
 
-       if (hdr->len <= entry->len)
+       if (hdr->len <= entry->len) {
+               entry->len = hdr->len;
                ntb_rx_copy_task(qp, entry, offset);
-       else {
+       else {
                ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry,
-                                 &qp->rx_pend_q);
+                            &qp->rx_pend_q);
 
-               /* Ensure that the data is fully copied out before clearing the
-                * done flag
-                */
-               wmb();
-               hdr->flags = 0;
                qp->rx_err_oflow++;
                dev_dbg(&ntb_query_pdev(qp->ndev)->dev,
                        "RX overflow! Wanted %d got %d\n",
@@ -990,9 +985,13 @@ static int ntb_process_rxc(struct ntb_transport_qp *qp)
        qp->rx_pkts++;
 
 out:
-       qp->rx_offset += qp->rx_max_frame;
-       if (qp->rx_offset + qp->rx_max_frame >= qp->rx_buff_end)
-               qp->rx_offset = qp->rx_buff_begin;
+       /* Ensure that the data is fully copied out before clearing the flag */
+       wmb();
+       hdr->flags = 0;
+       iowrite32(qp->rx_index, &qp->rx_info->entry);
+
+       qp->rx_index++;
+       qp->rx_index %= qp->rx_max_entry;
 
        return 0;
 }
@@ -1019,22 +1018,19 @@ static void ntb_transport_rxc_db(void *data, int db_num)
 
 static void ntb_tx_copy_task(struct ntb_transport_qp *qp,
                             struct ntb_queue_entry *entry,
-                            void *offset)
+                            void __iomem *offset)
 {
-       struct ntb_payload_header *hdr;
-
-       BUG_ON(offset < qp->tx_mw_begin ||
-              offset + qp->tx_max_frame >= qp->tx_mw_end);
+       struct ntb_payload_header __iomem *hdr;
 
        memcpy_toio(offset, entry->buf, entry->len);
 
        hdr = offset + qp->tx_max_frame - sizeof(struct ntb_payload_header);
-       hdr->len = entry->len;
-       hdr->ver = qp->tx_pkts;
+       iowrite32(entry->len, &hdr->len);
+       iowrite32((u32) qp->tx_pkts, &hdr->ver);
 
        /* Ensure that the data is fully copied out before setting the flag */
        wmb();
-       hdr->flags = entry->flags | DESC_DONE_FLAG;
+       iowrite32(entry->flags | DESC_DONE_FLAG, &hdr->flags);
 
        ntb_ring_sdb(qp->ndev, qp->qp_num);
 
@@ -1056,16 +1052,14 @@ static void ntb_tx_copy_task(struct ntb_transport_qp *qp,
 static int ntb_process_tx(struct ntb_transport_qp *qp,
                          struct ntb_queue_entry *entry)
 {
-       struct ntb_payload_header *hdr;
-       void *offset;
+       void __iomem *offset;
 
-       offset = qp->tx_offset;
-       hdr = offset + qp->tx_max_frame - sizeof(struct ntb_payload_header);
+       offset = qp->tx_mw + qp->tx_max_frame * qp->tx_index;
 
-       dev_dbg(&ntb_query_pdev(qp->ndev)->dev, "%lld - offset %p, tx %p, entry len %d flags %x buff %p\n",
-                qp->tx_pkts, offset, qp->tx_offset, entry->len, entry->flags,
-                entry->buf);
-       if (hdr->flags) {
+       dev_dbg(&ntb_query_pdev(qp->ndev)->dev, "%lld - offset %p, tx %u, entry len %d flags %x buff %p\n",
+               qp->tx_pkts, offset, qp->tx_index, entry->len, entry->flags,
+               entry->buf);
+       if (qp->tx_index == qp->remote_rx_info->entry) {
                qp->tx_ring_full++;
                return -EAGAIN;
        }
@@ -1081,9 +1075,8 @@ static int ntb_process_tx(struct ntb_transport_qp *qp,
 
        ntb_tx_copy_task(qp, entry, offset);
 
-       qp->tx_offset += qp->tx_max_frame;
-       if (qp->tx_offset + qp->tx_max_frame >= qp->tx_mw_end)
-               qp->tx_offset = qp->tx_mw_begin;
+       qp->tx_index++;
+       qp->tx_index %= qp->tx_max_entry;
 
        qp->tx_pkts++;
 
@@ -1103,8 +1096,7 @@ static void ntb_send_link_down(struct ntb_transport_qp *qp)
        dev_info(&pdev->dev, "qp %d: Link Down\n", qp->qp_num);
 
        for (i = 0; i < NTB_LINK_DOWN_TIMEOUT; i++) {
-               entry = ntb_list_rm(&qp->ntb_tx_free_q_lock,
-                                        &qp->tx_free_q);
+               entry = ntb_list_rm(&qp->ntb_tx_free_q_lock, &qp->tx_free_q);
                if (entry)
                        break;
                msleep(100);
@@ -1173,7 +1165,7 @@ ntb_transport_create_queue(void *data, struct pci_dev *pdev,
                        goto err1;
 
                ntb_list_add(&qp->ntb_rx_free_q_lock, &entry->entry,
-                                 &qp->rx_free_q);
+                            &qp->rx_free_q);
        }
 
        for (i = 0; i < NTB_QP_DEF_NUM_ENTRIES; i++) {
@@ -1182,7 +1174,7 @@ ntb_transport_create_queue(void *data, struct pci_dev *pdev,
                        goto err2;
 
                ntb_list_add(&qp->ntb_tx_free_q_lock, &entry->entry,
-                                 &qp->tx_free_q);
+                            &qp->tx_free_q);
        }
 
        tasklet_init(&qp->rx_work, ntb_transport_rx, (unsigned long) qp);
@@ -1199,12 +1191,10 @@ ntb_transport_create_queue(void *data, struct pci_dev *pdev,
 err3:
        tasklet_disable(&qp->rx_work);
 err2:
-       while ((entry =
-               ntb_list_rm(&qp->ntb_tx_free_q_lock, &qp->tx_free_q)))
+       while ((entry = ntb_list_rm(&qp->ntb_tx_free_q_lock, &qp->tx_free_q)))
                kfree(entry);
 err1:
-       while ((entry =
-               ntb_list_rm(&qp->ntb_rx_free_q_lock, &qp->rx_free_q)))
+       while ((entry = ntb_list_rm(&qp->ntb_rx_free_q_lock, &qp->rx_free_q)))
                kfree(entry);
        set_bit(free_queue, &nt->qp_bitmap);
 err:
@@ -1231,18 +1221,15 @@ void ntb_transport_free_queue(struct ntb_transport_qp *qp)
        ntb_unregister_db_callback(qp->ndev, qp->qp_num);
        tasklet_disable(&qp->rx_work);
 
-       while ((entry =
-               ntb_list_rm(&qp->ntb_rx_free_q_lock, &qp->rx_free_q)))
+       while ((entry = ntb_list_rm(&qp->ntb_rx_free_q_lock, &qp->rx_free_q)))
                kfree(entry);
 
-       while ((entry =
-               ntb_list_rm(&qp->ntb_rx_pend_q_lock, &qp->rx_pend_q))) {
+       while ((entry = ntb_list_rm(&qp->ntb_rx_pend_q_lock, &qp->rx_pend_q))) {
                dev_warn(&pdev->dev, "Freeing item from a non-empty queue\n");
                kfree(entry);
        }
 
-       while ((entry =
-               ntb_list_rm(&qp->ntb_tx_free_q_lock, &qp->tx_free_q)))
+       while ((entry = ntb_list_rm(&qp->ntb_tx_free_q_lock, &qp->tx_free_q)))
                kfree(entry);
 
        set_bit(qp->qp_num, &qp->transport->qp_bitmap);
@@ -1276,8 +1263,7 @@ void *ntb_transport_rx_remove(struct ntb_transport_qp *qp, unsigned int *len)
        buf = entry->cb_data;
        *len = entry->len;
 
-       ntb_list_add(&qp->ntb_rx_free_q_lock, &entry->entry,
-                         &qp->rx_free_q);
+       ntb_list_add(&qp->ntb_rx_free_q_lock, &entry->entry, &qp->rx_free_q);
 
        return buf;
 }
@@ -1311,8 +1297,7 @@ int ntb_transport_rx_enqueue(struct ntb_transport_qp *qp, void *cb, void *data,
        entry->buf = data;
        entry->len = len;
 
-       ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry,
-                         &qp->rx_pend_q);
+       ntb_list_add(&qp->ntb_rx_pend_q_lock, &entry->entry, &qp->rx_pend_q);
 
        return 0;
 }
This page took 0.040225 seconds and 5 git commands to generate.