Commit 6c67c7c3 authored by David Howells's avatar David Howells

AFS: Fix cache manager service handlers

Fix the cache manager RPC service handlers.  The afs_send_empty_reply() and
afs_send_simple_reply() functions:

 (a) Kill the call and free up the buffers associated with it if they fail.

 (b) Return with call intact if it they succeed.

However, none of the callers actually check the result or clean up if
successful - and may use the now non-existent data if it fails.

This was detected by Dan Carpenter using a static checker:

	The patch 08e0e7c8: "[AF_RXRPC]: Make the in-kernel AFS
	filesystem use AF_RXRPC." from Apr 26, 2007, leads to the following
	static checker warning:
	"fs/afs/cmservice.c:155 SRXAFSCB_CallBack()
		 warn: 'call' was already freed."
Reported-by: default avatarDan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: default avatarDavid Howells <dhowells@redhat.com>
parent 60b5f90d
...@@ -130,6 +130,15 @@ static void afs_cm_destructor(struct afs_call *call) ...@@ -130,6 +130,15 @@ static void afs_cm_destructor(struct afs_call *call)
{ {
_enter(""); _enter("");
/* Break the callbacks here so that we do it after the final ACK is
* received. The step number here must match the final number in
* afs_deliver_cb_callback().
*/
if (call->unmarshall == 6) {
ASSERT(call->server && call->count && call->request);
afs_break_callbacks(call->server, call->count, call->request);
}
afs_put_server(call->server); afs_put_server(call->server);
call->server = NULL; call->server = NULL;
kfree(call->buffer); kfree(call->buffer);
...@@ -272,6 +281,16 @@ static int afs_deliver_cb_callback(struct afs_call *call, struct sk_buff *skb, ...@@ -272,6 +281,16 @@ static int afs_deliver_cb_callback(struct afs_call *call, struct sk_buff *skb,
_debug("trailer"); _debug("trailer");
if (skb->len != 0) if (skb->len != 0)
return -EBADMSG; return -EBADMSG;
/* Record that the message was unmarshalled successfully so
* that the call destructor can know do the callback breaking
* work, even if the final ACK isn't received.
*
* If the step number changes, then afs_cm_destructor() must be
* updated also.
*/
call->unmarshall++;
case 6:
break; break;
} }
......
...@@ -183,6 +183,19 @@ static void afs_free_call(struct afs_call *call) ...@@ -183,6 +183,19 @@ static void afs_free_call(struct afs_call *call)
kfree(call); kfree(call);
} }
/*
* End a call
*/
static void afs_end_call(struct afs_call *call)
{
if (call->rxcall) {
rxrpc_kernel_end_call(call->rxcall);
call->rxcall = NULL;
}
call->type->destructor(call);
afs_free_call(call);
}
/* /*
* allocate a call with flat request and reply buffers * allocate a call with flat request and reply buffers
*/ */
...@@ -383,11 +396,8 @@ int afs_make_call(struct in_addr *addr, struct afs_call *call, gfp_t gfp, ...@@ -383,11 +396,8 @@ int afs_make_call(struct in_addr *addr, struct afs_call *call, gfp_t gfp,
rxrpc_kernel_abort_call(rxcall, RX_USER_ABORT); rxrpc_kernel_abort_call(rxcall, RX_USER_ABORT);
while ((skb = skb_dequeue(&call->rx_queue))) while ((skb = skb_dequeue(&call->rx_queue)))
afs_free_skb(skb); afs_free_skb(skb);
rxrpc_kernel_end_call(rxcall);
call->rxcall = NULL;
error_kill_call: error_kill_call:
call->type->destructor(call); afs_end_call(call);
afs_free_call(call);
_leave(" = %d", ret); _leave(" = %d", ret);
return ret; return ret;
} }
...@@ -509,12 +519,8 @@ static void afs_deliver_to_call(struct afs_call *call) ...@@ -509,12 +519,8 @@ static void afs_deliver_to_call(struct afs_call *call)
if (call->state >= AFS_CALL_COMPLETE) { if (call->state >= AFS_CALL_COMPLETE) {
while ((skb = skb_dequeue(&call->rx_queue))) while ((skb = skb_dequeue(&call->rx_queue)))
afs_free_skb(skb); afs_free_skb(skb);
if (call->incoming) { if (call->incoming)
rxrpc_kernel_end_call(call->rxcall); afs_end_call(call);
call->rxcall = NULL;
call->type->destructor(call);
afs_free_call(call);
}
} }
_leave(""); _leave("");
...@@ -564,10 +570,7 @@ static int afs_wait_for_call_to_complete(struct afs_call *call) ...@@ -564,10 +570,7 @@ static int afs_wait_for_call_to_complete(struct afs_call *call)
} }
_debug("call complete"); _debug("call complete");
rxrpc_kernel_end_call(call->rxcall); afs_end_call(call);
call->rxcall = NULL;
call->type->destructor(call);
afs_free_call(call);
_leave(" = %d", ret); _leave(" = %d", ret);
return ret; return ret;
} }
...@@ -790,10 +793,7 @@ void afs_send_empty_reply(struct afs_call *call) ...@@ -790,10 +793,7 @@ void afs_send_empty_reply(struct afs_call *call)
_debug("oom"); _debug("oom");
rxrpc_kernel_abort_call(call->rxcall, RX_USER_ABORT); rxrpc_kernel_abort_call(call->rxcall, RX_USER_ABORT);
default: default:
rxrpc_kernel_end_call(call->rxcall); afs_end_call(call);
call->rxcall = NULL;
call->type->destructor(call);
afs_free_call(call);
_leave(" [error]"); _leave(" [error]");
return; return;
} }
...@@ -823,17 +823,16 @@ void afs_send_simple_reply(struct afs_call *call, const void *buf, size_t len) ...@@ -823,17 +823,16 @@ void afs_send_simple_reply(struct afs_call *call, const void *buf, size_t len)
call->state = AFS_CALL_AWAIT_ACK; call->state = AFS_CALL_AWAIT_ACK;
n = rxrpc_kernel_send_data(call->rxcall, &msg, len); n = rxrpc_kernel_send_data(call->rxcall, &msg, len);
if (n >= 0) { if (n >= 0) {
/* Success */
_leave(" [replied]"); _leave(" [replied]");
return; return;
} }
if (n == -ENOMEM) { if (n == -ENOMEM) {
_debug("oom"); _debug("oom");
rxrpc_kernel_abort_call(call->rxcall, RX_USER_ABORT); rxrpc_kernel_abort_call(call->rxcall, RX_USER_ABORT);
} }
rxrpc_kernel_end_call(call->rxcall); afs_end_call(call);
call->rxcall = NULL;
call->type->destructor(call);
afs_free_call(call);
_leave(" [error]"); _leave(" [error]");
} }
......
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