aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/video/omap2
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/video/omap2')
-rw-r--r--drivers/video/omap2/dss/dsi.c22
1 files changed, 20 insertions, 2 deletions
diff --git a/drivers/video/omap2/dss/dsi.c b/drivers/video/omap2/dss/dsi.c
index 542c6e2a3ff1..de3fdab46f38 100644
--- a/drivers/video/omap2/dss/dsi.c
+++ b/drivers/video/omap2/dss/dsi.c
@@ -238,6 +238,8 @@ static struct
238 238
239 bool te_enabled; 239 bool te_enabled;
240 240
241 struct workqueue_struct *workqueue;
242
241 struct work_struct framedone_work; 243 struct work_struct framedone_work;
242 void (*framedone_callback)(int, void *); 244 void (*framedone_callback)(int, void *);
243 void *framedone_data; 245 void *framedone_data;
@@ -2759,6 +2761,7 @@ static void dsi_update_screen_dispc(struct omap_dss_device *dssdev,
2759 unsigned packet_payload; 2761 unsigned packet_payload;
2760 unsigned packet_len; 2762 unsigned packet_len;
2761 u32 l; 2763 u32 l;
2764 int r;
2762 const unsigned channel = dsi.update_channel; 2765 const unsigned channel = dsi.update_channel;
2763 /* line buffer is 1024 x 24bits */ 2766 /* line buffer is 1024 x 24bits */
2764 /* XXX: for some reason using full buffer size causes considerable TX 2767 /* XXX: for some reason using full buffer size causes considerable TX
@@ -2809,8 +2812,9 @@ static void dsi_update_screen_dispc(struct omap_dss_device *dssdev,
2809 2812
2810 dsi_perf_mark_start(); 2813 dsi_perf_mark_start();
2811 2814
2812 schedule_delayed_work(&dsi.framedone_timeout_work, 2815 r = queue_delayed_work(dsi.workqueue, &dsi.framedone_timeout_work,
2813 msecs_to_jiffies(250)); 2816 msecs_to_jiffies(250));
2817 BUG_ON(r == 0);
2814 2818
2815 dss_start_update(dssdev); 2819 dss_start_update(dssdev);
2816 2820
@@ -2841,6 +2845,11 @@ static void dsi_framedone_timeout_work_callback(struct work_struct *work)
2841 2845
2842 DSSERR("Framedone not received for 250ms!\n"); 2846 DSSERR("Framedone not received for 250ms!\n");
2843 2847
2848 /* XXX While extremely unlikely, we could get FRAMEDONE interrupt after
2849 * 250ms which would conflict with this timeout work. What should be
2850 * done is first cancel the transfer on the HW, and then cancel the
2851 * possibly scheduled framedone work */
2852
2844 /* SIDLEMODE back to smart-idle */ 2853 /* SIDLEMODE back to smart-idle */
2845 dispc_enable_sidle(); 2854 dispc_enable_sidle();
2846 2855
@@ -2873,6 +2882,7 @@ static void dsi_framedone_timeout_work_callback(struct work_struct *work)
2873 2882
2874static void dsi_framedone_irq_callback(void *data, u32 mask) 2883static void dsi_framedone_irq_callback(void *data, u32 mask)
2875{ 2884{
2885 int r;
2876 /* Note: We get FRAMEDONE when DISPC has finished sending pixels and 2886 /* Note: We get FRAMEDONE when DISPC has finished sending pixels and
2877 * turns itself off. However, DSI still has the pixels in its buffers, 2887 * turns itself off. However, DSI still has the pixels in its buffers,
2878 * and is sending the data. 2888 * and is sending the data.
@@ -2881,7 +2891,8 @@ static void dsi_framedone_irq_callback(void *data, u32 mask)
2881 /* SIDLEMODE back to smart-idle */ 2891 /* SIDLEMODE back to smart-idle */
2882 dispc_enable_sidle(); 2892 dispc_enable_sidle();
2883 2893
2884 schedule_work(&dsi.framedone_work); 2894 r = queue_work(dsi.workqueue, &dsi.framedone_work);
2895 BUG_ON(r == 0);
2885} 2896}
2886 2897
2887static void dsi_handle_framedone(void) 2898static void dsi_handle_framedone(void)
@@ -3292,6 +3303,10 @@ int dsi_init(struct platform_device *pdev)
3292 mutex_init(&dsi.lock); 3303 mutex_init(&dsi.lock);
3293 sema_init(&dsi.bus_lock, 1); 3304 sema_init(&dsi.bus_lock, 1);
3294 3305
3306 dsi.workqueue = create_singlethread_workqueue("dsi");
3307 if (dsi.workqueue == NULL)
3308 return -ENOMEM;
3309
3295 INIT_WORK(&dsi.framedone_work, dsi_framedone_work_callback); 3310 INIT_WORK(&dsi.framedone_work, dsi_framedone_work_callback);
3296 INIT_DELAYED_WORK_DEFERRABLE(&dsi.framedone_timeout_work, 3311 INIT_DELAYED_WORK_DEFERRABLE(&dsi.framedone_timeout_work,
3297 dsi_framedone_timeout_work_callback); 3312 dsi_framedone_timeout_work_callback);
@@ -3328,6 +3343,7 @@ int dsi_init(struct platform_device *pdev)
3328err2: 3343err2:
3329 iounmap(dsi.base); 3344 iounmap(dsi.base);
3330err1: 3345err1:
3346 destroy_workqueue(dsi.workqueue);
3331 return r; 3347 return r;
3332} 3348}
3333 3349
@@ -3335,6 +3351,8 @@ void dsi_exit(void)
3335{ 3351{
3336 iounmap(dsi.base); 3352 iounmap(dsi.base);
3337 3353
3354 destroy_workqueue(dsi.workqueue);
3355
3338 DSSDBG("omap_dsi_exit\n"); 3356 DSSDBG("omap_dsi_exit\n");
3339} 3357}
3340 3358