Commit 0581f28e authored by Johan Hovold's avatar Johan Hovold Committed by Greg Kroah-Hartman

greybus: operation: fix use-after-free in response receive path

Fix potential use-after-free in response receive path, due to lack of
reference counting when looking up operations on a connection.

Make sure to acquire a reference to the operation while holding the
connection-list lock.
Signed-off-by: default avatarJohan Hovold <johan@hovoldconsulting.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@google.com>
parent 85109f7d
...@@ -114,6 +114,10 @@ int gb_operation_result(struct gb_operation *operation) ...@@ -114,6 +114,10 @@ int gb_operation_result(struct gb_operation *operation)
} }
EXPORT_SYMBOL_GPL(gb_operation_result); EXPORT_SYMBOL_GPL(gb_operation_result);
/*
* Looks up an operation on a connection and returns a refcounted pointer if
* found, or NULL otherwise.
*/
static struct gb_operation * static struct gb_operation *
gb_operation_find(struct gb_connection *connection, u16 operation_id) gb_operation_find(struct gb_connection *connection, u16 operation_id)
{ {
...@@ -124,6 +128,7 @@ gb_operation_find(struct gb_connection *connection, u16 operation_id) ...@@ -124,6 +128,7 @@ gb_operation_find(struct gb_connection *connection, u16 operation_id)
spin_lock_irqsave(&gb_operations_lock, flags); spin_lock_irqsave(&gb_operations_lock, flags);
list_for_each_entry(operation, &connection->operations, links) list_for_each_entry(operation, &connection->operations, links)
if (operation->id == operation_id) { if (operation->id == operation_id) {
gb_operation_get(operation);
found = true; found = true;
break; break;
} }
...@@ -795,6 +800,8 @@ static void gb_connection_recv_response(struct gb_connection *connection, ...@@ -795,6 +800,8 @@ static void gb_connection_recv_response(struct gb_connection *connection,
/* The rest will be handled in work queue context */ /* The rest will be handled in work queue context */
if (gb_operation_result_set(operation, errno)) if (gb_operation_result_set(operation, errno))
queue_work(gb_operation_workqueue, &operation->work); queue_work(gb_operation_workqueue, &operation->work);
gb_operation_put(operation);
} }
/* /*
......
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