Commit dae4c1d2 authored by Roland Dreier's avatar Roland Dreier Committed by Linus Torvalds

[PATCH] IB: Fix race in sa_query

Use a copy of the id we'll return to the consumer so that we don't
dereference query->sa_query after calling send_mad().  A completion may
occur very quickly and end up freeing the query before we get to do
anything after send_mad().
Signed-off-by: default avatarRoland Dreier <rolandd@cisco.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent cae54bdf
...@@ -507,7 +507,13 @@ retry: ...@@ -507,7 +507,13 @@ retry:
spin_unlock_irqrestore(&idr_lock, flags); spin_unlock_irqrestore(&idr_lock, flags);
} }
return ret; /*
* It's not safe to dereference query any more, because the
* send may already have completed and freed the query in
* another context. So use wr.wr_id, which has a copy of the
* query's id.
*/
return ret ? ret : wr.wr_id;
} }
static void ib_sa_path_rec_callback(struct ib_sa_query *sa_query, static void ib_sa_path_rec_callback(struct ib_sa_query *sa_query,
...@@ -598,14 +604,15 @@ int ib_sa_path_rec_get(struct ib_device *device, u8 port_num, ...@@ -598,14 +604,15 @@ int ib_sa_path_rec_get(struct ib_device *device, u8 port_num,
rec, query->sa_query.mad->data); rec, query->sa_query.mad->data);
*sa_query = &query->sa_query; *sa_query = &query->sa_query;
ret = send_mad(&query->sa_query, timeout_ms); ret = send_mad(&query->sa_query, timeout_ms);
if (ret) { if (ret < 0) {
*sa_query = NULL; *sa_query = NULL;
kfree(query->sa_query.mad); kfree(query->sa_query.mad);
kfree(query); kfree(query);
} }
return ret ? ret : query->sa_query.id; return ret;
} }
EXPORT_SYMBOL(ib_sa_path_rec_get); EXPORT_SYMBOL(ib_sa_path_rec_get);
...@@ -674,14 +681,15 @@ int ib_sa_mcmember_rec_query(struct ib_device *device, u8 port_num, ...@@ -674,14 +681,15 @@ int ib_sa_mcmember_rec_query(struct ib_device *device, u8 port_num,
rec, query->sa_query.mad->data); rec, query->sa_query.mad->data);
*sa_query = &query->sa_query; *sa_query = &query->sa_query;
ret = send_mad(&query->sa_query, timeout_ms); ret = send_mad(&query->sa_query, timeout_ms);
if (ret) { if (ret < 0) {
*sa_query = NULL; *sa_query = NULL;
kfree(query->sa_query.mad); kfree(query->sa_query.mad);
kfree(query); kfree(query);
} }
return ret ? ret : query->sa_query.id; return ret;
} }
EXPORT_SYMBOL(ib_sa_mcmember_rec_query); EXPORT_SYMBOL(ib_sa_mcmember_rec_query);
......
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