HID: Bluetooth: hidp: register HID devices async

While l2cap_user callbacks are running, the whole hci_dev is locked. Even
if we would add more fine-grained locking to HCI core, it would still be
called from the non-reentrant rx work-queue and thus block the event
processing.

However, if we want to perform synchronous I/O during HID device
registration (eg., to perform device-detection), we need the HCI core
to be able to dispatch incoming data.

Therefore, we now move device-registration to a separate worker. The HCI
core can continue running and we add devices asynchronously in another
kernel thread. Device removal is synchronized and waits for the worker
to exit before calling the usual device removal functions.

If l2cap_user->remove is called before the thread registered the devices,
we set "terminate" to true and the thread will skip it. If
l2cap_user->remove is called after it, we notice this as the device
is no longer in HIDP_SESSION_PREPARING state and simply unregister the
device as we did before.
There is no new deadlock as we now call hidp_session_add_dev() with
one lock less held (the HCI lock) and it cannot itself call back into
HCI as it was called with the HCI-lock held before.

One might wonder whether this can block during device unregistration.
But we set "terminate" to true and wake the HIDP thread up _before_
unregistering the HID/input devices. Therefore, all pending HID I/O
operations are canceled. All further I/O attempts will fail with ENODEV
or EIO. So all latency we can get are few context-switches, but no
timeouts or blocking I/O waits!

This change also prepares for a long standing HID bug. All HID devices
that register power_supply devices need to be able to handle callbacks
during registration (a power_supply oddity that cannot easily be fixed).
So with this patch available, we can allow HID I/O during registration
by calling the recently introduced hid_device_io_start/stop helpers,
which currently are a no-op for bluetooth due to this locking.

Note that we cannot do the same for input devices. input-core doesn't
allow us to call input_event() asynchronously to input_register_device(),
which HID-core kindly allows (for good reasons).
Fixing input-core to allow this isn't as easy as it sounds and is,
beside simplifying HIDP, not really an improvement. Hence, we still
register input devices synchronously as we did before. Only HID devices
are registered asynchronously.

Signed-off-by: David Herrmann <dh.herrmann@gmail.com>
Acked-by: Jiri Kosina <jkosina@suse.cz>
Acked-by: Gustavo Padovan <gustavo.padovan@collabora.co.uk>
Tested-by: Daniel Nicoletti <dantti12@gmail.com>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
This commit is contained in:
David Herrmann 2013-05-23 13:10:25 +02:00 committed by Jiri Kosina
parent f755407dd1
commit 4e713cdffb
2 changed files with 49 additions and 9 deletions

View File

@ -850,6 +850,29 @@ static void hidp_session_dev_del(struct hidp_session *session)
input_unregister_device(session->input); input_unregister_device(session->input);
} }
/*
* Asynchronous device registration
* HID device drivers might want to perform I/O during initialization to
* detect device types. Therefore, call device registration in a separate
* worker so the HIDP thread can schedule I/O operations.
* Note that this must be called after the worker thread was initialized
* successfully. This will then add the devices and increase session state
* on success, otherwise it will terminate the session thread.
*/
static void hidp_session_dev_work(struct work_struct *work)
{
struct hidp_session *session = container_of(work,
struct hidp_session,
dev_init);
int ret;
ret = hidp_session_dev_add(session);
if (!ret)
atomic_inc(&session->state);
else
hidp_session_terminate(session);
}
/* /*
* Create new session object * Create new session object
* Allocate session object, initialize static fields, copy input data into the * Allocate session object, initialize static fields, copy input data into the
@ -897,6 +920,7 @@ static int hidp_session_new(struct hidp_session **out, const bdaddr_t *bdaddr,
session->idle_to = req->idle_to; session->idle_to = req->idle_to;
/* device management */ /* device management */
INIT_WORK(&session->dev_init, hidp_session_dev_work);
setup_timer(&session->timer, hidp_idle_timeout, setup_timer(&session->timer, hidp_idle_timeout,
(unsigned long)session); (unsigned long)session);
@ -1035,8 +1059,8 @@ static void hidp_session_terminate(struct hidp_session *session)
* Probe HIDP session * Probe HIDP session
* This is called from the l2cap_conn core when our l2cap_user object is bound * This is called from the l2cap_conn core when our l2cap_user object is bound
* to the hci-connection. We get the session via the \user object and can now * to the hci-connection. We get the session via the \user object and can now
* start the session thread, register the HID/input devices and link it into * start the session thread, link it into the global session list and
* the global session list. * schedule HID/input device registration.
* The global session-list owns its own reference to the session object so you * The global session-list owns its own reference to the session object so you
* can drop your own reference after registering the l2cap_user object. * can drop your own reference after registering the l2cap_user object.
*/ */
@ -1058,21 +1082,30 @@ static int hidp_session_probe(struct l2cap_conn *conn,
goto out_unlock; goto out_unlock;
} }
if (session->input) {
ret = hidp_session_dev_add(session);
if (ret)
goto out_unlock;
}
ret = hidp_session_start_sync(session); ret = hidp_session_start_sync(session);
if (ret) if (ret)
goto out_unlock; goto out_del;
ret = hidp_session_dev_add(session); /* HID device registration is async to allow I/O during probe */
if (ret) if (session->input)
goto out_stop; atomic_inc(&session->state);
else
schedule_work(&session->dev_init);
hidp_session_get(session); hidp_session_get(session);
list_add(&session->list, &hidp_session_list); list_add(&session->list, &hidp_session_list);
ret = 0; ret = 0;
goto out_unlock; goto out_unlock;
out_stop: out_del:
hidp_session_terminate(session); if (session->input)
hidp_session_dev_del(session);
out_unlock: out_unlock:
up_write(&hidp_session_sem); up_write(&hidp_session_sem);
return ret; return ret;
@ -1102,7 +1135,12 @@ static void hidp_session_remove(struct l2cap_conn *conn,
down_write(&hidp_session_sem); down_write(&hidp_session_sem);
hidp_session_terminate(session); hidp_session_terminate(session);
hidp_session_dev_del(session);
cancel_work_sync(&session->dev_init);
if (session->input ||
atomic_read(&session->state) > HIDP_SESSION_PREPARING)
hidp_session_dev_del(session);
list_del(&session->list); list_del(&session->list);
up_write(&hidp_session_sem); up_write(&hidp_session_sem);

View File

@ -128,6 +128,7 @@ int hidp_get_conninfo(struct hidp_conninfo *ci);
enum hidp_session_state { enum hidp_session_state {
HIDP_SESSION_IDLING, HIDP_SESSION_IDLING,
HIDP_SESSION_PREPARING,
HIDP_SESSION_RUNNING, HIDP_SESSION_RUNNING,
}; };
@ -156,6 +157,7 @@ struct hidp_session {
unsigned long idle_to; unsigned long idle_to;
/* device management */ /* device management */
struct work_struct dev_init;
struct input_dev *input; struct input_dev *input;
struct hid_device *hid; struct hid_device *hid;
struct timer_list timer; struct timer_list timer;