|
@@ -162,18 +162,21 @@ static int v9fs_recv(struct v9fs_session_info *v9ses, struct v9fs_rpcreq *req)
|
|
dprintk(DEBUG_MUX, "waiting for response: %d\n", req->tcall->tag);
|
|
dprintk(DEBUG_MUX, "waiting for response: %d\n", req->tcall->tag);
|
|
ret = wait_event_interruptible(v9ses->read_wait,
|
|
ret = wait_event_interruptible(v9ses->read_wait,
|
|
((v9ses->transport->status != Connected) ||
|
|
((v9ses->transport->status != Connected) ||
|
|
- (req->rcall != 0) || dprintcond(v9ses, req)));
|
|
|
|
|
|
+ (req->rcall != 0) || (req->err < 0) ||
|
|
|
|
+ dprintcond(v9ses, req)));
|
|
|
|
|
|
dprintk(DEBUG_MUX, "got it: rcall %p\n", req->rcall);
|
|
dprintk(DEBUG_MUX, "got it: rcall %p\n", req->rcall);
|
|
|
|
+
|
|
|
|
+ spin_lock(&v9ses->muxlock);
|
|
|
|
+ list_del(&req->next);
|
|
|
|
+ spin_unlock(&v9ses->muxlock);
|
|
|
|
+
|
|
|
|
+ if (req->err < 0)
|
|
|
|
+ return req->err;
|
|
|
|
+
|
|
if (v9ses->transport->status == Disconnected)
|
|
if (v9ses->transport->status == Disconnected)
|
|
return -ECONNRESET;
|
|
return -ECONNRESET;
|
|
|
|
|
|
- if (ret == 0) {
|
|
|
|
- spin_lock(&v9ses->muxlock);
|
|
|
|
- list_del(&req->next);
|
|
|
|
- spin_unlock(&v9ses->muxlock);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -245,6 +248,9 @@ v9fs_mux_rpc(struct v9fs_session_info *v9ses, struct v9fs_fcall *tcall,
|
|
if (!v9ses)
|
|
if (!v9ses)
|
|
return -EINVAL;
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
+ if (!v9ses->transport || v9ses->transport->status != Connected)
|
|
|
|
+ return -EIO;
|
|
|
|
+
|
|
if (rcall)
|
|
if (rcall)
|
|
*rcall = NULL;
|
|
*rcall = NULL;
|
|
|
|
|
|
@@ -257,6 +263,7 @@ v9fs_mux_rpc(struct v9fs_session_info *v9ses, struct v9fs_fcall *tcall,
|
|
tcall->tag = tid;
|
|
tcall->tag = tid;
|
|
|
|
|
|
req.tcall = tcall;
|
|
req.tcall = tcall;
|
|
|
|
+ req.err = 0;
|
|
req.rcall = NULL;
|
|
req.rcall = NULL;
|
|
|
|
|
|
ret = v9fs_send(v9ses, &req);
|
|
ret = v9fs_send(v9ses, &req);
|
|
@@ -371,16 +378,21 @@ static int v9fs_recvproc(void *data)
|
|
}
|
|
}
|
|
|
|
|
|
err = read_message(v9ses, rcall, v9ses->maxdata + V9FS_IOHDRSZ);
|
|
err = read_message(v9ses, rcall, v9ses->maxdata + V9FS_IOHDRSZ);
|
|
- if (err < 0) {
|
|
|
|
- kfree(rcall);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
spin_lock(&v9ses->muxlock);
|
|
spin_lock(&v9ses->muxlock);
|
|
- list_for_each_entry_safe(rreq, rptr, &v9ses->mux_fcalls, next) {
|
|
|
|
- if (rreq->tcall->tag == rcall->tag) {
|
|
|
|
- req = rreq;
|
|
|
|
- req->rcall = rcall;
|
|
|
|
- break;
|
|
|
|
|
|
+ if (err < 0) {
|
|
|
|
+ list_for_each_entry_safe(rreq, rptr, &v9ses->mux_fcalls, next) {
|
|
|
|
+ rreq->err = err;
|
|
|
|
+ }
|
|
|
|
+ if(err != -ERESTARTSYS)
|
|
|
|
+ eprintk(KERN_ERR,
|
|
|
|
+ "Transport error while reading message %d\n", err);
|
|
|
|
+ } else {
|
|
|
|
+ list_for_each_entry_safe(rreq, rptr, &v9ses->mux_fcalls, next) {
|
|
|
|
+ if (rreq->tcall->tag == rcall->tag) {
|
|
|
|
+ req = rreq;
|
|
|
|
+ req->rcall = rcall;
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -399,9 +411,10 @@ static int v9fs_recvproc(void *data)
|
|
spin_unlock(&v9ses->muxlock);
|
|
spin_unlock(&v9ses->muxlock);
|
|
|
|
|
|
if (!req) {
|
|
if (!req) {
|
|
- dprintk(DEBUG_ERROR,
|
|
|
|
- "unexpected response: id %d tag %d\n",
|
|
|
|
- rcall->id, rcall->tag);
|
|
|
|
|
|
+ if (err >= 0)
|
|
|
|
+ dprintk(DEBUG_ERROR,
|
|
|
|
+ "unexpected response: id %d tag %d\n",
|
|
|
|
+ rcall->id, rcall->tag);
|
|
|
|
|
|
kfree(rcall);
|
|
kfree(rcall);
|
|
}
|
|
}
|
|
@@ -410,6 +423,8 @@ static int v9fs_recvproc(void *data)
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ v9ses->transport->close(v9ses->transport);
|
|
|
|
+
|
|
/* Inform all pending processes about the failure */
|
|
/* Inform all pending processes about the failure */
|
|
wake_up_all(&v9ses->read_wait);
|
|
wake_up_all(&v9ses->read_wait);
|
|
|
|
|