greybus: simplify pending operations tracking
Greg raised the alarm when I first put in the red-black tree for tracking pending operations. The reality as that we're not likely to have that many operations in flight at any one time, so the complexity of the red-black tree is most likely unwarranted. I already This pulls out the red-black tree and uses a simple list instead. A connection maintains two lists of operations. An operation starts on its connection's operations list. It is moved to the pending list when its request message is sent. And it is moved back to the operations list when the response message arrives. It is removed from whatever list it's in when the operation is destroyed. We reuse the single operation->links field for both lists. Only outgoing requests are ever "pending." Incoming requests are transient--we receive them, process them, send the response, and then we're done. Change a few function names so it's clear we're working with the pending list. Signed-off-by: Alex Elder <elder@linaro.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
8350e7a011
commit
b8616da875
@ -209,7 +209,7 @@ struct gb_connection *gb_connection_create(struct gb_interface *interface,
|
|||||||
spin_unlock_irq(&gb_connections_lock);
|
spin_unlock_irq(&gb_connections_lock);
|
||||||
|
|
||||||
INIT_LIST_HEAD(&connection->operations);
|
INIT_LIST_HEAD(&connection->operations);
|
||||||
connection->pending = RB_ROOT;
|
INIT_LIST_HEAD(&connection->pending);
|
||||||
atomic_set(&connection->op_cycle, 0);
|
atomic_set(&connection->op_cycle, 0);
|
||||||
|
|
||||||
return connection;
|
return connection;
|
||||||
|
@ -36,7 +36,7 @@ struct gb_connection {
|
|||||||
enum gb_connection_state state;
|
enum gb_connection_state state;
|
||||||
|
|
||||||
struct list_head operations;
|
struct list_head operations;
|
||||||
struct rb_root pending; /* awaiting reponse */
|
struct list_head pending; /* awaiting reponse */
|
||||||
atomic_t op_cycle;
|
atomic_t op_cycle;
|
||||||
|
|
||||||
void *private;
|
void *private;
|
||||||
|
@ -56,13 +56,9 @@ struct gb_operation_msg_hdr {
|
|||||||
/* XXX Could be per-host device, per-module, or even per-connection */
|
/* XXX Could be per-host device, per-module, or even per-connection */
|
||||||
static DEFINE_SPINLOCK(gb_operations_lock);
|
static DEFINE_SPINLOCK(gb_operations_lock);
|
||||||
|
|
||||||
static void gb_operation_insert(struct gb_operation *operation)
|
static void gb_pending_operation_insert(struct gb_operation *operation)
|
||||||
{
|
{
|
||||||
struct gb_connection *connection = operation->connection;
|
struct gb_connection *connection = operation->connection;
|
||||||
struct rb_root *root = &connection->pending;
|
|
||||||
struct rb_node *node = &operation->node;
|
|
||||||
struct rb_node **link = &root->rb_node;
|
|
||||||
struct rb_node *above = NULL;
|
|
||||||
struct gb_operation_msg_hdr *header;
|
struct gb_operation_msg_hdr *header;
|
||||||
|
|
||||||
/* Assign the operation's id, and store it in the header of
|
/* Assign the operation's id, and store it in the header of
|
||||||
@ -72,25 +68,13 @@ static void gb_operation_insert(struct gb_operation *operation)
|
|||||||
header = operation->request->transfer_buffer;
|
header = operation->request->transfer_buffer;
|
||||||
header->id = cpu_to_le16(operation->id);
|
header->id = cpu_to_le16(operation->id);
|
||||||
|
|
||||||
/* OK, insert the operation into its connection's pending tree */
|
/* Insert the operation into its connection's pending list */
|
||||||
spin_lock_irq(&gb_operations_lock);
|
spin_lock_irq(&gb_operations_lock);
|
||||||
while (*link) {
|
list_move_tail(&operation->links, &connection->pending);
|
||||||
struct gb_operation *other;
|
|
||||||
|
|
||||||
above = *link;
|
|
||||||
other = rb_entry(above, struct gb_operation, node);
|
|
||||||
header = other->request->transfer_buffer;
|
|
||||||
if (other->id > operation->id)
|
|
||||||
link = &above->rb_left;
|
|
||||||
else if (other->id < operation->id)
|
|
||||||
link = &above->rb_right;
|
|
||||||
}
|
|
||||||
rb_link_node(node, above, link);
|
|
||||||
rb_insert_color(node, root);
|
|
||||||
spin_unlock_irq(&gb_operations_lock);
|
spin_unlock_irq(&gb_operations_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void gb_operation_remove(struct gb_operation *operation)
|
static void gb_pending_operation_remove(struct gb_operation *operation)
|
||||||
{
|
{
|
||||||
struct gb_connection *connection = operation->connection;
|
struct gb_connection *connection = operation->connection;
|
||||||
|
|
||||||
@ -99,29 +83,22 @@ static void gb_operation_remove(struct gb_operation *operation)
|
|||||||
|
|
||||||
/* Take us off of the list of pending operations */
|
/* Take us off of the list of pending operations */
|
||||||
spin_lock_irq(&gb_operations_lock);
|
spin_lock_irq(&gb_operations_lock);
|
||||||
rb_erase(&operation->node, &connection->pending);
|
list_move_tail(&operation->links, &connection->operations);
|
||||||
spin_unlock_irq(&gb_operations_lock);
|
spin_unlock_irq(&gb_operations_lock);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct gb_operation *
|
static struct gb_operation *
|
||||||
gb_operation_find(struct gb_connection *connection, u16 id)
|
gb_pending_operation_find(struct gb_connection *connection, u16 id)
|
||||||
{
|
{
|
||||||
struct gb_operation *operation = NULL;
|
struct gb_operation *operation;
|
||||||
struct rb_node *node;
|
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
spin_lock_irq(&gb_operations_lock);
|
spin_lock_irq(&gb_operations_lock);
|
||||||
node = connection->pending.rb_node;
|
list_for_each_entry(operation, &connection->pending, links)
|
||||||
while (node && !found) {
|
if (operation->id == id) {
|
||||||
operation = rb_entry(node, struct gb_operation, node);
|
|
||||||
if (operation->id > id)
|
|
||||||
node = node->rb_left;
|
|
||||||
else if (operation->id < id)
|
|
||||||
node = node->rb_right;
|
|
||||||
else
|
|
||||||
found = true;
|
found = true;
|
||||||
}
|
break;
|
||||||
|
}
|
||||||
spin_unlock_irq(&gb_operations_lock);
|
spin_unlock_irq(&gb_operations_lock);
|
||||||
|
|
||||||
return found ? operation : NULL;
|
return found ? operation : NULL;
|
||||||
@ -405,7 +382,7 @@ int gb_operation_request_send(struct gb_operation *operation,
|
|||||||
* setting the operation id and submitting the gbuf.
|
* setting the operation id and submitting the gbuf.
|
||||||
*/
|
*/
|
||||||
operation->callback = callback;
|
operation->callback = callback;
|
||||||
gb_operation_insert(operation);
|
gb_pending_operation_insert(operation);
|
||||||
ret = greybus_submit_gbuf(operation->request, GFP_KERNEL);
|
ret = greybus_submit_gbuf(operation->request, GFP_KERNEL);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
@ -424,7 +401,6 @@ int gb_operation_request_send(struct gb_operation *operation,
|
|||||||
*/
|
*/
|
||||||
int gb_operation_response_send(struct gb_operation *operation)
|
int gb_operation_response_send(struct gb_operation *operation)
|
||||||
{
|
{
|
||||||
gb_operation_remove(operation);
|
|
||||||
gb_operation_destroy(operation);
|
gb_operation_destroy(operation);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -456,12 +432,12 @@ void gb_connection_operation_recv(struct gb_connection *connection,
|
|||||||
if (header->type & GB_OPERATION_TYPE_RESPONSE) {
|
if (header->type & GB_OPERATION_TYPE_RESPONSE) {
|
||||||
u16 id = le16_to_cpu(header->id);
|
u16 id = le16_to_cpu(header->id);
|
||||||
|
|
||||||
operation = gb_operation_find(connection, id);
|
operation = gb_pending_operation_find(connection, id);
|
||||||
if (!operation) {
|
if (!operation) {
|
||||||
gb_connection_err(connection, "operation not found");
|
gb_connection_err(connection, "operation not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
gb_operation_remove(operation);
|
gb_pending_operation_remove(operation);
|
||||||
gbuf = operation->response;
|
gbuf = operation->response;
|
||||||
gbuf->status = GB_OP_SUCCESS; /* If we got here we're good */
|
gbuf->status = GB_OP_SUCCESS; /* If we got here we're good */
|
||||||
if (size > gbuf->transfer_buffer_length) {
|
if (size > gbuf->transfer_buffer_length) {
|
||||||
|
@ -64,8 +64,7 @@ struct gb_operation {
|
|||||||
struct completion completion; /* Used if no callback */
|
struct completion completion; /* Used if no callback */
|
||||||
struct delayed_work timeout_work;
|
struct delayed_work timeout_work;
|
||||||
|
|
||||||
struct list_head links; /* connection->operations */
|
struct list_head links; /* connection->{operations,pending} */
|
||||||
struct rb_node node; /* connection->pending */
|
|
||||||
|
|
||||||
/* These are what's used by caller */
|
/* These are what's used by caller */
|
||||||
void *request_payload;
|
void *request_payload;
|
||||||
|
Loading…
Reference in New Issue
Block a user