Commit 78b9c0f9 authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'for-linus' of master.kernel.org:/pub/scm/linux/kernel/git/roland/infiniband

parents 6b482c67 94382f35
...@@ -31,7 +31,7 @@ ...@@ -31,7 +31,7 @@
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE. * SOFTWARE.
* *
* $Id: user_mad.c 2814 2005-07-06 19:14:09Z halr $ * $Id: user_mad.c 4010 2005-11-09 23:11:56Z roland $
*/ */
#include <linux/module.h> #include <linux/module.h>
...@@ -116,7 +116,7 @@ struct ib_umad_file { ...@@ -116,7 +116,7 @@ struct ib_umad_file {
spinlock_t recv_lock; spinlock_t recv_lock;
wait_queue_head_t recv_wait; wait_queue_head_t recv_wait;
struct ib_mad_agent *agent[IB_UMAD_MAX_AGENTS]; struct ib_mad_agent *agent[IB_UMAD_MAX_AGENTS];
struct ib_mr *mr[IB_UMAD_MAX_AGENTS]; int agents_dead;
}; };
struct ib_umad_packet { struct ib_umad_packet {
...@@ -145,6 +145,12 @@ static void ib_umad_release_dev(struct kref *ref) ...@@ -145,6 +145,12 @@ static void ib_umad_release_dev(struct kref *ref)
kfree(dev); kfree(dev);
} }
/* caller must hold port->mutex at least for reading */
static struct ib_mad_agent *__get_agent(struct ib_umad_file *file, int id)
{
return file->agents_dead ? NULL : file->agent[id];
}
static int queue_packet(struct ib_umad_file *file, static int queue_packet(struct ib_umad_file *file,
struct ib_mad_agent *agent, struct ib_mad_agent *agent,
struct ib_umad_packet *packet) struct ib_umad_packet *packet)
...@@ -152,10 +158,11 @@ static int queue_packet(struct ib_umad_file *file, ...@@ -152,10 +158,11 @@ static int queue_packet(struct ib_umad_file *file,
int ret = 1; int ret = 1;
down_read(&file->port->mutex); down_read(&file->port->mutex);
for (packet->mad.hdr.id = 0; for (packet->mad.hdr.id = 0;
packet->mad.hdr.id < IB_UMAD_MAX_AGENTS; packet->mad.hdr.id < IB_UMAD_MAX_AGENTS;
packet->mad.hdr.id++) packet->mad.hdr.id++)
if (agent == file->agent[packet->mad.hdr.id]) { if (agent == __get_agent(file, packet->mad.hdr.id)) {
spin_lock_irq(&file->recv_lock); spin_lock_irq(&file->recv_lock);
list_add_tail(&packet->list, &file->recv_list); list_add_tail(&packet->list, &file->recv_list);
spin_unlock_irq(&file->recv_lock); spin_unlock_irq(&file->recv_lock);
...@@ -327,7 +334,7 @@ static ssize_t ib_umad_write(struct file *filp, const char __user *buf, ...@@ -327,7 +334,7 @@ static ssize_t ib_umad_write(struct file *filp, const char __user *buf,
down_read(&file->port->mutex); down_read(&file->port->mutex);
agent = file->agent[packet->mad.hdr.id]; agent = __get_agent(file, packet->mad.hdr.id);
if (!agent) { if (!agent) {
ret = -EINVAL; ret = -EINVAL;
goto err_up; goto err_up;
...@@ -481,7 +488,7 @@ static int ib_umad_reg_agent(struct ib_umad_file *file, unsigned long arg) ...@@ -481,7 +488,7 @@ static int ib_umad_reg_agent(struct ib_umad_file *file, unsigned long arg)
} }
for (agent_id = 0; agent_id < IB_UMAD_MAX_AGENTS; ++agent_id) for (agent_id = 0; agent_id < IB_UMAD_MAX_AGENTS; ++agent_id)
if (!file->agent[agent_id]) if (!__get_agent(file, agent_id))
goto found; goto found;
ret = -ENOMEM; ret = -ENOMEM;
...@@ -505,29 +512,15 @@ found: ...@@ -505,29 +512,15 @@ found:
goto out; goto out;
} }
file->agent[agent_id] = agent;
file->mr[agent_id] = ib_get_dma_mr(agent->qp->pd, IB_ACCESS_LOCAL_WRITE);
if (IS_ERR(file->mr[agent_id])) {
ret = -ENOMEM;
goto err;
}
if (put_user(agent_id, if (put_user(agent_id,
(u32 __user *) (arg + offsetof(struct ib_user_mad_reg_req, id)))) { (u32 __user *) (arg + offsetof(struct ib_user_mad_reg_req, id)))) {
ret = -EFAULT; ret = -EFAULT;
goto err_mr; ib_unregister_mad_agent(agent);
goto out;
} }
file->agent[agent_id] = agent;
ret = 0; ret = 0;
goto out;
err_mr:
ib_dereg_mr(file->mr[agent_id]);
err:
file->agent[agent_id] = NULL;
ib_unregister_mad_agent(agent);
out: out:
up_write(&file->port->mutex); up_write(&file->port->mutex);
...@@ -536,27 +529,29 @@ out: ...@@ -536,27 +529,29 @@ out:
static int ib_umad_unreg_agent(struct ib_umad_file *file, unsigned long arg) static int ib_umad_unreg_agent(struct ib_umad_file *file, unsigned long arg)
{ {
struct ib_mad_agent *agent = NULL;
u32 id; u32 id;
int ret = 0; int ret = 0;
down_write(&file->port->mutex); if (get_user(id, (u32 __user *) arg))
return -EFAULT;
if (get_user(id, (u32 __user *) arg)) { down_write(&file->port->mutex);
ret = -EFAULT;
goto out;
}
if (id < 0 || id >= IB_UMAD_MAX_AGENTS || !file->agent[id]) { if (id < 0 || id >= IB_UMAD_MAX_AGENTS || !__get_agent(file, id)) {
ret = -EINVAL; ret = -EINVAL;
goto out; goto out;
} }
ib_dereg_mr(file->mr[id]); agent = file->agent[id];
ib_unregister_mad_agent(file->agent[id]);
file->agent[id] = NULL; file->agent[id] = NULL;
out: out:
up_write(&file->port->mutex); up_write(&file->port->mutex);
if (agent)
ib_unregister_mad_agent(agent);
return ret; return ret;
} }
...@@ -621,23 +616,29 @@ static int ib_umad_close(struct inode *inode, struct file *filp) ...@@ -621,23 +616,29 @@ static int ib_umad_close(struct inode *inode, struct file *filp)
struct ib_umad_file *file = filp->private_data; struct ib_umad_file *file = filp->private_data;
struct ib_umad_device *dev = file->port->umad_dev; struct ib_umad_device *dev = file->port->umad_dev;
struct ib_umad_packet *packet, *tmp; struct ib_umad_packet *packet, *tmp;
int already_dead;
int i; int i;
down_write(&file->port->mutex); down_write(&file->port->mutex);
for (i = 0; i < IB_UMAD_MAX_AGENTS; ++i)
if (file->agent[i]) { already_dead = file->agents_dead;
ib_dereg_mr(file->mr[i]); file->agents_dead = 1;
ib_unregister_mad_agent(file->agent[i]);
}
list_for_each_entry_safe(packet, tmp, &file->recv_list, list) list_for_each_entry_safe(packet, tmp, &file->recv_list, list)
kfree(packet); kfree(packet);
list_del(&file->port_list); list_del(&file->port_list);
up_write(&file->port->mutex);
kfree(file); downgrade_write(&file->port->mutex);
if (!already_dead)
for (i = 0; i < IB_UMAD_MAX_AGENTS; ++i)
if (file->agent[i])
ib_unregister_mad_agent(file->agent[i]);
up_read(&file->port->mutex);
kfree(file);
kref_put(&dev->ref, ib_umad_release_dev); kref_put(&dev->ref, ib_umad_release_dev);
return 0; return 0;
...@@ -801,7 +802,7 @@ static int ib_umad_init_port(struct ib_device *device, int port_num, ...@@ -801,7 +802,7 @@ static int ib_umad_init_port(struct ib_device *device, int port_num,
goto err_class; goto err_class;
port->sm_dev->owner = THIS_MODULE; port->sm_dev->owner = THIS_MODULE;
port->sm_dev->ops = &umad_sm_fops; port->sm_dev->ops = &umad_sm_fops;
kobject_set_name(&port->dev->kobj, "issm%d", port->dev_num); kobject_set_name(&port->sm_dev->kobj, "issm%d", port->dev_num);
if (cdev_add(port->sm_dev, base_dev + port->dev_num + IB_UMAD_MAX_PORTS, 1)) if (cdev_add(port->sm_dev, base_dev + port->dev_num + IB_UMAD_MAX_PORTS, 1))
goto err_sm_cdev; goto err_sm_cdev;
...@@ -863,13 +864,35 @@ static void ib_umad_kill_port(struct ib_umad_port *port) ...@@ -863,13 +864,35 @@ static void ib_umad_kill_port(struct ib_umad_port *port)
port->ib_dev = NULL; port->ib_dev = NULL;
list_for_each_entry(file, &port->file_list, port_list) /*
for (id = 0; id < IB_UMAD_MAX_AGENTS; ++id) { * Now go through the list of files attached to this port and
if (!file->agent[id]) * unregister all of their MAD agents. We need to hold
continue; * port->mutex while doing this to avoid racing with
ib_dereg_mr(file->mr[id]); * ib_umad_close(), but we can't hold the mutex for writing
* while calling ib_unregister_mad_agent(), since that might
* deadlock by calling back into queue_packet(). So we
* downgrade our lock to a read lock, and then drop and
* reacquire the write lock for the next iteration.
*
* We do list_del_init() on the file's list_head so that the
* list_del in ib_umad_close() is still OK, even after the
* file is removed from the list.
*/
while (!list_empty(&port->file_list)) {
file = list_entry(port->file_list.next, struct ib_umad_file,
port_list);
file->agents_dead = 1;
list_del_init(&file->port_list);
downgrade_write(&port->mutex);
for (id = 0; id < IB_UMAD_MAX_AGENTS; ++id)
if (file->agent[id])
ib_unregister_mad_agent(file->agent[id]); ib_unregister_mad_agent(file->agent[id]);
file->agent[id] = NULL;
up_read(&port->mutex);
down_write(&port->mutex);
} }
up_write(&port->mutex); up_write(&port->mutex);
...@@ -913,7 +936,7 @@ static void ib_umad_add_one(struct ib_device *device) ...@@ -913,7 +936,7 @@ static void ib_umad_add_one(struct ib_device *device)
err: err:
while (--i >= s) while (--i >= s)
ib_umad_kill_port(&umad_dev->port[i]); ib_umad_kill_port(&umad_dev->port[i - s]);
kref_put(&umad_dev->ref, ib_umad_release_dev); kref_put(&umad_dev->ref, ib_umad_release_dev);
} }
......
...@@ -708,7 +708,7 @@ ssize_t ib_uverbs_poll_cq(struct ib_uverbs_file *file, ...@@ -708,7 +708,7 @@ ssize_t ib_uverbs_poll_cq(struct ib_uverbs_file *file,
resp->wc[i].opcode = wc[i].opcode; resp->wc[i].opcode = wc[i].opcode;
resp->wc[i].vendor_err = wc[i].vendor_err; resp->wc[i].vendor_err = wc[i].vendor_err;
resp->wc[i].byte_len = wc[i].byte_len; resp->wc[i].byte_len = wc[i].byte_len;
resp->wc[i].imm_data = wc[i].imm_data; resp->wc[i].imm_data = (__u32 __force) wc[i].imm_data;
resp->wc[i].qp_num = wc[i].qp_num; resp->wc[i].qp_num = wc[i].qp_num;
resp->wc[i].src_qp = wc[i].src_qp; resp->wc[i].src_qp = wc[i].src_qp;
resp->wc[i].wc_flags = wc[i].wc_flags; resp->wc[i].wc_flags = wc[i].wc_flags;
...@@ -909,6 +909,11 @@ retry: ...@@ -909,6 +909,11 @@ retry:
goto err_destroy; goto err_destroy;
resp.qp_handle = uobj->uobject.id; resp.qp_handle = uobj->uobject.id;
resp.max_recv_sge = attr.cap.max_recv_sge;
resp.max_send_sge = attr.cap.max_send_sge;
resp.max_recv_wr = attr.cap.max_recv_wr;
resp.max_send_wr = attr.cap.max_send_wr;
resp.max_inline_data = attr.cap.max_inline_data;
if (copy_to_user((void __user *) (unsigned long) cmd.response, if (copy_to_user((void __user *) (unsigned long) cmd.response,
&resp, sizeof resp)) { &resp, sizeof resp)) {
...@@ -1135,7 +1140,7 @@ ssize_t ib_uverbs_post_send(struct ib_uverbs_file *file, ...@@ -1135,7 +1140,7 @@ ssize_t ib_uverbs_post_send(struct ib_uverbs_file *file,
next->num_sge = user_wr->num_sge; next->num_sge = user_wr->num_sge;
next->opcode = user_wr->opcode; next->opcode = user_wr->opcode;
next->send_flags = user_wr->send_flags; next->send_flags = user_wr->send_flags;
next->imm_data = user_wr->imm_data; next->imm_data = (__be32 __force) user_wr->imm_data;
if (qp->qp_type == IB_QPT_UD) { if (qp->qp_type == IB_QPT_UD) {
next->wr.ud.ah = idr_find(&ib_uverbs_ah_idr, next->wr.ud.ah = idr_find(&ib_uverbs_ah_idr,
...@@ -1701,7 +1706,6 @@ ssize_t ib_uverbs_modify_srq(struct ib_uverbs_file *file, ...@@ -1701,7 +1706,6 @@ ssize_t ib_uverbs_modify_srq(struct ib_uverbs_file *file,
} }
attr.max_wr = cmd.max_wr; attr.max_wr = cmd.max_wr;
attr.max_sge = cmd.max_sge;
attr.srq_limit = cmd.srq_limit; attr.srq_limit = cmd.srq_limit;
ret = ib_modify_srq(srq, &attr, cmd.attr_mask); ret = ib_modify_srq(srq, &attr, cmd.attr_mask);
......
...@@ -325,16 +325,8 @@ EXPORT_SYMBOL(ib_destroy_cq); ...@@ -325,16 +325,8 @@ EXPORT_SYMBOL(ib_destroy_cq);
int ib_resize_cq(struct ib_cq *cq, int ib_resize_cq(struct ib_cq *cq,
int cqe) int cqe)
{ {
int ret; return cq->device->resize_cq ?
cq->device->resize_cq(cq, cqe) : -ENOSYS;
if (!cq->device->resize_cq)
return -ENOSYS;
ret = cq->device->resize_cq(cq, &cqe);
if (!ret)
cq->cqe = cqe;
return ret;
} }
EXPORT_SYMBOL(ib_resize_cq); EXPORT_SYMBOL(ib_resize_cq);
......
...@@ -97,7 +97,7 @@ static void poll_catas(unsigned long dev_ptr) ...@@ -97,7 +97,7 @@ static void poll_catas(unsigned long dev_ptr)
} }
spin_lock_irqsave(&catas_lock, flags); spin_lock_irqsave(&catas_lock, flags);
if (dev->catas_err.stop) if (!dev->catas_err.stop)
mod_timer(&dev->catas_err.timer, mod_timer(&dev->catas_err.timer,
jiffies + MTHCA_CATAS_POLL_INTERVAL); jiffies + MTHCA_CATAS_POLL_INTERVAL);
spin_unlock_irqrestore(&catas_lock, flags); spin_unlock_irqrestore(&catas_lock, flags);
......
...@@ -1060,6 +1060,8 @@ int mthca_QUERY_DEV_LIM(struct mthca_dev *dev, ...@@ -1060,6 +1060,8 @@ int mthca_QUERY_DEV_LIM(struct mthca_dev *dev,
dev_lim->hca.arbel.resize_srq = field & 1; dev_lim->hca.arbel.resize_srq = field & 1;
MTHCA_GET(field, outbox, QUERY_DEV_LIM_MAX_SG_RQ_OFFSET); MTHCA_GET(field, outbox, QUERY_DEV_LIM_MAX_SG_RQ_OFFSET);
dev_lim->max_sg = min_t(int, field, dev_lim->max_sg); dev_lim->max_sg = min_t(int, field, dev_lim->max_sg);
MTHCA_GET(size, outbox, QUERY_DEV_LIM_MAX_DESC_SZ_RQ_OFFSET);
dev_lim->max_desc_sz = min_t(int, size, dev_lim->max_desc_sz);
MTHCA_GET(size, outbox, QUERY_DEV_LIM_MPT_ENTRY_SZ_OFFSET); MTHCA_GET(size, outbox, QUERY_DEV_LIM_MPT_ENTRY_SZ_OFFSET);
dev_lim->mpt_entry_sz = size; dev_lim->mpt_entry_sz = size;
MTHCA_GET(field, outbox, QUERY_DEV_LIM_PBL_SZ_OFFSET); MTHCA_GET(field, outbox, QUERY_DEV_LIM_PBL_SZ_OFFSET);
......
...@@ -258,7 +258,7 @@ void mthca_cq_clean(struct mthca_dev *dev, u32 cqn, u32 qpn, ...@@ -258,7 +258,7 @@ void mthca_cq_clean(struct mthca_dev *dev, u32 cqn, u32 qpn,
{ {
struct mthca_cq *cq; struct mthca_cq *cq;
struct mthca_cqe *cqe; struct mthca_cqe *cqe;
int prod_index; u32 prod_index;
int nfreed = 0; int nfreed = 0;
spin_lock_irq(&dev->cq_table.lock); spin_lock_irq(&dev->cq_table.lock);
...@@ -293,19 +293,15 @@ void mthca_cq_clean(struct mthca_dev *dev, u32 cqn, u32 qpn, ...@@ -293,19 +293,15 @@ void mthca_cq_clean(struct mthca_dev *dev, u32 cqn, u32 qpn,
* Now sweep backwards through the CQ, removing CQ entries * Now sweep backwards through the CQ, removing CQ entries
* that match our QP by copying older entries on top of them. * that match our QP by copying older entries on top of them.
*/ */
while (prod_index > cq->cons_index) { while ((int) --prod_index - (int) cq->cons_index >= 0) {
cqe = get_cqe(cq, (prod_index - 1) & cq->ibcq.cqe); cqe = get_cqe(cq, prod_index & cq->ibcq.cqe);
if (cqe->my_qpn == cpu_to_be32(qpn)) { if (cqe->my_qpn == cpu_to_be32(qpn)) {
if (srq) if (srq)
mthca_free_srq_wqe(srq, be32_to_cpu(cqe->wqe)); mthca_free_srq_wqe(srq, be32_to_cpu(cqe->wqe));
++nfreed; ++nfreed;
} } else if (nfreed)
else if (nfreed) memcpy(get_cqe(cq, (prod_index + nfreed) & cq->ibcq.cqe),
memcpy(get_cqe(cq, (prod_index - 1 + nfreed) & cqe, MTHCA_CQ_ENTRY_SIZE);
cq->ibcq.cqe),
cqe,
MTHCA_CQ_ENTRY_SIZE);
--prod_index;
} }
if (nfreed) { if (nfreed) {
......
...@@ -131,6 +131,7 @@ struct mthca_limits { ...@@ -131,6 +131,7 @@ struct mthca_limits {
int max_sg; int max_sg;
int num_qps; int num_qps;
int max_wqes; int max_wqes;
int max_desc_sz;
int max_qp_init_rdma; int max_qp_init_rdma;
int reserved_qps; int reserved_qps;
int num_srqs; int num_srqs;
...@@ -154,6 +155,7 @@ struct mthca_limits { ...@@ -154,6 +155,7 @@ struct mthca_limits {
int reserved_mcgs; int reserved_mcgs;
int num_pds; int num_pds;
int reserved_pds; int reserved_pds;
u32 page_size_cap;
u32 flags; u32 flags;
u8 port_width_cap; u8 port_width_cap;
}; };
......
...@@ -168,6 +168,7 @@ static int __devinit mthca_dev_lim(struct mthca_dev *mdev, struct mthca_dev_lim ...@@ -168,6 +168,7 @@ static int __devinit mthca_dev_lim(struct mthca_dev *mdev, struct mthca_dev_lim
mdev->limits.max_srq_wqes = dev_lim->max_srq_sz; mdev->limits.max_srq_wqes = dev_lim->max_srq_sz;
mdev->limits.reserved_srqs = dev_lim->reserved_srqs; mdev->limits.reserved_srqs = dev_lim->reserved_srqs;
mdev->limits.reserved_eecs = dev_lim->reserved_eecs; mdev->limits.reserved_eecs = dev_lim->reserved_eecs;
mdev->limits.max_desc_sz = dev_lim->max_desc_sz;
/* /*
* Subtract 1 from the limit because we need to allocate a * Subtract 1 from the limit because we need to allocate a
* spare CQE so the HCA HW can tell the difference between an * spare CQE so the HCA HW can tell the difference between an
...@@ -181,6 +182,7 @@ static int __devinit mthca_dev_lim(struct mthca_dev *mdev, struct mthca_dev_lim ...@@ -181,6 +182,7 @@ static int __devinit mthca_dev_lim(struct mthca_dev *mdev, struct mthca_dev_lim
mdev->limits.reserved_uars = dev_lim->reserved_uars; mdev->limits.reserved_uars = dev_lim->reserved_uars;
mdev->limits.reserved_pds = dev_lim->reserved_pds; mdev->limits.reserved_pds = dev_lim->reserved_pds;
mdev->limits.port_width_cap = dev_lim->max_port_width; mdev->limits.port_width_cap = dev_lim->max_port_width;
mdev->limits.page_size_cap = ~(u32) (dev_lim->min_page_sz - 1);
mdev->limits.flags = dev_lim->flags; mdev->limits.flags = dev_lim->flags;
/* IB_DEVICE_RESIZE_MAX_WR not supported by driver. /* IB_DEVICE_RESIZE_MAX_WR not supported by driver.
......
...@@ -90,6 +90,7 @@ static int mthca_query_device(struct ib_device *ibdev, ...@@ -90,6 +90,7 @@ static int mthca_query_device(struct ib_device *ibdev,
memcpy(&props->node_guid, out_mad->data + 12, 8); memcpy(&props->node_guid, out_mad->data + 12, 8);
props->max_mr_size = ~0ull; props->max_mr_size = ~0ull;
props->page_size_cap = mdev->limits.page_size_cap;
props->max_qp = mdev->limits.num_qps - mdev->limits.reserved_qps; props->max_qp = mdev->limits.num_qps - mdev->limits.reserved_qps;
props->max_qp_wr = mdev->limits.max_wqes; props->max_qp_wr = mdev->limits.max_wqes;
props->max_sge = mdev->limits.max_sg; props->max_sge = mdev->limits.max_sg;
...@@ -615,11 +616,11 @@ static struct ib_qp *mthca_create_qp(struct ib_pd *pd, ...@@ -615,11 +616,11 @@ static struct ib_qp *mthca_create_qp(struct ib_pd *pd,
return ERR_PTR(err); return ERR_PTR(err);
} }
init_attr->cap.max_inline_data = 0;
init_attr->cap.max_send_wr = qp->sq.max; init_attr->cap.max_send_wr = qp->sq.max;
init_attr->cap.max_recv_wr = qp->rq.max; init_attr->cap.max_recv_wr = qp->rq.max;
init_attr->cap.max_send_sge = qp->sq.max_gs; init_attr->cap.max_send_sge = qp->sq.max_gs;
init_attr->cap.max_recv_sge = qp->rq.max_gs; init_attr->cap.max_recv_sge = qp->rq.max_gs;
init_attr->cap.max_inline_data = qp->max_inline_data;
return &qp->ibqp; return &qp->ibqp;
} }
......
...@@ -251,6 +251,7 @@ struct mthca_qp { ...@@ -251,6 +251,7 @@ struct mthca_qp {
struct mthca_wq sq; struct mthca_wq sq;
enum ib_sig_type sq_policy; enum ib_sig_type sq_policy;
int send_wqe_offset; int send_wqe_offset;
int max_inline_data;
u64 *wrid; u64 *wrid;
union mthca_buf queue; union mthca_buf queue;
......
...@@ -885,6 +885,48 @@ int mthca_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, int attr_mask) ...@@ -885,6 +885,48 @@ int mthca_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, int attr_mask)
return err; return err;
} }
static void mthca_adjust_qp_caps(struct mthca_dev *dev,
struct mthca_pd *pd,
struct mthca_qp *qp)
{
int max_data_size;
/*
* Calculate the maximum size of WQE s/g segments, excluding
* the next segment and other non-data segments.
*/
max_data_size = min(dev->limits.max_desc_sz, 1 << qp->sq.wqe_shift) -
sizeof (struct mthca_next_seg);
switch (qp->transport) {
case MLX:
max_data_size -= 2 * sizeof (struct mthca_data_seg);
break;
case UD:
if (mthca_is_memfree(dev))
max_data_size -= sizeof (struct mthca_arbel_ud_seg);
else
max_data_size -= sizeof (struct mthca_tavor_ud_seg);
break;
default:
max_data_size -= sizeof (struct mthca_raddr_seg);
break;
}
/* We don't support inline data for kernel QPs (yet). */
if (!pd->ibpd.uobject)
qp->max_inline_data = 0;
else
qp->max_inline_data = max_data_size - MTHCA_INLINE_HEADER_SIZE;
qp->sq.max_gs = max_data_size / sizeof (struct mthca_data_seg);
qp->rq.max_gs = (min(dev->limits.max_desc_sz, 1 << qp->rq.wqe_shift) -
sizeof (struct mthca_next_seg)) /
sizeof (struct mthca_data_seg);
}
/* /*
* Allocate and register buffer for WQEs. qp->rq.max, sq.max, * Allocate and register buffer for WQEs. qp->rq.max, sq.max,
* rq.max_gs and sq.max_gs must all be assigned. * rq.max_gs and sq.max_gs must all be assigned.
...@@ -902,27 +944,53 @@ static int mthca_alloc_wqe_buf(struct mthca_dev *dev, ...@@ -902,27 +944,53 @@ static int mthca_alloc_wqe_buf(struct mthca_dev *dev,
size = sizeof (struct mthca_next_seg) + size = sizeof (struct mthca_next_seg) +
qp->rq.max_gs * sizeof (struct mthca_data_seg); qp->rq.max_gs * sizeof (struct mthca_data_seg);
if (size > dev->limits.max_desc_sz)
return -EINVAL;
for (qp->rq.wqe_shift = 6; 1 << qp->rq.wqe_shift < size; for (qp->rq.wqe_shift = 6; 1 << qp->rq.wqe_shift < size;
qp->rq.wqe_shift++) qp->rq.wqe_shift++)
; /* nothing */ ; /* nothing */
size = sizeof (struct mthca_next_seg) + size = qp->sq.max_gs * sizeof (struct mthca_data_seg);
qp->sq.max_gs * sizeof (struct mthca_data_seg);
switch (qp->transport) { switch (qp->transport) {
case MLX: case MLX:
size += 2 * sizeof (struct mthca_data_seg); size += 2 * sizeof (struct mthca_data_seg);
break; break;
case UD: case UD:
if (mthca_is_memfree(dev)) size += mthca_is_memfree(dev) ?
size += sizeof (struct mthca_arbel_ud_seg); sizeof (struct mthca_arbel_ud_seg) :
else sizeof (struct mthca_tavor_ud_seg);
size += sizeof (struct mthca_tavor_ud_seg);
break; break;
case UC:
size += sizeof (struct mthca_raddr_seg);
break;
case RC:
size += sizeof (struct mthca_raddr_seg);
/*
* An atomic op will require an atomic segment, a
* remote address segment and one scatter entry.
*/
size = max_t(int, size,
sizeof (struct mthca_atomic_seg) +
sizeof (struct mthca_raddr_seg) +
sizeof (struct mthca_data_seg));
break;
default: default:
/* bind seg is as big as atomic + raddr segs */ break;
size += sizeof (struct mthca_bind_seg);
} }
/* Make sure that we have enough space for a bind request */
size = max_t(int, size, sizeof (struct mthca_bind_seg));
size += sizeof (struct mthca_next_seg);
if (size > dev->limits.max_desc_sz)
return -EINVAL;
for (qp->sq.wqe_shift = 6; 1 << qp->sq.wqe_shift < size; for (qp->sq.wqe_shift = 6; 1 << qp->sq.wqe_shift < size;
qp->sq.wqe_shift++) qp->sq.wqe_shift++)
; /* nothing */ ; /* nothing */
...@@ -1066,6 +1134,8 @@ static int mthca_alloc_qp_common(struct mthca_dev *dev, ...@@ -1066,6 +1134,8 @@ static int mthca_alloc_qp_common(struct mthca_dev *dev,
return ret; return ret;
} }
mthca_adjust_qp_caps(dev, pd, qp);
/* /*
* If this is a userspace QP, we're done now. The doorbells * If this is a userspace QP, we're done now. The doorbells
* will be allocated and buffers will be initialized in * will be allocated and buffers will be initialized in
...@@ -1486,8 +1556,8 @@ int mthca_tavor_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr, ...@@ -1486,8 +1556,8 @@ int mthca_tavor_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
} }
wqe += sizeof (struct mthca_atomic_seg); wqe += sizeof (struct mthca_atomic_seg);
size += sizeof (struct mthca_raddr_seg) / 16 + size += (sizeof (struct mthca_raddr_seg) +
sizeof (struct mthca_atomic_seg); sizeof (struct mthca_atomic_seg)) / 16;
break; break;
case IB_WR_RDMA_WRITE: case IB_WR_RDMA_WRITE:
...@@ -1637,6 +1707,7 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr, ...@@ -1637,6 +1707,7 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
{ {
struct mthca_dev *dev = to_mdev(ibqp->device); struct mthca_dev *dev = to_mdev(ibqp->device);
struct mthca_qp *qp = to_mqp(ibqp); struct mthca_qp *qp = to_mqp(ibqp);
__be32 doorbell[2];
unsigned long flags; unsigned long flags;
int err = 0; int err = 0;
int nreq; int nreq;
...@@ -1654,6 +1725,22 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr, ...@@ -1654,6 +1725,22 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
ind = qp->rq.next_ind; ind = qp->rq.next_ind;
for (nreq = 0; wr; ++nreq, wr = wr->next) { for (nreq = 0; wr; ++nreq, wr = wr->next) {
if (unlikely(nreq == MTHCA_TAVOR_MAX_WQES_PER_RECV_DB)) {
nreq = 0;
doorbell[0] = cpu_to_be32((qp->rq.next_ind << qp->rq.wqe_shift) | size0);
doorbell[1] = cpu_to_be32(qp->qpn << 8);
wmb();
mthca_write64(doorbell,
dev->kar + MTHCA_RECEIVE_DOORBELL,
MTHCA_GET_DOORBELL_LOCK(&dev->doorbell_lock));
qp->rq.head += MTHCA_TAVOR_MAX_WQES_PER_RECV_DB;
size0 = 0;
}
if (mthca_wq_overflow(&qp->rq, nreq, qp->ibqp.recv_cq)) { if (mthca_wq_overflow(&qp->rq, nreq, qp->ibqp.recv_cq)) {
mthca_err(dev, "RQ %06x full (%u head, %u tail," mthca_err(dev, "RQ %06x full (%u head, %u tail,"
" %d max, %d nreq)\n", qp->qpn, " %d max, %d nreq)\n", qp->qpn,
...@@ -1711,8 +1798,6 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr, ...@@ -1711,8 +1798,6 @@ int mthca_tavor_post_receive(struct ib_qp *ibqp, struct ib_recv_wr *wr,
out: out:
if (likely(nreq)) { if (likely(nreq)) {
__be32 doorbell[2];
doorbell[0] = cpu_to_be32((qp->rq.next_ind << qp->rq.wqe_shift) | size0); doorbell[0] = cpu_to_be32((qp->rq.next_ind << qp->rq.wqe_shift) | size0);
doorbell[1] = cpu_to_be32((qp->qpn << 8) | nreq); doorbell[1] = cpu_to_be32((qp->qpn << 8) | nreq);
...@@ -1806,8 +1891,8 @@ int mthca_arbel_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr, ...@@ -1806,8 +1891,8 @@ int mthca_arbel_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr,
} }
wqe += sizeof (struct mthca_atomic_seg); wqe += sizeof (struct mthca_atomic_seg);
size += sizeof (struct mthca_raddr_seg) / 16 + size += (sizeof (struct mthca_raddr_seg) +
sizeof (struct mthca_atomic_seg); sizeof (struct mthca_atomic_seg)) / 16;
break; break;
case IB_WR_RDMA_READ: case IB_WR_RDMA_READ:
......
...@@ -417,6 +417,7 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr, ...@@ -417,6 +417,7 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr,
{ {
struct mthca_dev *dev = to_mdev(ibsrq->device); struct mthca_dev *dev = to_mdev(ibsrq->device);
struct mthca_srq *srq = to_msrq(ibsrq); struct mthca_srq *srq = to_msrq(ibsrq);
__be32 doorbell[2];
unsigned long flags; unsigned long flags;
int err = 0; int err = 0;
int first_ind; int first_ind;
...@@ -432,6 +433,25 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr, ...@@ -432,6 +433,25 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr,
first_ind = srq->first_free; first_ind = srq->first_free;
for (nreq = 0; wr; ++nreq, wr = wr->next) { for (nreq = 0; wr; ++nreq, wr = wr->next) {
if (unlikely(nreq == MTHCA_TAVOR_MAX_WQES_PER_RECV_DB)) {
nreq = 0;
doorbell[0] = cpu_to_be32(first_ind << srq->wqe_shift);
doorbell[1] = cpu_to_be32(srq->srqn << 8);
/*
* Make sure that descriptors are written
* before doorbell is rung.
*/
wmb();
mthca_write64(doorbell,
dev->kar + MTHCA_RECEIVE_DOORBELL,
MTHCA_GET_DOORBELL_LOCK(&dev->doorbell_lock));
first_ind = srq->first_free;
}
ind = srq->first_free; ind = srq->first_free;
if (ind < 0) { if (ind < 0) {
...@@ -494,8 +514,6 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr, ...@@ -494,8 +514,6 @@ int mthca_tavor_post_srq_recv(struct ib_srq *ibsrq, struct ib_recv_wr *wr,
} }
if (likely(nreq)) { if (likely(nreq)) {
__be32 doorbell[2];
doorbell[0] = cpu_to_be32(first_ind << srq->wqe_shift); doorbell[0] = cpu_to_be32(first_ind << srq->wqe_shift);
doorbell[1] = cpu_to_be32((srq->srqn << 8) | nreq); doorbell[1] = cpu_to_be32((srq->srqn << 8) | nreq);
......
...@@ -49,7 +49,8 @@ enum { ...@@ -49,7 +49,8 @@ enum {
}; };
enum { enum {
MTHCA_INVAL_LKEY = 0x100 MTHCA_INVAL_LKEY = 0x100,
MTHCA_TAVOR_MAX_WQES_PER_RECV_DB = 256
}; };
struct mthca_next_seg { struct mthca_next_seg {
......
...@@ -179,6 +179,7 @@ struct ipoib_dev_priv { ...@@ -179,6 +179,7 @@ struct ipoib_dev_priv {
#ifdef CONFIG_INFINIBAND_IPOIB_DEBUG #ifdef CONFIG_INFINIBAND_IPOIB_DEBUG
struct list_head fs_list; struct list_head fs_list;
struct dentry *mcg_dentry; struct dentry *mcg_dentry;
struct dentry *path_dentry;
#endif #endif
}; };
...@@ -270,7 +271,6 @@ void ipoib_mcast_dev_flush(struct net_device *dev); ...@@ -270,7 +271,6 @@ void ipoib_mcast_dev_flush(struct net_device *dev);
#ifdef CONFIG_INFINIBAND_IPOIB_DEBUG #ifdef CONFIG_INFINIBAND_IPOIB_DEBUG
struct ipoib_mcast_iter *ipoib_mcast_iter_init(struct net_device *dev); struct ipoib_mcast_iter *ipoib_mcast_iter_init(struct net_device *dev);
void ipoib_mcast_iter_free(struct ipoib_mcast_iter *iter);
int ipoib_mcast_iter_next(struct ipoib_mcast_iter *iter); int ipoib_mcast_iter_next(struct ipoib_mcast_iter *iter);
void ipoib_mcast_iter_read(struct ipoib_mcast_iter *iter, void ipoib_mcast_iter_read(struct ipoib_mcast_iter *iter,
union ib_gid *gid, union ib_gid *gid,
...@@ -278,6 +278,11 @@ void ipoib_mcast_iter_read(struct ipoib_mcast_iter *iter, ...@@ -278,6 +278,11 @@ void ipoib_mcast_iter_read(struct ipoib_mcast_iter *iter,
unsigned int *queuelen, unsigned int *queuelen,
unsigned int *complete, unsigned int *complete,
unsigned int *send_only); unsigned int *send_only);
struct ipoib_path_iter *ipoib_path_iter_init(struct net_device *dev);
int ipoib_path_iter_next(struct ipoib_path_iter *iter);
void ipoib_path_iter_read(struct ipoib_path_iter *iter,
struct ipoib_path *path);
#endif #endif
int ipoib_mcast_attach(struct net_device *dev, u16 mlid, int ipoib_mcast_attach(struct net_device *dev, u16 mlid,
...@@ -299,13 +304,13 @@ void ipoib_pkey_poll(void *dev); ...@@ -299,13 +304,13 @@ void ipoib_pkey_poll(void *dev);
int ipoib_pkey_dev_delay_open(struct net_device *dev); int ipoib_pkey_dev_delay_open(struct net_device *dev);
#ifdef CONFIG_INFINIBAND_IPOIB_DEBUG #ifdef CONFIG_INFINIBAND_IPOIB_DEBUG
int ipoib_create_debug_file(struct net_device *dev); void ipoib_create_debug_files(struct net_device *dev);
void ipoib_delete_debug_file(struct net_device *dev); void ipoib_delete_debug_files(struct net_device *dev);
int ipoib_register_debugfs(void); int ipoib_register_debugfs(void);
void ipoib_unregister_debugfs(void); void ipoib_unregister_debugfs(void);
#else #else
static inline int ipoib_create_debug_file(struct net_device *dev) { return 0; } static inline void ipoib_create_debug_files(struct net_device *dev) { }
static inline void ipoib_delete_debug_file(struct net_device *dev) { } static inline void ipoib_delete_debug_files(struct net_device *dev) { }
static inline int ipoib_register_debugfs(void) { return 0; } static inline int ipoib_register_debugfs(void) { return 0; }
static inline void ipoib_unregister_debugfs(void) { } static inline void ipoib_unregister_debugfs(void) { }
#endif #endif
......
...@@ -43,6 +43,18 @@ struct file_operations; ...@@ -43,6 +43,18 @@ struct file_operations;
static struct dentry *ipoib_root; static struct dentry *ipoib_root;
static void format_gid(union ib_gid *gid, char *buf)
{
int i, n;
for (n = 0, i = 0; i < 8; ++i) {
n += sprintf(buf + n, "%x",
be16_to_cpu(((__be16 *) gid->raw)[i]));
if (i < 7)
buf[n++] = ':';
}
}
static void *ipoib_mcg_seq_start(struct seq_file *file, loff_t *pos) static void *ipoib_mcg_seq_start(struct seq_file *file, loff_t *pos)
{ {
struct ipoib_mcast_iter *iter; struct ipoib_mcast_iter *iter;
...@@ -54,7 +66,7 @@ static void *ipoib_mcg_seq_start(struct seq_file *file, loff_t *pos) ...@@ -54,7 +66,7 @@ static void *ipoib_mcg_seq_start(struct seq_file *file, loff_t *pos)
while (n--) { while (n--) {
if (ipoib_mcast_iter_next(iter)) { if (ipoib_mcast_iter_next(iter)) {
ipoib_mcast_iter_free(iter); kfree(iter);
return NULL; return NULL;
} }
} }
...@@ -70,7 +82,7 @@ static void *ipoib_mcg_seq_next(struct seq_file *file, void *iter_ptr, ...@@ -70,7 +82,7 @@ static void *ipoib_mcg_seq_next(struct seq_file *file, void *iter_ptr,
(*pos)++; (*pos)++;
if (ipoib_mcast_iter_next(iter)) { if (ipoib_mcast_iter_next(iter)) {
ipoib_mcast_iter_free(iter); kfree(iter);
return NULL; return NULL;
} }
...@@ -87,32 +99,32 @@ static int ipoib_mcg_seq_show(struct seq_file *file, void *iter_ptr) ...@@ -87,32 +99,32 @@ static int ipoib_mcg_seq_show(struct seq_file *file, void *iter_ptr)
struct ipoib_mcast_iter *iter = iter_ptr; struct ipoib_mcast_iter *iter = iter_ptr;
char gid_buf[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff"]; char gid_buf[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff"];
union ib_gid mgid; union ib_gid mgid;
int i, n;
unsigned long created; unsigned long created;
unsigned int queuelen, complete, send_only; unsigned int queuelen, complete, send_only;
if (iter) { if (!iter)
return 0;
ipoib_mcast_iter_read(iter, &mgid, &created, &queuelen, ipoib_mcast_iter_read(iter, &mgid, &created, &queuelen,
&complete, &send_only); &complete, &send_only);
for (n = 0, i = 0; i < sizeof mgid / 2; ++i) { format_gid(&mgid, gid_buf);
n += sprintf(gid_buf + n, "%x",
be16_to_cpu(((__be16 *) mgid.raw)[i]));
if (i < sizeof mgid / 2 - 1)
gid_buf[n++] = ':';
}
}
seq_printf(file, "GID: %*s", -(1 + (int) sizeof gid_buf), gid_buf);
seq_printf(file, seq_printf(file,
" created: %10ld queuelen: %4d complete: %d send_only: %d\n", "GID: %s\n"
created, queuelen, complete, send_only); " created: %10ld\n"
" queuelen: %9d\n"
" complete: %9s\n"
" send_only: %8s\n"
"\n",
gid_buf, created, queuelen,
complete ? "yes" : "no",
send_only ? "yes" : "no");
return 0; return 0;
} }
static struct seq_operations ipoib_seq_ops = { static struct seq_operations ipoib_mcg_seq_ops = {
.start = ipoib_mcg_seq_start, .start = ipoib_mcg_seq_start,
.next = ipoib_mcg_seq_next, .next = ipoib_mcg_seq_next,
.stop = ipoib_mcg_seq_stop, .stop = ipoib_mcg_seq_stop,
...@@ -124,7 +136,7 @@ static int ipoib_mcg_open(struct inode *inode, struct file *file) ...@@ -124,7 +136,7 @@ static int ipoib_mcg_open(struct inode *inode, struct file *file)
struct seq_file *seq; struct seq_file *seq;
int ret; int ret;
ret = seq_open(file, &ipoib_seq_ops); ret = seq_open(file, &ipoib_mcg_seq_ops);
if (ret) if (ret)
return ret; return ret;
...@@ -134,7 +146,7 @@ static int ipoib_mcg_open(struct inode *inode, struct file *file) ...@@ -134,7 +146,7 @@ static int ipoib_mcg_open(struct inode *inode, struct file *file)
return 0; return 0;
} }
static struct file_operations ipoib_fops = { static struct file_operations ipoib_mcg_fops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.open = ipoib_mcg_open, .open = ipoib_mcg_open,
.read = seq_read, .read = seq_read,
...@@ -142,25 +154,138 @@ static struct file_operations ipoib_fops = { ...@@ -142,25 +154,138 @@ static struct file_operations ipoib_fops = {
.release = seq_release .release = seq_release
}; };
int ipoib_create_debug_file(struct net_device *dev) static void *ipoib_path_seq_start(struct seq_file *file, loff_t *pos)
{
struct ipoib_path_iter *iter;
loff_t n = *pos;
iter = ipoib_path_iter_init(file->private);
if (!iter)
return NULL;
while (n--) {
if (ipoib_path_iter_next(iter)) {
kfree(iter);
return NULL;
}
}
return iter;
}
static void *ipoib_path_seq_next(struct seq_file *file, void *iter_ptr,
loff_t *pos)
{
struct ipoib_path_iter *iter = iter_ptr;
(*pos)++;
if (ipoib_path_iter_next(iter)) {
kfree(iter);
return NULL;
}
return iter;
}
static void ipoib_path_seq_stop(struct seq_file *file, void *iter_ptr)
{
/* nothing for now */
}
static int ipoib_path_seq_show(struct seq_file *file, void *iter_ptr)
{
struct ipoib_path_iter *iter = iter_ptr;
char gid_buf[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff"];
struct ipoib_path path;
int rate;
if (!iter)
return 0;
ipoib_path_iter_read(iter, &path);
format_gid(&path.pathrec.dgid, gid_buf);
seq_printf(file,
"GID: %s\n"
" complete: %6s\n",
gid_buf, path.pathrec.dlid ? "yes" : "no");
if (path.pathrec.dlid) {
rate = ib_sa_rate_enum_to_int(path.pathrec.rate) * 25;
seq_printf(file,
" DLID: 0x%04x\n"
" SL: %12d\n"
" rate: %*d%s Gb/sec\n",
be16_to_cpu(path.pathrec.dlid),
path.pathrec.sl,
10 - ((rate % 10) ? 2 : 0),
rate / 10, rate % 10 ? ".5" : "");
}
seq_putc(file, '\n');
return 0;
}
static struct seq_operations ipoib_path_seq_ops = {
.start = ipoib_path_seq_start,
.next = ipoib_path_seq_next,
.stop = ipoib_path_seq_stop,
.show = ipoib_path_seq_show,
};
static int ipoib_path_open(struct inode *inode, struct file *file)
{
struct seq_file *seq;
int ret;
ret = seq_open(file, &ipoib_path_seq_ops);
if (ret)
return ret;
seq = file->private_data;
seq->private = inode->u.generic_ip;
return 0;
}
static struct file_operations ipoib_path_fops = {
.owner = THIS_MODULE,
.open = ipoib_path_open,
.read = seq_read,
.llseek = seq_lseek,
.release = seq_release
};
void ipoib_create_debug_files(struct net_device *dev)
{ {
struct ipoib_dev_priv *priv = netdev_priv(dev); struct ipoib_dev_priv *priv = netdev_priv(dev);
char name[IFNAMSIZ + sizeof "_mcg"]; char name[IFNAMSIZ + sizeof "_path"];
snprintf(name, sizeof name, "%s_mcg", dev->name); snprintf(name, sizeof name, "%s_mcg", dev->name);
priv->mcg_dentry = debugfs_create_file(name, S_IFREG | S_IRUGO, priv->mcg_dentry = debugfs_create_file(name, S_IFREG | S_IRUGO,
ipoib_root, dev, &ipoib_fops); ipoib_root, dev, &ipoib_mcg_fops);
if (!priv->mcg_dentry)
return priv->mcg_dentry ? 0 : -ENOMEM; ipoib_warn(priv, "failed to create mcg debug file\n");
snprintf(name, sizeof name, "%s_path", dev->name);
priv->path_dentry = debugfs_create_file(name, S_IFREG | S_IRUGO,
ipoib_root, dev, &ipoib_path_fops);
if (!priv->path_dentry)
ipoib_warn(priv, "failed to create path debug file\n");
} }
void ipoib_delete_debug_file(struct net_device *dev) void ipoib_delete_debug_files(struct net_device *dev)
{ {
struct ipoib_dev_priv *priv = netdev_priv(dev); struct ipoib_dev_priv *priv = netdev_priv(dev);
if (priv->mcg_dentry) if (priv->mcg_dentry)
debugfs_remove(priv->mcg_dentry); debugfs_remove(priv->mcg_dentry);
if (priv->path_dentry)
debugfs_remove(priv->path_dentry);
} }
int ipoib_register_debugfs(void) int ipoib_register_debugfs(void)
......
...@@ -58,6 +58,11 @@ module_param_named(debug_level, ipoib_debug_level, int, 0644); ...@@ -58,6 +58,11 @@ module_param_named(debug_level, ipoib_debug_level, int, 0644);
MODULE_PARM_DESC(debug_level, "Enable debug tracing if > 0"); MODULE_PARM_DESC(debug_level, "Enable debug tracing if > 0");
#endif #endif
struct ipoib_path_iter {
struct net_device *dev;
struct ipoib_path path;
};
static const u8 ipv4_bcast_addr[] = { static const u8 ipv4_bcast_addr[] = {
0x00, 0xff, 0xff, 0xff, 0x00, 0xff, 0xff, 0xff,
0xff, 0x12, 0x40, 0x1b, 0x00, 0x00, 0x00, 0x00, 0xff, 0x12, 0x40, 0x1b, 0x00, 0x00, 0x00, 0x00,
...@@ -250,6 +255,64 @@ static void path_free(struct net_device *dev, struct ipoib_path *path) ...@@ -250,6 +255,64 @@ static void path_free(struct net_device *dev, struct ipoib_path *path)
kfree(path); kfree(path);
} }
#ifdef CONFIG_INFINIBAND_IPOIB_DEBUG
struct ipoib_path_iter *ipoib_path_iter_init(struct net_device *dev)
{
struct ipoib_path_iter *iter;
iter = kmalloc(sizeof *iter, GFP_KERNEL);
if (!iter)
return NULL;
iter->dev = dev;
memset(iter->path.pathrec.dgid.raw, 0, 16);
if (ipoib_path_iter_next(iter)) {
kfree(iter);
return NULL;
}
return iter;
}
int ipoib_path_iter_next(struct ipoib_path_iter *iter)
{
struct ipoib_dev_priv *priv = netdev_priv(iter->dev);
struct rb_node *n;
struct ipoib_path *path;
int ret = 1;
spin_lock_irq(&priv->lock);
n = rb_first(&priv->path_tree);
while (n) {
path = rb_entry(n, struct ipoib_path, rb_node);
if (memcmp(iter->path.pathrec.dgid.raw, path->pathrec.dgid.raw,
sizeof (union ib_gid)) < 0) {
iter->path = *path;
ret = 0;
break;
}
n = rb_next(n);
}
spin_unlock_irq(&priv->lock);
return ret;
}
void ipoib_path_iter_read(struct ipoib_path_iter *iter,
struct ipoib_path *path)
{
*path = iter->path;
}
#endif /* CONFIG_INFINIBAND_IPOIB_DEBUG */
void ipoib_flush_paths(struct net_device *dev) void ipoib_flush_paths(struct net_device *dev)
{ {
struct ipoib_dev_priv *priv = netdev_priv(dev); struct ipoib_dev_priv *priv = netdev_priv(dev);
...@@ -763,7 +826,7 @@ void ipoib_dev_cleanup(struct net_device *dev) ...@@ -763,7 +826,7 @@ void ipoib_dev_cleanup(struct net_device *dev)
{ {
struct ipoib_dev_priv *priv = netdev_priv(dev), *cpriv, *tcpriv; struct ipoib_dev_priv *priv = netdev_priv(dev), *cpriv, *tcpriv;
ipoib_delete_debug_file(dev); ipoib_delete_debug_files(dev);
/* Delete any child interfaces first */ /* Delete any child interfaces first */
list_for_each_entry_safe(cpriv, tcpriv, &priv->child_intfs, list) { list_for_each_entry_safe(cpriv, tcpriv, &priv->child_intfs, list) {
...@@ -972,8 +1035,7 @@ static struct net_device *ipoib_add_port(const char *format, ...@@ -972,8 +1035,7 @@ static struct net_device *ipoib_add_port(const char *format,
goto register_failed; goto register_failed;
} }
if (ipoib_create_debug_file(priv->dev)) ipoib_create_debug_files(priv->dev);
goto debug_failed;
if (ipoib_add_pkey_attr(priv->dev)) if (ipoib_add_pkey_attr(priv->dev))
goto sysfs_failed; goto sysfs_failed;
...@@ -987,9 +1049,7 @@ static struct net_device *ipoib_add_port(const char *format, ...@@ -987,9 +1049,7 @@ static struct net_device *ipoib_add_port(const char *format,
return priv->dev; return priv->dev;
sysfs_failed: sysfs_failed:
ipoib_delete_debug_file(priv->dev); ipoib_delete_debug_files(priv->dev);
debug_failed:
unregister_netdev(priv->dev); unregister_netdev(priv->dev);
register_failed: register_failed:
......
...@@ -120,12 +120,8 @@ static void ipoib_mcast_free(struct ipoib_mcast *mcast) ...@@ -120,12 +120,8 @@ static void ipoib_mcast_free(struct ipoib_mcast *mcast)
if (mcast->ah) if (mcast->ah)
ipoib_put_ah(mcast->ah); ipoib_put_ah(mcast->ah);
while (!skb_queue_empty(&mcast->pkt_queue)) { while (!skb_queue_empty(&mcast->pkt_queue))
struct sk_buff *skb = skb_dequeue(&mcast->pkt_queue); dev_kfree_skb_any(skb_dequeue(&mcast->pkt_queue));
skb->dev = dev;
dev_kfree_skb_any(skb);
}
kfree(mcast); kfree(mcast);
} }
...@@ -317,13 +313,8 @@ ipoib_mcast_sendonly_join_complete(int status, ...@@ -317,13 +313,8 @@ ipoib_mcast_sendonly_join_complete(int status,
IPOIB_GID_ARG(mcast->mcmember.mgid), status); IPOIB_GID_ARG(mcast->mcmember.mgid), status);
/* Flush out any queued packets */ /* Flush out any queued packets */
while (!skb_queue_empty(&mcast->pkt_queue)) { while (!skb_queue_empty(&mcast->pkt_queue))
struct sk_buff *skb = skb_dequeue(&mcast->pkt_queue); dev_kfree_skb_any(skb_dequeue(&mcast->pkt_queue));
skb->dev = dev;
dev_kfree_skb_any(skb);
}
/* Clear the busy flag so we try again */ /* Clear the busy flag so we try again */
clear_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags); clear_bit(IPOIB_MCAST_FLAG_BUSY, &mcast->flags);
...@@ -928,21 +919,16 @@ struct ipoib_mcast_iter *ipoib_mcast_iter_init(struct net_device *dev) ...@@ -928,21 +919,16 @@ struct ipoib_mcast_iter *ipoib_mcast_iter_init(struct net_device *dev)
return NULL; return NULL;
iter->dev = dev; iter->dev = dev;
memset(iter->mgid.raw, 0, sizeof iter->mgid); memset(iter->mgid.raw, 0, 16);
if (ipoib_mcast_iter_next(iter)) { if (ipoib_mcast_iter_next(iter)) {
ipoib_mcast_iter_free(iter); kfree(iter);
return NULL; return NULL;
} }
return iter; return iter;
} }
void ipoib_mcast_iter_free(struct ipoib_mcast_iter *iter)
{
kfree(iter);
}
int ipoib_mcast_iter_next(struct ipoib_mcast_iter *iter) int ipoib_mcast_iter_next(struct ipoib_mcast_iter *iter)
{ {
struct ipoib_dev_priv *priv = netdev_priv(iter->dev); struct ipoib_dev_priv *priv = netdev_priv(iter->dev);
......
...@@ -113,8 +113,7 @@ int ipoib_vlan_add(struct net_device *pdev, unsigned short pkey) ...@@ -113,8 +113,7 @@ int ipoib_vlan_add(struct net_device *pdev, unsigned short pkey)
priv->parent = ppriv->dev; priv->parent = ppriv->dev;
if (ipoib_create_debug_file(priv->dev)) ipoib_create_debug_files(priv->dev);
goto debug_failed;
if (ipoib_add_pkey_attr(priv->dev)) if (ipoib_add_pkey_attr(priv->dev))
goto sysfs_failed; goto sysfs_failed;
...@@ -130,9 +129,7 @@ int ipoib_vlan_add(struct net_device *pdev, unsigned short pkey) ...@@ -130,9 +129,7 @@ int ipoib_vlan_add(struct net_device *pdev, unsigned short pkey)
return 0; return 0;
sysfs_failed: sysfs_failed:
ipoib_delete_debug_file(priv->dev); ipoib_delete_debug_files(priv->dev);
debug_failed:
unregister_netdev(priv->dev); unregister_netdev(priv->dev);
register_failed: register_failed:
......
...@@ -43,7 +43,7 @@ ...@@ -43,7 +43,7 @@
* Increment this value if any changes that break userspace ABI * Increment this value if any changes that break userspace ABI
* compatibility are made. * compatibility are made.
*/ */
#define IB_USER_VERBS_ABI_VERSION 3 #define IB_USER_VERBS_ABI_VERSION 4
enum { enum {
IB_USER_VERBS_CMD_GET_CONTEXT, IB_USER_VERBS_CMD_GET_CONTEXT,
...@@ -333,6 +333,11 @@ struct ib_uverbs_create_qp { ...@@ -333,6 +333,11 @@ struct ib_uverbs_create_qp {
struct ib_uverbs_create_qp_resp { struct ib_uverbs_create_qp_resp {
__u32 qp_handle; __u32 qp_handle;
__u32 qpn; __u32 qpn;
__u32 max_send_wr;
__u32 max_recv_wr;
__u32 max_send_sge;
__u32 max_recv_sge;
__u32 max_inline_data;
}; };
/* /*
...@@ -552,9 +557,7 @@ struct ib_uverbs_modify_srq { ...@@ -552,9 +557,7 @@ struct ib_uverbs_modify_srq {
__u32 srq_handle; __u32 srq_handle;
__u32 attr_mask; __u32 attr_mask;
__u32 max_wr; __u32 max_wr;
__u32 max_sge;
__u32 srq_limit; __u32 srq_limit;
__u32 reserved;
__u64 driver_data[0]; __u64 driver_data[0];
}; };
......
...@@ -881,7 +881,7 @@ struct ib_device { ...@@ -881,7 +881,7 @@ struct ib_device {
struct ib_ucontext *context, struct ib_ucontext *context,
struct ib_udata *udata); struct ib_udata *udata);
int (*destroy_cq)(struct ib_cq *cq); int (*destroy_cq)(struct ib_cq *cq);
int (*resize_cq)(struct ib_cq *cq, int *cqe); int (*resize_cq)(struct ib_cq *cq, int cqe);
int (*poll_cq)(struct ib_cq *cq, int num_entries, int (*poll_cq)(struct ib_cq *cq, int num_entries,
struct ib_wc *wc); struct ib_wc *wc);
int (*peek_cq)(struct ib_cq *cq, int wc_cnt); int (*peek_cq)(struct ib_cq *cq, int wc_cnt);
......
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