Commit c1ea8823 authored by Sage Weil's avatar Sage Weil

ceph: fix osd request submission race

The osd request submission path registers the request, drops and retakes
the request_mutex, then sends it to the OSD.  A racing kick_requests could
sent it during that interval, causing the same msg to be sent twice and
BUGing in the msgr.

Fix by only sending the message if it hasn't been touched by other
threads.
Signed-off-by: default avatarSage Weil <sage@newdream.net>
parent 0656d11b
...@@ -837,7 +837,8 @@ static void kick_requests(struct ceph_osd_client *osdc, ...@@ -837,7 +837,8 @@ static void kick_requests(struct ceph_osd_client *osdc,
} }
kick: kick:
dout("kicking tid %llu osd%d\n", req->r_tid, req->r_osd->o_osd); dout("kicking %p tid %llu osd%d\n", req, req->r_tid,
req->r_osd->o_osd);
req->r_flags |= CEPH_OSD_FLAG_RETRY; req->r_flags |= CEPH_OSD_FLAG_RETRY;
err = __send_request(osdc, req); err = __send_request(osdc, req);
if (err) { if (err) {
...@@ -1016,7 +1017,7 @@ int ceph_osdc_start_request(struct ceph_osd_client *osdc, ...@@ -1016,7 +1017,7 @@ int ceph_osdc_start_request(struct ceph_osd_client *osdc,
struct ceph_osd_request *req, struct ceph_osd_request *req,
bool nofail) bool nofail)
{ {
int rc; int rc = 0;
req->r_request->pages = req->r_pages; req->r_request->pages = req->r_pages;
req->r_request->nr_pages = req->r_num_pages; req->r_request->nr_pages = req->r_num_pages;
...@@ -1025,15 +1026,22 @@ int ceph_osdc_start_request(struct ceph_osd_client *osdc, ...@@ -1025,15 +1026,22 @@ int ceph_osdc_start_request(struct ceph_osd_client *osdc,
down_read(&osdc->map_sem); down_read(&osdc->map_sem);
mutex_lock(&osdc->request_mutex); mutex_lock(&osdc->request_mutex);
rc = __send_request(osdc, req); /*
if (rc) { * a racing kick_requests() may have sent the message for us
if (nofail) { * while we dropped request_mutex above, so only send now if
dout("osdc_start_request failed send, marking %lld\n", * the request still han't been touched yet.
req->r_tid); */
req->r_resend = true; if (req->r_sent == 0) {
rc = 0; rc = __send_request(osdc, req);
} else { if (rc) {
__unregister_request(osdc, req); if (nofail) {
dout("osdc_start_request failed send, "
" marking %lld\n", req->r_tid);
req->r_resend = true;
rc = 0;
} else {
__unregister_request(osdc, req);
}
} }
} }
mutex_unlock(&osdc->request_mutex); mutex_unlock(&osdc->request_mutex);
......
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