Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

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

[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
Loading
Loading
Loading
Loading
+13 −5
Original line number Original line Diff line number Diff line
@@ -507,7 +507,13 @@ static int send_mad(struct ib_sa_query *query, int timeout_ms)
		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,
		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,
		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);