drm/qxl: add delayed fb operations

Due to the nature of qxl hw we cannot queue operations while in an irq
context, so we queue these operations as best we can until atomic allocations
fail, and dequeue them later in a work queue.

Daniel looked over the locking on the list and agrees it should be sufficent.

The atomic allocs use no warn, as the last thing we want if we haven't memory
to allocate space for a printk in an irq context is more printks.

Signed-off-by: Dave Airlie <airlied@redhat.com>
This commit is contained in:
Dave Airlie 2013-07-22 14:37:39 +10:00
parent 3b2f64d00c
commit 0665f9f852
2 changed files with 166 additions and 23 deletions

View File

@ -314,6 +314,7 @@ struct qxl_device {
struct workqueue_struct *gc_queue; struct workqueue_struct *gc_queue;
struct work_struct gc_work; struct work_struct gc_work;
struct work_struct fb_work;
}; };
/* forward declaration for QXL_INFO_IO */ /* forward declaration for QXL_INFO_IO */

View File

@ -37,12 +37,29 @@
#define QXL_DIRTY_DELAY (HZ / 30) #define QXL_DIRTY_DELAY (HZ / 30)
#define QXL_FB_OP_FILLRECT 1
#define QXL_FB_OP_COPYAREA 2
#define QXL_FB_OP_IMAGEBLIT 3
struct qxl_fb_op {
struct list_head head;
int op_type;
union {
struct fb_fillrect fr;
struct fb_copyarea ca;
struct fb_image ib;
} op;
void *img_data;
};
struct qxl_fbdev { struct qxl_fbdev {
struct drm_fb_helper helper; struct drm_fb_helper helper;
struct qxl_framebuffer qfb; struct qxl_framebuffer qfb;
struct list_head fbdev_list; struct list_head fbdev_list;
struct qxl_device *qdev; struct qxl_device *qdev;
spinlock_t delayed_ops_lock;
struct list_head delayed_ops;
void *shadow; void *shadow;
int size; int size;
@ -164,8 +181,69 @@ static struct fb_deferred_io qxl_defio = {
.deferred_io = qxl_deferred_io, .deferred_io = qxl_deferred_io,
}; };
static void qxl_fb_fillrect(struct fb_info *info, static void qxl_fb_delayed_fillrect(struct qxl_fbdev *qfbdev,
const struct fb_fillrect *fb_rect) const struct fb_fillrect *fb_rect)
{
struct qxl_fb_op *op;
unsigned long flags;
op = kmalloc(sizeof(struct qxl_fb_op), GFP_ATOMIC | __GFP_NOWARN);
if (!op)
return;
op->op.fr = *fb_rect;
op->img_data = NULL;
op->op_type = QXL_FB_OP_FILLRECT;
spin_lock_irqsave(&qfbdev->delayed_ops_lock, flags);
list_add_tail(&op->head, &qfbdev->delayed_ops);
spin_unlock_irqrestore(&qfbdev->delayed_ops_lock, flags);
}
static void qxl_fb_delayed_copyarea(struct qxl_fbdev *qfbdev,
const struct fb_copyarea *fb_copy)
{
struct qxl_fb_op *op;
unsigned long flags;
op = kmalloc(sizeof(struct qxl_fb_op), GFP_ATOMIC | __GFP_NOWARN);
if (!op)
return;
op->op.ca = *fb_copy;
op->img_data = NULL;
op->op_type = QXL_FB_OP_COPYAREA;
spin_lock_irqsave(&qfbdev->delayed_ops_lock, flags);
list_add_tail(&op->head, &qfbdev->delayed_ops);
spin_unlock_irqrestore(&qfbdev->delayed_ops_lock, flags);
}
static void qxl_fb_delayed_imageblit(struct qxl_fbdev *qfbdev,
const struct fb_image *fb_image)
{
struct qxl_fb_op *op;
unsigned long flags;
uint32_t size = fb_image->width * fb_image->height * (fb_image->depth >= 8 ? fb_image->depth / 8 : 1);
op = kmalloc(sizeof(struct qxl_fb_op) + size, GFP_ATOMIC | __GFP_NOWARN);
if (!op)
return;
op->op.ib = *fb_image;
op->img_data = (void *)(op + 1);
op->op_type = QXL_FB_OP_IMAGEBLIT;
memcpy(op->img_data, fb_image->data, size);
op->op.ib.data = op->img_data;
spin_lock_irqsave(&qfbdev->delayed_ops_lock, flags);
list_add_tail(&op->head, &qfbdev->delayed_ops);
spin_unlock_irqrestore(&qfbdev->delayed_ops_lock, flags);
}
static void qxl_fb_fillrect_internal(struct fb_info *info,
const struct fb_fillrect *fb_rect)
{ {
struct qxl_fbdev *qfbdev = info->par; struct qxl_fbdev *qfbdev = info->par;
struct qxl_device *qdev = qfbdev->qdev; struct qxl_device *qdev = qfbdev->qdev;
@ -203,17 +281,28 @@ static void qxl_fb_fillrect(struct fb_info *info,
qxl_draw_fill_rec.rect = rect; qxl_draw_fill_rec.rect = rect;
qxl_draw_fill_rec.color = color; qxl_draw_fill_rec.color = color;
qxl_draw_fill_rec.rop = rop; qxl_draw_fill_rec.rop = rop;
if (!drm_can_sleep()) {
qxl_io_log(qdev,
"%s: TODO use RCU, mysterious locks with spin_lock\n",
__func__);
return;
}
qxl_draw_fill(&qxl_draw_fill_rec); qxl_draw_fill(&qxl_draw_fill_rec);
} }
static void qxl_fb_copyarea(struct fb_info *info, static void qxl_fb_fillrect(struct fb_info *info,
const struct fb_copyarea *region) const struct fb_fillrect *fb_rect)
{
struct qxl_fbdev *qfbdev = info->par;
struct qxl_device *qdev = qfbdev->qdev;
if (!drm_can_sleep()) {
qxl_fb_delayed_fillrect(qfbdev, fb_rect);
schedule_work(&qdev->fb_work);
return;
}
/* make sure any previous work is done */
flush_work(&qdev->fb_work);
qxl_fb_fillrect_internal(info, fb_rect);
}
static void qxl_fb_copyarea_internal(struct fb_info *info,
const struct fb_copyarea *region)
{ {
struct qxl_fbdev *qfbdev = info->par; struct qxl_fbdev *qfbdev = info->par;
@ -223,37 +312,89 @@ static void qxl_fb_copyarea(struct fb_info *info,
region->dx, region->dy); region->dx, region->dy);
} }
static void qxl_fb_copyarea(struct fb_info *info,
const struct fb_copyarea *region)
{
struct qxl_fbdev *qfbdev = info->par;
struct qxl_device *qdev = qfbdev->qdev;
if (!drm_can_sleep()) {
qxl_fb_delayed_copyarea(qfbdev, region);
schedule_work(&qdev->fb_work);
return;
}
/* make sure any previous work is done */
flush_work(&qdev->fb_work);
qxl_fb_copyarea_internal(info, region);
}
static void qxl_fb_imageblit_safe(struct qxl_fb_image *qxl_fb_image) static void qxl_fb_imageblit_safe(struct qxl_fb_image *qxl_fb_image)
{ {
qxl_draw_opaque_fb(qxl_fb_image, 0); qxl_draw_opaque_fb(qxl_fb_image, 0);
} }
static void qxl_fb_imageblit_internal(struct fb_info *info,
const struct fb_image *image)
{
struct qxl_fbdev *qfbdev = info->par;
struct qxl_fb_image qxl_fb_image;
/* ensure proper order rendering operations - TODO: must do this
* for everything. */
qxl_fb_image_init(&qxl_fb_image, qfbdev->qdev, info, image);
qxl_fb_imageblit_safe(&qxl_fb_image);
}
static void qxl_fb_imageblit(struct fb_info *info, static void qxl_fb_imageblit(struct fb_info *info,
const struct fb_image *image) const struct fb_image *image)
{ {
struct qxl_fbdev *qfbdev = info->par; struct qxl_fbdev *qfbdev = info->par;
struct qxl_device *qdev = qfbdev->qdev; struct qxl_device *qdev = qfbdev->qdev;
struct qxl_fb_image qxl_fb_image;
if (!drm_can_sleep()) { if (!drm_can_sleep()) {
/* we cannot do any ttm_bo allocation since that will fail on qxl_fb_delayed_imageblit(qfbdev, image);
* ioremap_wc..__get_vm_area_node, so queue the work item schedule_work(&qdev->fb_work);
* instead This can happen from printk inside an interrupt
* context, i.e.: smp_apic_timer_interrupt..check_cpu_stall */
qxl_io_log(qdev,
"%s: TODO use RCU, mysterious locks with spin_lock\n",
__func__);
return; return;
} }
/* make sure any previous work is done */
flush_work(&qdev->fb_work);
qxl_fb_imageblit_internal(info, image);
}
/* ensure proper order of rendering operations - TODO: must do this static void qxl_fb_work(struct work_struct *work)
* for everything. */ {
qxl_fb_image_init(&qxl_fb_image, qfbdev->qdev, info, image); struct qxl_device *qdev = container_of(work, struct qxl_device, fb_work);
qxl_fb_imageblit_safe(&qxl_fb_image); unsigned long flags;
struct qxl_fb_op *entry, *tmp;
struct qxl_fbdev *qfbdev = qdev->mode_info.qfbdev;
/* since the irq context just adds entries to the end of the
list dropping the lock should be fine, as entry isn't modified
in the operation code */
spin_lock_irqsave(&qfbdev->delayed_ops_lock, flags);
list_for_each_entry_safe(entry, tmp, &qfbdev->delayed_ops, head) {
spin_unlock_irqrestore(&qfbdev->delayed_ops_lock, flags);
switch (entry->op_type) {
case QXL_FB_OP_FILLRECT:
qxl_fb_fillrect_internal(qfbdev->helper.fbdev, &entry->op.fr);
break;
case QXL_FB_OP_COPYAREA:
qxl_fb_copyarea_internal(qfbdev->helper.fbdev, &entry->op.ca);
break;
case QXL_FB_OP_IMAGEBLIT:
qxl_fb_imageblit_internal(qfbdev->helper.fbdev, &entry->op.ib);
break;
}
spin_lock_irqsave(&qfbdev->delayed_ops_lock, flags);
list_del(&entry->head);
kfree(entry);
}
spin_unlock_irqrestore(&qfbdev->delayed_ops_lock, flags);
} }
int qxl_fb_init(struct qxl_device *qdev) int qxl_fb_init(struct qxl_device *qdev)
{ {
INIT_WORK(&qdev->fb_work, qxl_fb_work);
return 0; return 0;
} }
@ -536,7 +677,8 @@ int qxl_fbdev_init(struct qxl_device *qdev)
qfbdev->qdev = qdev; qfbdev->qdev = qdev;
qdev->mode_info.qfbdev = qfbdev; qdev->mode_info.qfbdev = qfbdev;
qfbdev->helper.funcs = &qxl_fb_helper_funcs; qfbdev->helper.funcs = &qxl_fb_helper_funcs;
spin_lock_init(&qfbdev->delayed_ops_lock);
INIT_LIST_HEAD(&qfbdev->delayed_ops);
ret = drm_fb_helper_init(qdev->ddev, &qfbdev->helper, ret = drm_fb_helper_init(qdev->ddev, &qfbdev->helper,
qxl_num_crtc /* num_crtc - QXL supports just 1 */, qxl_num_crtc /* num_crtc - QXL supports just 1 */,
QXLFB_CONN_LIMIT); QXLFB_CONN_LIMIT);