Commit acd45c56 authored by Takashi Iwai's avatar Takashi Iwai Committed by Thomas Zimmermann

drm/udl: Replace semaphore with a simple wait queue

UDL driver uses a semaphore for controlling the emptiness of FIFO in a
slightly funky way.  This patch replaces it with a wait queue and
controls the emptiness with the standard wait_event*() macro instead,
which is a more straightforward implementation.

While we are at it, drop the dead code for delayed semaphore down,
too.
Tested-by: default avatarThomas Zimmermann <tzimmermann@suse.de>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
Signed-off-by: default avatarThomas Zimmermann <tzimmermann@suse.de>
Reviewed-by: default avatarThomas Zimmermann <tzimmermann@suse.de>
Link: https://patchwork.freedesktop.org/patch/msgid/20220804075826.27036-2-tiwai@suse.de
parent 504a51d7
...@@ -34,14 +34,13 @@ struct udl_device; ...@@ -34,14 +34,13 @@ struct udl_device;
struct urb_node { struct urb_node {
struct list_head entry; struct list_head entry;
struct udl_device *dev; struct udl_device *dev;
struct delayed_work release_urb_work;
struct urb *urb; struct urb *urb;
}; };
struct urb_list { struct urb_list {
struct list_head list; struct list_head list;
spinlock_t lock; spinlock_t lock;
struct semaphore limit_sem; wait_queue_head_t sleep;
int available; int available;
int count; int count;
size_t size; size_t size;
...@@ -75,7 +74,13 @@ static inline struct usb_device *udl_to_usb_device(struct udl_device *udl) ...@@ -75,7 +74,13 @@ static inline struct usb_device *udl_to_usb_device(struct udl_device *udl)
int udl_modeset_init(struct drm_device *dev); int udl_modeset_init(struct drm_device *dev);
struct drm_connector *udl_connector_init(struct drm_device *dev); struct drm_connector *udl_connector_init(struct drm_device *dev);
struct urb *udl_get_urb(struct drm_device *dev); struct urb *udl_get_urb_timeout(struct drm_device *dev, long timeout);
#define GET_URB_TIMEOUT HZ
static inline struct urb *udl_get_urb(struct drm_device *dev)
{
return udl_get_urb_timeout(dev, GET_URB_TIMEOUT);
}
int udl_submit_urb(struct drm_device *dev, struct urb *urb, size_t len); int udl_submit_urb(struct drm_device *dev, struct urb *urb, size_t len);
void udl_urb_completion(struct urb *urb); void udl_urb_completion(struct urb *urb);
......
...@@ -23,9 +23,6 @@ ...@@ -23,9 +23,6 @@
#define WRITES_IN_FLIGHT (4) #define WRITES_IN_FLIGHT (4)
#define MAX_VENDOR_DESCRIPTOR_SIZE 256 #define MAX_VENDOR_DESCRIPTOR_SIZE 256
#define GET_URB_TIMEOUT HZ
#define FREE_URB_TIMEOUT (HZ*2)
static int udl_parse_vendor_descriptor(struct udl_device *udl) static int udl_parse_vendor_descriptor(struct udl_device *udl)
{ {
struct usb_device *udev = udl_to_usb_device(udl); struct usb_device *udev = udl_to_usb_device(udl);
...@@ -119,14 +116,6 @@ static int udl_select_std_channel(struct udl_device *udl) ...@@ -119,14 +116,6 @@ static int udl_select_std_channel(struct udl_device *udl)
return ret < 0 ? ret : 0; return ret < 0 ? ret : 0;
} }
static void udl_release_urb_work(struct work_struct *work)
{
struct urb_node *unode = container_of(work, struct urb_node,
release_urb_work.work);
up(&unode->dev->urbs.limit_sem);
}
void udl_urb_completion(struct urb *urb) void udl_urb_completion(struct urb *urb)
{ {
struct urb_node *unode = urb->context; struct urb_node *unode = urb->context;
...@@ -150,23 +139,13 @@ void udl_urb_completion(struct urb *urb) ...@@ -150,23 +139,13 @@ void udl_urb_completion(struct urb *urb)
udl->urbs.available++; udl->urbs.available++;
spin_unlock_irqrestore(&udl->urbs.lock, flags); spin_unlock_irqrestore(&udl->urbs.lock, flags);
#if 0 wake_up(&udl->urbs.sleep);
/*
* When using fb_defio, we deadlock if up() is called
* while another is waiting. So queue to another process.
*/
if (fb_defio)
schedule_delayed_work(&unode->release_urb_work, 0);
else
#endif
up(&udl->urbs.limit_sem);
} }
static void udl_free_urb_list(struct drm_device *dev) static void udl_free_urb_list(struct drm_device *dev)
{ {
struct udl_device *udl = to_udl(dev); struct udl_device *udl = to_udl(dev);
int count = udl->urbs.count; int count = udl->urbs.count;
struct list_head *node;
struct urb_node *unode; struct urb_node *unode;
struct urb *urb; struct urb *urb;
...@@ -174,23 +153,15 @@ static void udl_free_urb_list(struct drm_device *dev) ...@@ -174,23 +153,15 @@ static void udl_free_urb_list(struct drm_device *dev)
/* keep waiting and freeing, until we've got 'em all */ /* keep waiting and freeing, until we've got 'em all */
while (count--) { while (count--) {
down(&udl->urbs.limit_sem); urb = udl_get_urb_timeout(dev, MAX_SCHEDULE_TIMEOUT);
if (WARN_ON(!urb))
spin_lock_irq(&udl->urbs.lock); break;
unode = urb->context;
node = udl->urbs.list.next; /* have reserved one with sem */
list_del_init(node);
spin_unlock_irq(&udl->urbs.lock);
unode = list_entry(node, struct urb_node, entry);
urb = unode->urb;
/* Free each separately allocated piece */ /* Free each separately allocated piece */
usb_free_coherent(urb->dev, udl->urbs.size, usb_free_coherent(urb->dev, udl->urbs.size,
urb->transfer_buffer, urb->transfer_dma); urb->transfer_buffer, urb->transfer_dma);
usb_free_urb(urb); usb_free_urb(urb);
kfree(node); kfree(unode);
} }
udl->urbs.count = 0; udl->urbs.count = 0;
} }
...@@ -210,7 +181,7 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size) ...@@ -210,7 +181,7 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size)
udl->urbs.size = size; udl->urbs.size = size;
INIT_LIST_HEAD(&udl->urbs.list); INIT_LIST_HEAD(&udl->urbs.list);
sema_init(&udl->urbs.limit_sem, 0); init_waitqueue_head(&udl->urbs.sleep);
udl->urbs.count = 0; udl->urbs.count = 0;
udl->urbs.available = 0; udl->urbs.available = 0;
...@@ -220,9 +191,6 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size) ...@@ -220,9 +191,6 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size)
break; break;
unode->dev = udl; unode->dev = udl;
INIT_DELAYED_WORK(&unode->release_urb_work,
udl_release_urb_work);
urb = usb_alloc_urb(0, GFP_KERNEL); urb = usb_alloc_urb(0, GFP_KERNEL);
if (!urb) { if (!urb) {
kfree(unode); kfree(unode);
...@@ -250,7 +218,6 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size) ...@@ -250,7 +218,6 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size)
list_add_tail(&unode->entry, &udl->urbs.list); list_add_tail(&unode->entry, &udl->urbs.list);
up(&udl->urbs.limit_sem);
udl->urbs.count++; udl->urbs.count++;
udl->urbs.available++; udl->urbs.available++;
} }
...@@ -260,36 +227,31 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size) ...@@ -260,36 +227,31 @@ static int udl_alloc_urb_list(struct drm_device *dev, int count, size_t size)
return udl->urbs.count; return udl->urbs.count;
} }
struct urb *udl_get_urb(struct drm_device *dev) struct urb *udl_get_urb_timeout(struct drm_device *dev, long timeout)
{ {
struct udl_device *udl = to_udl(dev); struct udl_device *udl = to_udl(dev);
int ret = 0; struct urb_node *unode = NULL;
struct list_head *entry;
struct urb_node *unode;
struct urb *urb = NULL;
/* Wait for an in-flight buffer to complete and get re-queued */ if (!udl->urbs.count)
ret = down_timeout(&udl->urbs.limit_sem, GET_URB_TIMEOUT); return NULL;
if (ret) {
DRM_INFO("wait for urb interrupted: %x available: %d\n",
ret, udl->urbs.available);
goto error;
}
/* Wait for an in-flight buffer to complete and get re-queued */
spin_lock_irq(&udl->urbs.lock); spin_lock_irq(&udl->urbs.lock);
if (!wait_event_lock_irq_timeout(udl->urbs.sleep,
!list_empty(&udl->urbs.list),
udl->urbs.lock, timeout)) {
DRM_INFO("wait for urb interrupted: available: %d\n",
udl->urbs.available);
goto unlock;
}
BUG_ON(list_empty(&udl->urbs.list)); /* reserved one with limit_sem */ unode = list_first_entry(&udl->urbs.list, struct urb_node, entry);
entry = udl->urbs.list.next; list_del_init(&unode->entry);
list_del_init(entry);
udl->urbs.available--; udl->urbs.available--;
unlock:
spin_unlock_irq(&udl->urbs.lock); spin_unlock_irq(&udl->urbs.lock);
return unode ? unode->urb : NULL;
unode = list_entry(entry, struct urb_node, entry);
urb = unode->urb;
error:
return urb;
} }
int udl_submit_urb(struct drm_device *dev, struct urb *urb, size_t len) int udl_submit_urb(struct drm_device *dev, struct urb *urb, size_t len)
......
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