summaryrefslogtreecommitdiffstats
path: root/drivers/staging/greybus/connection.c
diff options
context:
space:
mode:
authorJohan Hovold <johan@hovoldconsulting.com>2016-01-19 12:51:07 +0100
committerGreg Kroah-Hartman <gregkh@google.com>2016-01-19 12:12:40 -0800
commit570dfa7c55a76258b03a2d93f9db12dacc2ad3c6 (patch)
treeb4b4f11f9c7b0e86577909d615baa8256959eb28 /drivers/staging/greybus/connection.c
parent23268785b288c25ab5ee2ec523982546b5353880 (diff)
greybus: connection: add unidirectional enabled state
Add a new connection state ENABLED_TX in which only outgoing operations are allowed. This allows drivers to query the device during probe before allocating their state containers without having to worry about racing incoming requests. Reviewed-by: Viresh Kumar <viresh.kumar@linaro.org> Signed-off-by: Johan Hovold <johan@hovoldconsulting.com> Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
Diffstat (limited to 'drivers/staging/greybus/connection.c')
-rw-r--r--drivers/staging/greybus/connection.c19
1 files changed, 18 insertions, 1 deletions
diff --git a/drivers/staging/greybus/connection.c b/drivers/staging/greybus/connection.c
index 56588f357b20..8ae099d20b48 100644
--- a/drivers/staging/greybus/connection.c
+++ b/drivers/staging/greybus/connection.c
@@ -396,6 +396,18 @@ int gb_connection_enable(struct gb_connection *connection,
if (connection->state == GB_CONNECTION_STATE_ENABLED)
goto out_unlock;
+ if (connection->state == GB_CONNECTION_STATE_ENABLED_TX) {
+ if (!handler)
+ goto out_unlock;
+
+ spin_lock_irq(&connection->lock);
+ connection->handler = handler;
+ connection->state = GB_CONNECTION_STATE_ENABLED;
+ spin_unlock_irq(&connection->lock);
+
+ goto out_unlock;
+ }
+
ret = gb_connection_hd_cport_enable(connection);
if (ret)
goto err_unlock;
@@ -406,7 +418,10 @@ int gb_connection_enable(struct gb_connection *connection,
spin_lock_irq(&connection->lock);
connection->handler = handler;
- connection->state = GB_CONNECTION_STATE_ENABLED;
+ if (handler)
+ connection->state = GB_CONNECTION_STATE_ENABLED;
+ else
+ connection->state = GB_CONNECTION_STATE_ENABLED_TX;
spin_unlock_irq(&connection->lock);
ret = gb_connection_control_connected(connection);
@@ -422,6 +437,7 @@ err_svc_destroy:
spin_lock_irq(&connection->lock);
connection->state = GB_CONNECTION_STATE_DISABLED;
gb_connection_cancel_operations(connection, -ESHUTDOWN);
+ connection->handler = NULL;
spin_unlock_irq(&connection->lock);
gb_connection_svc_connection_destroy(connection);
@@ -446,6 +462,7 @@ void gb_connection_disable(struct gb_connection *connection)
spin_lock_irq(&connection->lock);
connection->state = GB_CONNECTION_STATE_DISABLED;
gb_connection_cancel_operations(connection, -ESHUTDOWN);
+ connection->handler = NULL;
spin_unlock_irq(&connection->lock);
gb_connection_svc_connection_destroy(connection);