1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright (C) 2013 Red Hat
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Permission is hereby granted, free of charge, to any person obtaining a
5*4882a593Smuzhiyun * copy of this software and associated documentation files (the "Software"),
6*4882a593Smuzhiyun * to deal in the Software without restriction, including without limitation
7*4882a593Smuzhiyun * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*4882a593Smuzhiyun * and/or sell copies of the Software, and to permit persons to whom the
9*4882a593Smuzhiyun * Software is furnished to do so, subject to the following conditions:
10*4882a593Smuzhiyun *
11*4882a593Smuzhiyun * The above copyright notice and this permission notice (including the next
12*4882a593Smuzhiyun * paragraph) shall be included in all copies or substantial portions of the
13*4882a593Smuzhiyun * Software.
14*4882a593Smuzhiyun *
15*4882a593Smuzhiyun * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*4882a593Smuzhiyun * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*4882a593Smuzhiyun * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*4882a593Smuzhiyun * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*4882a593Smuzhiyun * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20*4882a593Smuzhiyun * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21*4882a593Smuzhiyun * SOFTWARE.
22*4882a593Smuzhiyun */
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #include <linux/slab.h>
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun #include <drm/drm_flip_work.h>
27*4882a593Smuzhiyun #include <drm/drm_print.h>
28*4882a593Smuzhiyun #include <drm/drm_util.h>
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun /**
31*4882a593Smuzhiyun * drm_flip_work_allocate_task - allocate a flip-work task
32*4882a593Smuzhiyun * @data: data associated to the task
33*4882a593Smuzhiyun * @flags: allocator flags
34*4882a593Smuzhiyun *
35*4882a593Smuzhiyun * Allocate a drm_flip_task object and attach private data to it.
36*4882a593Smuzhiyun */
drm_flip_work_allocate_task(void * data,gfp_t flags)37*4882a593Smuzhiyun struct drm_flip_task *drm_flip_work_allocate_task(void *data, gfp_t flags)
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun struct drm_flip_task *task;
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun task = kzalloc(sizeof(*task), flags);
42*4882a593Smuzhiyun if (task)
43*4882a593Smuzhiyun task->data = data;
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun return task;
46*4882a593Smuzhiyun }
47*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_allocate_task);
48*4882a593Smuzhiyun
49*4882a593Smuzhiyun /**
50*4882a593Smuzhiyun * drm_flip_work_queue_task - queue a specific task
51*4882a593Smuzhiyun * @work: the flip-work
52*4882a593Smuzhiyun * @task: the task to handle
53*4882a593Smuzhiyun *
54*4882a593Smuzhiyun * Queues task, that will later be run (passed back to drm_flip_func_t
55*4882a593Smuzhiyun * func) on a work queue after drm_flip_work_commit() is called.
56*4882a593Smuzhiyun */
drm_flip_work_queue_task(struct drm_flip_work * work,struct drm_flip_task * task)57*4882a593Smuzhiyun void drm_flip_work_queue_task(struct drm_flip_work *work,
58*4882a593Smuzhiyun struct drm_flip_task *task)
59*4882a593Smuzhiyun {
60*4882a593Smuzhiyun unsigned long flags;
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun spin_lock_irqsave(&work->lock, flags);
63*4882a593Smuzhiyun list_add_tail(&task->node, &work->queued);
64*4882a593Smuzhiyun spin_unlock_irqrestore(&work->lock, flags);
65*4882a593Smuzhiyun }
66*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_queue_task);
67*4882a593Smuzhiyun
68*4882a593Smuzhiyun /**
69*4882a593Smuzhiyun * drm_flip_work_queue - queue work
70*4882a593Smuzhiyun * @work: the flip-work
71*4882a593Smuzhiyun * @val: the value to queue
72*4882a593Smuzhiyun *
73*4882a593Smuzhiyun * Queues work, that will later be run (passed back to drm_flip_func_t
74*4882a593Smuzhiyun * func) on a work queue after drm_flip_work_commit() is called.
75*4882a593Smuzhiyun */
drm_flip_work_queue(struct drm_flip_work * work,void * val)76*4882a593Smuzhiyun void drm_flip_work_queue(struct drm_flip_work *work, void *val)
77*4882a593Smuzhiyun {
78*4882a593Smuzhiyun struct drm_flip_task *task;
79*4882a593Smuzhiyun
80*4882a593Smuzhiyun task = drm_flip_work_allocate_task(val,
81*4882a593Smuzhiyun drm_can_sleep() ? GFP_KERNEL : GFP_ATOMIC);
82*4882a593Smuzhiyun if (task) {
83*4882a593Smuzhiyun drm_flip_work_queue_task(work, task);
84*4882a593Smuzhiyun } else {
85*4882a593Smuzhiyun DRM_ERROR("%s could not allocate task!\n", work->name);
86*4882a593Smuzhiyun work->func(work, val);
87*4882a593Smuzhiyun }
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_queue);
90*4882a593Smuzhiyun
91*4882a593Smuzhiyun /**
92*4882a593Smuzhiyun * drm_flip_work_commit - commit queued work
93*4882a593Smuzhiyun * @work: the flip-work
94*4882a593Smuzhiyun * @wq: the work-queue to run the queued work on
95*4882a593Smuzhiyun *
96*4882a593Smuzhiyun * Trigger work previously queued by drm_flip_work_queue() to run
97*4882a593Smuzhiyun * on a workqueue. The typical usage would be to queue work (via
98*4882a593Smuzhiyun * drm_flip_work_queue()) at any point (from vblank irq and/or
99*4882a593Smuzhiyun * prior), and then from vblank irq commit the queued work.
100*4882a593Smuzhiyun */
drm_flip_work_commit(struct drm_flip_work * work,struct workqueue_struct * wq)101*4882a593Smuzhiyun void drm_flip_work_commit(struct drm_flip_work *work,
102*4882a593Smuzhiyun struct workqueue_struct *wq)
103*4882a593Smuzhiyun {
104*4882a593Smuzhiyun unsigned long flags;
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun spin_lock_irqsave(&work->lock, flags);
107*4882a593Smuzhiyun list_splice_tail(&work->queued, &work->commited);
108*4882a593Smuzhiyun INIT_LIST_HEAD(&work->queued);
109*4882a593Smuzhiyun spin_unlock_irqrestore(&work->lock, flags);
110*4882a593Smuzhiyun queue_work(wq, &work->worker);
111*4882a593Smuzhiyun }
112*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_commit);
113*4882a593Smuzhiyun
flip_worker(struct work_struct * w)114*4882a593Smuzhiyun static void flip_worker(struct work_struct *w)
115*4882a593Smuzhiyun {
116*4882a593Smuzhiyun struct drm_flip_work *work = container_of(w, struct drm_flip_work, worker);
117*4882a593Smuzhiyun struct list_head tasks;
118*4882a593Smuzhiyun unsigned long flags;
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun while (1) {
121*4882a593Smuzhiyun struct drm_flip_task *task, *tmp;
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun INIT_LIST_HEAD(&tasks);
124*4882a593Smuzhiyun spin_lock_irqsave(&work->lock, flags);
125*4882a593Smuzhiyun list_splice_tail(&work->commited, &tasks);
126*4882a593Smuzhiyun INIT_LIST_HEAD(&work->commited);
127*4882a593Smuzhiyun spin_unlock_irqrestore(&work->lock, flags);
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun if (list_empty(&tasks))
130*4882a593Smuzhiyun break;
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun list_for_each_entry_safe(task, tmp, &tasks, node) {
133*4882a593Smuzhiyun work->func(work, task->data);
134*4882a593Smuzhiyun kfree(task);
135*4882a593Smuzhiyun }
136*4882a593Smuzhiyun }
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun /**
140*4882a593Smuzhiyun * drm_flip_work_init - initialize flip-work
141*4882a593Smuzhiyun * @work: the flip-work to initialize
142*4882a593Smuzhiyun * @name: debug name
143*4882a593Smuzhiyun * @func: the callback work function
144*4882a593Smuzhiyun *
145*4882a593Smuzhiyun * Initializes/allocates resources for the flip-work
146*4882a593Smuzhiyun */
drm_flip_work_init(struct drm_flip_work * work,const char * name,drm_flip_func_t func)147*4882a593Smuzhiyun void drm_flip_work_init(struct drm_flip_work *work,
148*4882a593Smuzhiyun const char *name, drm_flip_func_t func)
149*4882a593Smuzhiyun {
150*4882a593Smuzhiyun work->name = name;
151*4882a593Smuzhiyun INIT_LIST_HEAD(&work->queued);
152*4882a593Smuzhiyun INIT_LIST_HEAD(&work->commited);
153*4882a593Smuzhiyun spin_lock_init(&work->lock);
154*4882a593Smuzhiyun work->func = func;
155*4882a593Smuzhiyun
156*4882a593Smuzhiyun INIT_WORK(&work->worker, flip_worker);
157*4882a593Smuzhiyun }
158*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_init);
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun /**
161*4882a593Smuzhiyun * drm_flip_work_cleanup - cleans up flip-work
162*4882a593Smuzhiyun * @work: the flip-work to cleanup
163*4882a593Smuzhiyun *
164*4882a593Smuzhiyun * Destroy resources allocated for the flip-work
165*4882a593Smuzhiyun */
drm_flip_work_cleanup(struct drm_flip_work * work)166*4882a593Smuzhiyun void drm_flip_work_cleanup(struct drm_flip_work *work)
167*4882a593Smuzhiyun {
168*4882a593Smuzhiyun WARN_ON(!list_empty(&work->queued) || !list_empty(&work->commited));
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun EXPORT_SYMBOL(drm_flip_work_cleanup);
171