greybus: svc: defer processing of hot-unplug requests

Defer processing also of hot-unplug events.

This is a step towards serialising hot-plug and unplug event processing.

Signed-off-by: Johan Hovold <johan@hovoldconsulting.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
This commit is contained in:
Johan Hovold 2015-12-02 18:23:30 +01:00 committed by Greg Kroah-Hartman
parent 9ae4109e5d
commit 57ccd4b087
1 changed files with 29 additions and 15 deletions
drivers/staging/greybus

View File

@ -458,6 +458,30 @@ static void gb_svc_process_intf_hotplug(struct gb_operation *operation)
gb_interface_remove(intf); gb_interface_remove(intf);
} }
static void gb_svc_process_intf_hot_unplug(struct gb_operation *operation)
{
struct gb_svc *svc = operation->connection->private;
struct gb_svc_intf_hot_unplug_request *request;
struct gb_host_device *hd = operation->connection->hd;
struct gb_interface *intf;
u8 intf_id;
/* The request message size has already been verified. */
request = operation->request->payload;
intf_id = request->intf_id;
dev_dbg(&svc->dev, "%s - id = %u\n", __func__, intf_id);
intf = gb_interface_find(hd, intf_id);
if (!intf) {
dev_warn(&svc->dev, "could not find hot-unplug interface %hhu\n",
intf_id);
return;
}
gb_svc_intf_remove(svc, intf);
}
static void gb_svc_process_deferred_request(struct work_struct *work) static void gb_svc_process_deferred_request(struct work_struct *work)
{ {
struct gb_svc_deferred_request *dr; struct gb_svc_deferred_request *dr;
@ -474,6 +498,9 @@ static void gb_svc_process_deferred_request(struct work_struct *work)
case GB_SVC_TYPE_INTF_HOTPLUG: case GB_SVC_TYPE_INTF_HOTPLUG:
gb_svc_process_intf_hotplug(operation); gb_svc_process_intf_hotplug(operation);
break; break;
case GB_SVC_TYPE_INTF_HOT_UNPLUG:
gb_svc_process_intf_hot_unplug(operation);
break;
default: default:
dev_err(&svc->dev, "bad deferred request type: %02x\n", type); dev_err(&svc->dev, "bad deferred request type: %02x\n", type);
} }
@ -531,9 +558,6 @@ static int gb_svc_intf_hot_unplug_recv(struct gb_operation *op)
{ {
struct gb_svc *svc = op->connection->private; struct gb_svc *svc = op->connection->private;
struct gb_svc_intf_hot_unplug_request *request; struct gb_svc_intf_hot_unplug_request *request;
struct gb_host_device *hd = op->connection->hd;
struct gb_interface *intf;
u8 intf_id;
if (op->request->payload_size < sizeof(*request)) { if (op->request->payload_size < sizeof(*request)) {
dev_warn(&svc->dev, "short hot unplug request received (%zu < %zu)\n", dev_warn(&svc->dev, "short hot unplug request received (%zu < %zu)\n",
@ -542,20 +566,10 @@ static int gb_svc_intf_hot_unplug_recv(struct gb_operation *op)
} }
request = op->request->payload; request = op->request->payload;
intf_id = request->intf_id;
dev_dbg(&svc->dev, "%s - id = %u\n", __func__, intf_id); dev_dbg(&svc->dev, "%s - id = %u\n", __func__, request->intf_id);
intf = gb_interface_find(hd, intf_id); return gb_svc_queue_deferred_request(op);
if (!intf) {
dev_warn(&svc->dev, "could not find hot-unplug interface %hhu\n",
intf_id);
return -EINVAL;
}
gb_svc_intf_remove(svc, intf);
return 0;
} }
static int gb_svc_intf_reset_recv(struct gb_operation *op) static int gb_svc_intf_reset_recv(struct gb_operation *op)