2 * Epson Blizzard LCD controller driver
4 * Copyright (C) 2004-2005 Nokia Corporation
5 * Authors: Juha Yrjola <juha.yrjola@nokia.com>
6 * Imre Deak <imre.deak@nokia.com>
7 * YUV support: Jussi Laako <jussi.laako@nokia.com>
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU General Public License as published by the
11 * Free Software Foundation; either version 2 of the License, or (at your
12 * option) any later version.
14 * This program is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * General Public License for more details.
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, write to the Free Software Foundation, Inc.,
21 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23 #include <linux/module.h>
26 #include <linux/delay.h>
27 #include <linux/clk.h>
30 #include <plat/omapfb.h>
31 #include <plat/blizzard.h>
35 #define MODULE_NAME "blizzard"
37 #define BLIZZARD_REV_CODE 0x00
38 #define BLIZZARD_CONFIG 0x02
39 #define BLIZZARD_PLL_DIV 0x04
40 #define BLIZZARD_PLL_LOCK_RANGE 0x06
41 #define BLIZZARD_PLL_CLOCK_SYNTH_0 0x08
42 #define BLIZZARD_PLL_CLOCK_SYNTH_1 0x0a
43 #define BLIZZARD_PLL_MODE 0x0c
44 #define BLIZZARD_CLK_SRC 0x0e
45 #define BLIZZARD_MEM_BANK0_ACTIVATE 0x10
46 #define BLIZZARD_MEM_BANK0_STATUS 0x14
47 #define BLIZZARD_PANEL_CONFIGURATION 0x28
48 #define BLIZZARD_HDISP 0x2a
49 #define BLIZZARD_HNDP 0x2c
50 #define BLIZZARD_VDISP0 0x2e
51 #define BLIZZARD_VDISP1 0x30
52 #define BLIZZARD_VNDP 0x32
53 #define BLIZZARD_HSW 0x34
54 #define BLIZZARD_VSW 0x38
55 #define BLIZZARD_DISPLAY_MODE 0x68
56 #define BLIZZARD_INPUT_WIN_X_START_0 0x6c
57 #define BLIZZARD_DATA_SOURCE_SELECT 0x8e
58 #define BLIZZARD_DISP_MEM_DATA_PORT 0x90
59 #define BLIZZARD_DISP_MEM_READ_ADDR0 0x92
60 #define BLIZZARD_POWER_SAVE 0xE6
61 #define BLIZZARD_NDISP_CTRL_STATUS 0xE8
63 /* Data source select */
65 #define BLIZZARD_SRC_WRITE_LCD_BACKGROUND 0x00
66 #define BLIZZARD_SRC_WRITE_LCD_DESTRUCTIVE 0x01
67 #define BLIZZARD_SRC_WRITE_OVERLAY_ENABLE 0x04
68 #define BLIZZARD_SRC_DISABLE_OVERLAY 0x05
70 #define BLIZZARD_SRC_WRITE_LCD 0x00
71 #define BLIZZARD_SRC_BLT_LCD 0x06
73 #define BLIZZARD_COLOR_RGB565 0x01
74 #define BLIZZARD_COLOR_YUV420 0x09
76 #define BLIZZARD_VERSION_S1D13745 0x01 /* Hailstorm */
77 #define BLIZZARD_VERSION_S1D13744 0x02 /* Blizzard */
79 #define BLIZZARD_AUTO_UPDATE_TIME (HZ / 20)
81 /* Reserve 4 request slots for requests in irq context */
82 #define REQ_POOL_SIZE 24
83 #define IRQ_REQ_POOL_SIZE 4
85 #define REQ_FROM_IRQ_POOL 0x01
87 #define REQ_COMPLETE 0
90 struct blizzard_reg_list
{
95 /* These need to be saved / restored separately from the rest. */
96 static const struct blizzard_reg_list blizzard_pll_regs
[] = {
98 .start
= 0x04, /* Don't save PLL ctrl (0x0C) */
102 .start
= 0x0e, /* Clock configuration */
107 static const struct blizzard_reg_list blizzard_gen_regs
[] = {
109 .start
= 0x18, /* SDRAM control */
113 .start
= 0x28, /* LCD Panel configuration */
114 .end
= 0x5a, /* HSSI interface, TV configuration */
118 static u8 blizzard_reg_cache
[0x5a / 2];
120 struct update_param
{
122 int x
, y
, width
, height
;
124 int out_width
, out_height
;
130 struct blizzard_request
{
131 struct list_head entry
;
134 int (*handler
)(struct blizzard_request
*req
);
135 void (*complete
)(void *data
);
139 struct update_param update
;
140 struct completion
*sync
;
145 unsigned long offset
;
148 int out_width
, out_height
;
154 struct blizzard_struct
{
155 enum omapfb_update_mode update_mode
;
156 enum omapfb_update_mode update_mode_before_suspend
;
158 struct timer_list auto_update_timer
;
159 int stop_auto_update
;
160 struct omapfb_update_window auto_update_window
;
162 int vid_nonstd_color
;
172 unsigned te_connected
:1;
173 unsigned vsync_only
:1;
175 struct plane_info plane
[OMAPFB_PLANE_NUM
];
177 struct blizzard_request req_pool
[REQ_POOL_SIZE
];
178 struct list_head pending_req_list
;
179 struct list_head free_req_list
;
180 struct semaphore req_sema
;
183 unsigned long sys_ck_rate
;
184 struct extif_timings reg_timings
, lut_timings
;
186 u32 max_transmit_size
;
187 u32 extif_clk_period
;
189 unsigned long pix_tx_time
;
190 unsigned long line_upd_time
;
192 struct omapfb_device
*fbdev
;
193 struct lcd_ctrl_extif
*extif
;
194 const struct lcd_ctrl
*int_ctrl
;
196 void (*power_up
)(struct device
*dev
);
197 void (*power_down
)(struct device
*dev
);
202 struct lcd_ctrl blizzard_ctrl
;
204 static u8
blizzard_read_reg(u8 reg
)
208 blizzard
.extif
->set_bits_per_cycle(8);
209 blizzard
.extif
->write_command(®
, 1);
210 blizzard
.extif
->read_data(&data
, 1);
215 static void blizzard_write_reg(u8 reg
, u8 val
)
217 blizzard
.extif
->set_bits_per_cycle(8);
218 blizzard
.extif
->write_command(®
, 1);
219 blizzard
.extif
->write_data(&val
, 1);
222 static void blizzard_restart_sdram(void)
226 blizzard_write_reg(BLIZZARD_MEM_BANK0_ACTIVATE
, 0);
228 blizzard_write_reg(BLIZZARD_MEM_BANK0_ACTIVATE
, 1);
229 tmo
= jiffies
+ msecs_to_jiffies(200);
230 while (!(blizzard_read_reg(BLIZZARD_MEM_BANK0_STATUS
) & 0x01)) {
231 if (time_after(jiffies
, tmo
)) {
232 dev_err(blizzard
.fbdev
->dev
,
233 "s1d1374x: SDRAM not ready\n");
240 static void blizzard_stop_sdram(void)
242 blizzard_write_reg(BLIZZARD_MEM_BANK0_ACTIVATE
, 0);
245 /* Wait until the last window was completely written into the controllers
246 * SDRAM and we can start transferring the next window.
248 static void blizzard_wait_line_buffer(void)
250 unsigned long tmo
= jiffies
+ msecs_to_jiffies(30);
252 while (blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
) & (1 << 7)) {
253 if (time_after(jiffies
, tmo
)) {
254 if (printk_ratelimit())
255 dev_err(blizzard
.fbdev
->dev
,
256 "s1d1374x: line buffer not ready\n");
262 /* Wait until the YYC color space converter is idle. */
263 static void blizzard_wait_yyc(void)
265 unsigned long tmo
= jiffies
+ msecs_to_jiffies(30);
267 while (blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
) & (1 << 4)) {
268 if (time_after(jiffies
, tmo
)) {
269 if (printk_ratelimit())
270 dev_err(blizzard
.fbdev
->dev
,
271 "s1d1374x: YYC not ready\n");
277 static void disable_overlay(void)
279 blizzard_write_reg(BLIZZARD_DATA_SOURCE_SELECT
,
280 BLIZZARD_SRC_DISABLE_OVERLAY
);
283 static void set_window_regs(int x_start
, int y_start
, int x_end
, int y_end
,
284 int x_out_start
, int y_out_start
,
285 int x_out_end
, int y_out_end
, int color_mode
,
286 int zoom_off
, int flags
)
294 tmp
[1] = x_start
>> 8;
296 tmp
[3] = y_start
>> 8;
304 tmp
[8] = x_out_start
;
305 tmp
[9] = x_out_start
>> 8;
306 tmp
[10] = y_out_start
;
307 tmp
[11] = y_out_start
>> 8;
309 tmp
[13] = x_out_end
>> 8;
311 tmp
[15] = y_out_end
>> 8;
313 tmp
[16] = color_mode
;
314 if (zoom_off
&& blizzard
.version
== BLIZZARD_VERSION_S1D13745
)
315 tmp
[17] = BLIZZARD_SRC_WRITE_LCD_BACKGROUND
;
316 else if (flags
& OMAPFB_FORMAT_FLAG_ENABLE_OVERLAY
)
317 tmp
[17] = BLIZZARD_SRC_WRITE_OVERLAY_ENABLE
;
319 tmp
[17] = blizzard
.version
== BLIZZARD_VERSION_S1D13744
?
320 BLIZZARD_SRC_WRITE_LCD
:
321 BLIZZARD_SRC_WRITE_LCD_DESTRUCTIVE
;
323 blizzard
.extif
->set_bits_per_cycle(8);
324 cmd
= BLIZZARD_INPUT_WIN_X_START_0
;
325 blizzard
.extif
->write_command(&cmd
, 1);
326 blizzard
.extif
->write_data(tmp
, 18);
329 static void enable_tearsync(int y
, int width
, int height
, int screen_height
,
330 int out_height
, int force_vsync
)
334 b
= blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
);
336 blizzard_write_reg(BLIZZARD_NDISP_CTRL_STATUS
, b
);
338 if (likely(blizzard
.vsync_only
|| force_vsync
)) {
339 blizzard
.extif
->enable_tearsync(1, 0);
343 if (width
* blizzard
.pix_tx_time
< blizzard
.line_upd_time
) {
344 blizzard
.extif
->enable_tearsync(1, 0);
348 if ((width
* blizzard
.pix_tx_time
/ 1000) * height
<
349 (y
+ out_height
) * (blizzard
.line_upd_time
/ 1000)) {
350 blizzard
.extif
->enable_tearsync(1, 0);
354 blizzard
.extif
->enable_tearsync(1, y
+ 1);
357 static void disable_tearsync(void)
361 blizzard
.extif
->enable_tearsync(0, 0);
362 b
= blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
);
364 blizzard_write_reg(BLIZZARD_NDISP_CTRL_STATUS
, b
);
365 b
= blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
);
368 static inline void set_extif_timings(const struct extif_timings
*t
);
370 static inline struct blizzard_request
*alloc_req(void)
373 struct blizzard_request
*req
;
377 down(&blizzard
.req_sema
);
379 req_flags
= REQ_FROM_IRQ_POOL
;
381 spin_lock_irqsave(&blizzard
.req_lock
, flags
);
382 BUG_ON(list_empty(&blizzard
.free_req_list
));
383 req
= list_entry(blizzard
.free_req_list
.next
,
384 struct blizzard_request
, entry
);
385 list_del(&req
->entry
);
386 spin_unlock_irqrestore(&blizzard
.req_lock
, flags
);
388 INIT_LIST_HEAD(&req
->entry
);
389 req
->flags
= req_flags
;
394 static inline void free_req(struct blizzard_request
*req
)
398 spin_lock_irqsave(&blizzard
.req_lock
, flags
);
400 list_del(&req
->entry
);
401 list_add(&req
->entry
, &blizzard
.free_req_list
);
402 if (!(req
->flags
& REQ_FROM_IRQ_POOL
))
403 up(&blizzard
.req_sema
);
405 spin_unlock_irqrestore(&blizzard
.req_lock
, flags
);
408 static void process_pending_requests(void)
412 spin_lock_irqsave(&blizzard
.req_lock
, flags
);
414 while (!list_empty(&blizzard
.pending_req_list
)) {
415 struct blizzard_request
*req
;
416 void (*complete
)(void *);
419 req
= list_entry(blizzard
.pending_req_list
.next
,
420 struct blizzard_request
, entry
);
421 spin_unlock_irqrestore(&blizzard
.req_lock
, flags
);
423 if (req
->handler(req
) == REQ_PENDING
)
426 complete
= req
->complete
;
427 complete_data
= req
->complete_data
;
431 complete(complete_data
);
433 spin_lock_irqsave(&blizzard
.req_lock
, flags
);
436 spin_unlock_irqrestore(&blizzard
.req_lock
, flags
);
439 static void submit_req_list(struct list_head
*head
)
444 spin_lock_irqsave(&blizzard
.req_lock
, flags
);
445 if (likely(!list_empty(&blizzard
.pending_req_list
)))
447 list_splice_init(head
, blizzard
.pending_req_list
.prev
);
448 spin_unlock_irqrestore(&blizzard
.req_lock
, flags
);
451 process_pending_requests();
454 static void request_complete(void *data
)
456 struct blizzard_request
*req
= (struct blizzard_request
*)data
;
457 void (*complete
)(void *);
460 complete
= req
->complete
;
461 complete_data
= req
->complete_data
;
466 complete(complete_data
);
468 process_pending_requests();
472 static int do_full_screen_update(struct blizzard_request
*req
)
477 for (i
= 0; i
< 3; i
++) {
478 struct plane_info
*p
= &blizzard
.plane
[i
];
479 if (!(blizzard
.enabled_planes
& (1 << i
))) {
480 blizzard
.int_ctrl
->enable_plane(i
, 0);
483 dev_dbg(blizzard
.fbdev
->dev
, "pw %d ph %d\n",
484 p
->width
, p
->height
);
485 blizzard
.int_ctrl
->setup_plane(i
,
486 OMAPFB_CHANNEL_OUT_LCD
, p
->offset
,
487 p
->scr_width
, p
->pos_x
, p
->pos_y
,
490 blizzard
.int_ctrl
->enable_plane(i
, 1);
493 dev_dbg(blizzard
.fbdev
->dev
, "sw %d sh %d\n",
494 blizzard
.screen_width
, blizzard
.screen_height
);
495 blizzard_wait_line_buffer();
496 flags
= req
->par
.update
.flags
;
497 if (flags
& OMAPFB_FORMAT_FLAG_TEARSYNC
)
498 enable_tearsync(0, blizzard
.screen_width
,
499 blizzard
.screen_height
,
500 blizzard
.screen_height
,
501 blizzard
.screen_height
,
502 flags
& OMAPFB_FORMAT_FLAG_FORCE_VSYNC
);
506 set_window_regs(0, 0, blizzard
.screen_width
, blizzard
.screen_height
,
507 0, 0, blizzard
.screen_width
, blizzard
.screen_height
,
508 BLIZZARD_COLOR_RGB565
, blizzard
.zoom_on
, flags
);
509 blizzard
.zoom_on
= 0;
511 blizzard
.extif
->set_bits_per_cycle(16);
512 /* set_window_regs has left the register index at the right
513 * place, so no need to set it here.
515 blizzard
.extif
->transfer_area(blizzard
.screen_width
,
516 blizzard
.screen_height
,
517 request_complete
, req
);
521 static int check_1d_intersect(int a1
, int a2
, int b1
, int b2
)
523 if (a2
<= b1
|| b2
<= a1
)
528 /* Setup all planes with an overlapping area with the update window. */
529 static int do_partial_update(struct blizzard_request
*req
, int plane
,
530 int x
, int y
, int w
, int h
,
531 int x_out
, int y_out
, int w_out
, int h_out
,
532 int wnd_color_mode
, int bpp
)
535 int gx1
, gy1
, gx2
, gy2
;
536 int gx1_out
, gy1_out
, gx2_out
, gy2_out
;
540 int have_zoom_for_this_update
= 0;
542 /* Global coordinates, relative to pixel 0,0 of the LCD */
543 gx1
= x
+ blizzard
.plane
[plane
].pos_x
;
544 gy1
= y
+ blizzard
.plane
[plane
].pos_y
;
548 flags
= req
->par
.update
.flags
;
549 if (flags
& OMAPFB_FORMAT_FLAG_DOUBLE
) {
552 gx2_out
= gx1
+ w
* 2;
553 gy2_out
= gy1
+ h
* 2;
555 gx1_out
= x_out
+ blizzard
.plane
[plane
].pos_x
;
556 gy1_out
= y_out
+ blizzard
.plane
[plane
].pos_y
;
557 gx2_out
= gx1_out
+ w_out
;
558 gy2_out
= gy1_out
+ h_out
;
561 for (i
= 0; i
< OMAPFB_PLANE_NUM
; i
++) {
562 struct plane_info
*p
= &blizzard
.plane
[i
];
567 unsigned long offset
;
569 if (!(blizzard
.enabled_planes
& (1 << i
)) ||
570 (wnd_color_mode
&& i
!= plane
)) {
571 blizzard
.int_ctrl
->enable_plane(i
, 0);
574 /* Plane coordinates */
576 /* Plane in which we are doing the update.
577 * Local coordinates are the one in the update
587 /* Check if this plane has an overlapping part */
588 px1
= gx1
- p
->pos_x
;
589 py1
= gy1
- p
->pos_y
;
590 px2
= gx2
- p
->pos_x
;
591 py2
= gy2
- p
->pos_y
;
592 if (px1
>= p
->width
|| py1
>= p
->height
||
593 px2
<= 0 || py2
<= 0) {
594 blizzard
.int_ctrl
->enable_plane(i
, 0);
597 /* Calculate the coordinates for the overlapping
598 * part in the plane's local coordinates.
617 offset
= p
->offset
+ (p
->scr_width
* py1
+ px1
) * p
->bpp
/ 8;
619 /* Window embedded in the plane with a differing
620 * color mode / bpp. Calculate the number of DMA
621 * transfer elements in terms of the plane's bpp.
623 pw
= (pw
+ 1) * bpp
/ p
->bpp
;
625 dev_dbg(blizzard
.fbdev
->dev
,
626 "plane %d offset %#08lx pposx %d pposy %d "
627 "px1 %d py1 %d pw %d ph %d\n",
628 i
, offset
, pposx
, pposy
, px1
, py1
, pw
, ph
);
630 blizzard
.int_ctrl
->setup_plane(i
,
631 OMAPFB_CHANNEL_OUT_LCD
, offset
,
633 pposx
, pposy
, pw
, ph
,
636 blizzard
.int_ctrl
->enable_plane(i
, 1);
639 switch (wnd_color_mode
) {
640 case OMAPFB_COLOR_YUV420
:
641 color_mode
= BLIZZARD_COLOR_YUV420
;
642 /* Currently only the 16 bits/pixel cycle format is
643 * supported on the external interface. Adjust the number
644 * of transfer elements per line for 12bpp format.
649 color_mode
= BLIZZARD_COLOR_RGB565
;
653 blizzard_wait_line_buffer();
654 if (blizzard
.last_color_mode
== BLIZZARD_COLOR_YUV420
)
656 blizzard
.last_color_mode
= color_mode
;
657 if (flags
& OMAPFB_FORMAT_FLAG_TEARSYNC
)
658 enable_tearsync(gy1
, w
, h
,
659 blizzard
.screen_height
,
661 flags
& OMAPFB_FORMAT_FLAG_FORCE_VSYNC
);
665 if ((gx2_out
- gx1_out
) != (gx2
- gx1
) ||
666 (gy2_out
- gy1_out
) != (gy2
- gy1
))
667 have_zoom_for_this_update
= 1;
669 /* 'background' type of screen update (as opposed to 'destructive')
670 can be used to disable scaling if scaling is active */
671 zoom_off
= blizzard
.zoom_on
&& !have_zoom_for_this_update
&&
672 (gx1_out
== 0) && (gx2_out
== blizzard
.screen_width
) &&
673 (gy1_out
== 0) && (gy2_out
== blizzard
.screen_height
) &&
674 (gx1
== 0) && (gy1
== 0);
676 if (blizzard
.zoom_on
&& !have_zoom_for_this_update
&& !zoom_off
&&
677 check_1d_intersect(blizzard
.zoom_area_gx1
, blizzard
.zoom_area_gx2
,
679 check_1d_intersect(blizzard
.zoom_area_gy1
, blizzard
.zoom_area_gy2
,
681 /* Previous screen update was using scaling, current update
682 * is not using it. Additionally, current screen update is
683 * going to overlap with the scaled area. Scaling needs to be
684 * disabled in order to avoid 'magnifying glass' effect.
685 * Dummy setup of background window can be used for this.
687 set_window_regs(0, 0, blizzard
.screen_width
,
688 blizzard
.screen_height
,
689 0, 0, blizzard
.screen_width
,
690 blizzard
.screen_height
,
691 BLIZZARD_COLOR_RGB565
, 1, flags
);
692 blizzard
.zoom_on
= 0;
695 /* remember scaling settings if we have scaled update */
696 if (have_zoom_for_this_update
) {
697 blizzard
.zoom_on
= 1;
698 blizzard
.zoom_area_gx1
= gx1_out
;
699 blizzard
.zoom_area_gx2
= gx2_out
;
700 blizzard
.zoom_area_gy1
= gy1_out
;
701 blizzard
.zoom_area_gy2
= gy2_out
;
704 set_window_regs(gx1
, gy1
, gx2
, gy2
, gx1_out
, gy1_out
, gx2_out
, gy2_out
,
705 color_mode
, zoom_off
, flags
);
707 blizzard
.zoom_on
= 0;
709 blizzard
.extif
->set_bits_per_cycle(16);
710 /* set_window_regs has left the register index at the right
711 * place, so no need to set it here.
713 blizzard
.extif
->transfer_area(w
, h
, request_complete
, req
);
718 static int send_frame_handler(struct blizzard_request
*req
)
720 struct update_param
*par
= &req
->par
.update
;
721 int plane
= par
->plane
;
724 dev_dbg(blizzard
.fbdev
->dev
,
725 "send_frame: x %d y %d w %d h %d "
726 "x_out %d y_out %d w_out %d h_out %d "
727 "color_mode %04x flags %04x planes %01x\n",
728 par
->x
, par
->y
, par
->width
, par
->height
,
729 par
->out_x
, par
->out_y
, par
->out_width
, par
->out_height
,
730 par
->color_mode
, par
->flags
, blizzard
.enabled_planes
);
732 if (par
->flags
& OMAPFB_FORMAT_FLAG_DISABLE_OVERLAY
)
735 if ((blizzard
.enabled_planes
& blizzard
.vid_nonstd_color
) ||
736 (blizzard
.enabled_planes
& blizzard
.vid_scaled
))
737 return do_full_screen_update(req
);
739 return do_partial_update(req
, plane
, par
->x
, par
->y
,
740 par
->width
, par
->height
,
741 par
->out_x
, par
->out_y
,
742 par
->out_width
, par
->out_height
,
743 par
->color_mode
, par
->bpp
);
746 static void send_frame_complete(void *data
)
750 #define ADD_PREQ(_x, _y, _w, _h, _x_out, _y_out, _w_out, _h_out) do { \
752 req->handler = send_frame_handler; \
753 req->complete = send_frame_complete; \
754 req->par.update.plane = plane_idx; \
755 req->par.update.x = _x; \
756 req->par.update.y = _y; \
757 req->par.update.width = _w; \
758 req->par.update.height = _h; \
759 req->par.update.out_x = _x_out; \
760 req->par.update.out_y = _y_out; \
761 req->par.update.out_width = _w_out; \
762 req->par.update.out_height = _h_out; \
763 req->par.update.bpp = bpp; \
764 req->par.update.color_mode = color_mode;\
765 req->par.update.flags = flags; \
766 list_add_tail(&req->entry, req_head); \
769 static void create_req_list(int plane_idx
,
770 struct omapfb_update_window
*win
,
771 struct list_head
*req_head
)
773 struct blizzard_request
*req
;
776 int width
= win
->width
;
777 int height
= win
->height
;
778 int x_out
= win
->out_x
;
779 int y_out
= win
->out_y
;
780 int width_out
= win
->out_width
;
781 int height_out
= win
->out_height
;
785 unsigned int ystart
= y
;
786 unsigned int yspan
= height
;
787 unsigned int ystart_out
= y_out
;
788 unsigned int yspan_out
= height_out
;
790 flags
= win
->format
& ~OMAPFB_FORMAT_MASK
;
791 color_mode
= win
->format
& OMAPFB_FORMAT_MASK
;
792 switch (color_mode
) {
793 case OMAPFB_COLOR_YUV420
:
794 /* Embedded window with different color mode */
796 /* X, Y, height must be aligned at 2, width at 4 pixels */
799 height
= yspan
= height
& ~1;
803 /* Same as the plane color mode */
804 bpp
= blizzard
.plane
[plane_idx
].bpp
;
807 if (width
* height
* bpp
/ 8 > blizzard
.max_transmit_size
) {
808 yspan
= blizzard
.max_transmit_size
/ (width
* bpp
/ 8);
809 yspan_out
= yspan
* height_out
/ height
;
810 ADD_PREQ(x
, ystart
, width
, yspan
, x_out
, ystart_out
,
811 width_out
, yspan_out
);
813 ystart_out
+= yspan_out
;
814 yspan
= height
- yspan
;
815 yspan_out
= height_out
- yspan_out
;
816 flags
&= ~OMAPFB_FORMAT_FLAG_TEARSYNC
;
819 ADD_PREQ(x
, ystart
, width
, yspan
, x_out
, ystart_out
,
820 width_out
, yspan_out
);
823 static void auto_update_complete(void *data
)
825 if (!blizzard
.stop_auto_update
)
826 mod_timer(&blizzard
.auto_update_timer
,
827 jiffies
+ BLIZZARD_AUTO_UPDATE_TIME
);
830 static void blizzard_update_window_auto(unsigned long arg
)
833 struct blizzard_request
*last
;
834 struct omapfb_plane_struct
*plane
;
836 plane
= blizzard
.fbdev
->fb_info
[0]->par
;
837 create_req_list(plane
->idx
,
838 &blizzard
.auto_update_window
, &req_list
);
839 last
= list_entry(req_list
.prev
, struct blizzard_request
, entry
);
841 last
->complete
= auto_update_complete
;
842 last
->complete_data
= NULL
;
844 submit_req_list(&req_list
);
847 int blizzard_update_window_async(struct fb_info
*fbi
,
848 struct omapfb_update_window
*win
,
849 void (*complete_callback
)(void *arg
),
850 void *complete_callback_data
)
853 struct blizzard_request
*last
;
854 struct omapfb_plane_struct
*plane
= fbi
->par
;
856 if (unlikely(blizzard
.update_mode
!= OMAPFB_MANUAL_UPDATE
))
858 if (unlikely(!blizzard
.te_connected
&&
859 (win
->format
& OMAPFB_FORMAT_FLAG_TEARSYNC
)))
862 create_req_list(plane
->idx
, win
, &req_list
);
863 last
= list_entry(req_list
.prev
, struct blizzard_request
, entry
);
865 last
->complete
= complete_callback
;
866 last
->complete_data
= (void *)complete_callback_data
;
868 submit_req_list(&req_list
);
872 EXPORT_SYMBOL(blizzard_update_window_async
);
874 static int update_full_screen(void)
876 return blizzard_update_window_async(blizzard
.fbdev
->fb_info
[0],
877 &blizzard
.auto_update_window
, NULL
, NULL
);
881 static int blizzard_setup_plane(int plane
, int channel_out
,
882 unsigned long offset
, int screen_width
,
883 int pos_x
, int pos_y
, int width
, int height
,
886 struct plane_info
*p
;
889 dev_dbg(blizzard
.fbdev
->dev
,
890 "plane %d ch_out %d offset %#08lx scr_width %d "
891 "pos_x %d pos_y %d width %d height %d color_mode %d\n",
892 plane
, channel_out
, offset
, screen_width
,
893 pos_x
, pos_y
, width
, height
, color_mode
);
895 if ((unsigned)plane
> OMAPFB_PLANE_NUM
)
897 p
= &blizzard
.plane
[plane
];
899 switch (color_mode
) {
900 case OMAPFB_COLOR_YUV422
:
901 case OMAPFB_COLOR_YUY422
:
903 blizzard
.vid_nonstd_color
&= ~(1 << plane
);
905 case OMAPFB_COLOR_YUV420
:
907 blizzard
.vid_nonstd_color
|= 1 << plane
;
909 case OMAPFB_COLOR_RGB565
:
911 blizzard
.vid_nonstd_color
&= ~(1 << plane
);
922 p
->scr_width
= screen_width
;
924 p
->out_width
= width
;
926 p
->out_height
= height
;
928 p
->color_mode
= color_mode
;
933 static int blizzard_set_scale(int plane
, int orig_w
, int orig_h
,
934 int out_w
, int out_h
)
936 struct plane_info
*p
= &blizzard
.plane
[plane
];
939 dev_dbg(blizzard
.fbdev
->dev
,
940 "plane %d orig_w %d orig_h %d out_w %d out_h %d\n",
941 plane
, orig_w
, orig_h
, out_w
, out_h
);
942 if ((unsigned)plane
> OMAPFB_PLANE_NUM
)
945 r
= blizzard
.int_ctrl
->set_scale(plane
, orig_w
, orig_h
, out_w
, out_h
);
951 p
->out_width
= out_w
;
952 p
->out_height
= out_h
;
953 if (orig_w
== out_w
&& orig_h
== out_h
)
954 blizzard
.vid_scaled
&= ~(1 << plane
);
956 blizzard
.vid_scaled
|= 1 << plane
;
961 static int blizzard_set_rotate(int angle
)
965 l
= blizzard_read_reg(BLIZZARD_PANEL_CONFIGURATION
);
985 blizzard_write_reg(BLIZZARD_PANEL_CONFIGURATION
, l
);
990 static int blizzard_enable_plane(int plane
, int enable
)
993 blizzard
.enabled_planes
|= 1 << plane
;
995 blizzard
.enabled_planes
&= ~(1 << plane
);
1000 static int sync_handler(struct blizzard_request
*req
)
1002 complete(req
->par
.sync
);
1003 return REQ_COMPLETE
;
1006 static void blizzard_sync(void)
1008 LIST_HEAD(req_list
);
1009 struct blizzard_request
*req
;
1010 struct completion comp
;
1014 req
->handler
= sync_handler
;
1015 req
->complete
= NULL
;
1016 init_completion(&comp
);
1017 req
->par
.sync
= &comp
;
1019 list_add(&req
->entry
, &req_list
);
1020 submit_req_list(&req_list
);
1022 wait_for_completion(&comp
);
1026 static void blizzard_bind_client(struct omapfb_notifier_block
*nb
)
1028 if (blizzard
.update_mode
== OMAPFB_MANUAL_UPDATE
) {
1029 omapfb_notify_clients(blizzard
.fbdev
, OMAPFB_EVENT_READY
);
1033 static int blizzard_set_update_mode(enum omapfb_update_mode mode
)
1035 if (unlikely(mode
!= OMAPFB_MANUAL_UPDATE
&&
1036 mode
!= OMAPFB_AUTO_UPDATE
&&
1037 mode
!= OMAPFB_UPDATE_DISABLED
))
1040 if (mode
== blizzard
.update_mode
)
1043 dev_info(blizzard
.fbdev
->dev
, "s1d1374x: setting update mode to %s\n",
1044 mode
== OMAPFB_UPDATE_DISABLED
? "disabled" :
1045 (mode
== OMAPFB_AUTO_UPDATE
? "auto" : "manual"));
1047 switch (blizzard
.update_mode
) {
1048 case OMAPFB_MANUAL_UPDATE
:
1049 omapfb_notify_clients(blizzard
.fbdev
, OMAPFB_EVENT_DISABLED
);
1051 case OMAPFB_AUTO_UPDATE
:
1052 blizzard
.stop_auto_update
= 1;
1053 del_timer_sync(&blizzard
.auto_update_timer
);
1055 case OMAPFB_UPDATE_DISABLED
:
1059 blizzard
.update_mode
= mode
;
1061 blizzard
.stop_auto_update
= 0;
1064 case OMAPFB_MANUAL_UPDATE
:
1065 omapfb_notify_clients(blizzard
.fbdev
, OMAPFB_EVENT_READY
);
1067 case OMAPFB_AUTO_UPDATE
:
1068 blizzard_update_window_auto(0);
1070 case OMAPFB_UPDATE_DISABLED
:
1077 static enum omapfb_update_mode
blizzard_get_update_mode(void)
1079 return blizzard
.update_mode
;
1082 static inline void set_extif_timings(const struct extif_timings
*t
)
1084 blizzard
.extif
->set_timings(t
);
1087 static inline unsigned long round_to_extif_ticks(unsigned long ps
, int div
)
1089 int bus_tick
= blizzard
.extif_clk_period
* div
;
1090 return (ps
+ bus_tick
- 1) / bus_tick
* bus_tick
;
1093 static int calc_reg_timing(unsigned long sysclk
, int div
)
1095 struct extif_timings
*t
;
1096 unsigned long systim
;
1098 /* CSOnTime 0, WEOnTime 2 ns, REOnTime 2 ns,
1099 * AccessTime 2 ns + 12.2 ns (regs),
1100 * WEOffTime = WEOnTime + 1 ns,
1101 * REOffTime = REOnTime + 12 ns (regs),
1102 * CSOffTime = REOffTime + 1 ns
1103 * ReadCycle = 2ns + 2*SYSCLK (regs),
1104 * WriteCycle = 2*SYSCLK + 2 ns,
1105 * CSPulseWidth = 10 ns */
1107 systim
= 1000000000 / (sysclk
/ 1000);
1108 dev_dbg(blizzard
.fbdev
->dev
,
1109 "Blizzard systim %lu ps extif_clk_period %u div %d\n",
1110 systim
, blizzard
.extif_clk_period
, div
);
1112 t
= &blizzard
.reg_timings
;
1113 memset(t
, 0, sizeof(*t
));
1118 t
->we_on_time
= round_to_extif_ticks(t
->cs_on_time
+ 2000, div
);
1119 t
->re_on_time
= round_to_extif_ticks(t
->cs_on_time
+ 2000, div
);
1120 t
->access_time
= round_to_extif_ticks(t
->re_on_time
+ 12200, div
);
1121 t
->we_off_time
= round_to_extif_ticks(t
->we_on_time
+ 1000, div
);
1122 t
->re_off_time
= round_to_extif_ticks(t
->re_on_time
+ 13000, div
);
1123 t
->cs_off_time
= round_to_extif_ticks(t
->re_off_time
+ 1000, div
);
1124 t
->we_cycle_time
= round_to_extif_ticks(2 * systim
+ 2000, div
);
1125 if (t
->we_cycle_time
< t
->we_off_time
)
1126 t
->we_cycle_time
= t
->we_off_time
;
1127 t
->re_cycle_time
= round_to_extif_ticks(2 * systim
+ 2000, div
);
1128 if (t
->re_cycle_time
< t
->re_off_time
)
1129 t
->re_cycle_time
= t
->re_off_time
;
1130 t
->cs_pulse_width
= 0;
1132 dev_dbg(blizzard
.fbdev
->dev
, "[reg]cson %d csoff %d reon %d reoff %d\n",
1133 t
->cs_on_time
, t
->cs_off_time
, t
->re_on_time
, t
->re_off_time
);
1134 dev_dbg(blizzard
.fbdev
->dev
, "[reg]weon %d weoff %d recyc %d wecyc %d\n",
1135 t
->we_on_time
, t
->we_off_time
, t
->re_cycle_time
,
1137 dev_dbg(blizzard
.fbdev
->dev
, "[reg]rdaccess %d cspulse %d\n",
1138 t
->access_time
, t
->cs_pulse_width
);
1140 return blizzard
.extif
->convert_timings(t
);
1143 static int calc_lut_timing(unsigned long sysclk
, int div
)
1145 struct extif_timings
*t
;
1146 unsigned long systim
;
1148 /* CSOnTime 0, WEOnTime 2 ns, REOnTime 2 ns,
1149 * AccessTime 2 ns + 4 * SYSCLK + 26 (lut),
1150 * WEOffTime = WEOnTime + 1 ns,
1151 * REOffTime = REOnTime + 4*SYSCLK + 26 ns (lut),
1152 * CSOffTime = REOffTime + 1 ns
1153 * ReadCycle = 2ns + 4*SYSCLK + 26 ns (lut),
1154 * WriteCycle = 2*SYSCLK + 2 ns,
1155 * CSPulseWidth = 10 ns */
1157 systim
= 1000000000 / (sysclk
/ 1000);
1158 dev_dbg(blizzard
.fbdev
->dev
,
1159 "Blizzard systim %lu ps extif_clk_period %u div %d\n",
1160 systim
, blizzard
.extif_clk_period
, div
);
1162 t
= &blizzard
.lut_timings
;
1163 memset(t
, 0, sizeof(*t
));
1168 t
->we_on_time
= round_to_extif_ticks(t
->cs_on_time
+ 2000, div
);
1169 t
->re_on_time
= round_to_extif_ticks(t
->cs_on_time
+ 2000, div
);
1170 t
->access_time
= round_to_extif_ticks(t
->re_on_time
+ 4 * systim
+
1172 t
->we_off_time
= round_to_extif_ticks(t
->we_on_time
+ 1000, div
);
1173 t
->re_off_time
= round_to_extif_ticks(t
->re_on_time
+ 4 * systim
+
1175 t
->cs_off_time
= round_to_extif_ticks(t
->re_off_time
+ 1000, div
);
1176 t
->we_cycle_time
= round_to_extif_ticks(2 * systim
+ 2000, div
);
1177 if (t
->we_cycle_time
< t
->we_off_time
)
1178 t
->we_cycle_time
= t
->we_off_time
;
1179 t
->re_cycle_time
= round_to_extif_ticks(2000 + 4 * systim
+ 26000, div
);
1180 if (t
->re_cycle_time
< t
->re_off_time
)
1181 t
->re_cycle_time
= t
->re_off_time
;
1182 t
->cs_pulse_width
= 0;
1184 dev_dbg(blizzard
.fbdev
->dev
,
1185 "[lut]cson %d csoff %d reon %d reoff %d\n",
1186 t
->cs_on_time
, t
->cs_off_time
, t
->re_on_time
, t
->re_off_time
);
1187 dev_dbg(blizzard
.fbdev
->dev
,
1188 "[lut]weon %d weoff %d recyc %d wecyc %d\n",
1189 t
->we_on_time
, t
->we_off_time
, t
->re_cycle_time
,
1191 dev_dbg(blizzard
.fbdev
->dev
, "[lut]rdaccess %d cspulse %d\n",
1192 t
->access_time
, t
->cs_pulse_width
);
1194 return blizzard
.extif
->convert_timings(t
);
1197 static int calc_extif_timings(unsigned long sysclk
, int *extif_mem_div
)
1202 blizzard
.extif
->get_clk_info(&blizzard
.extif_clk_period
, &max_clk_div
);
1203 for (div
= 1; div
<= max_clk_div
; div
++) {
1204 if (calc_reg_timing(sysclk
, div
) == 0)
1207 if (div
> max_clk_div
) {
1208 dev_dbg(blizzard
.fbdev
->dev
, "reg timing failed\n");
1211 *extif_mem_div
= div
;
1213 for (div
= 1; div
<= max_clk_div
; div
++) {
1214 if (calc_lut_timing(sysclk
, div
) == 0)
1218 if (div
> max_clk_div
)
1221 blizzard
.extif_clk_div
= div
;
1225 dev_err(blizzard
.fbdev
->dev
, "can't setup timings\n");
1229 static void calc_blizzard_clk_rates(unsigned long ext_clk
,
1230 unsigned long *sys_clk
, unsigned long *pix_clk
)
1233 int sys_div
= 0, sys_mul
= 0;
1236 pix_clk_src
= blizzard_read_reg(BLIZZARD_CLK_SRC
);
1237 pix_div
= ((pix_clk_src
>> 3) & 0x1f) + 1;
1238 if ((pix_clk_src
& (0x3 << 1)) == 0) {
1239 /* Source is the PLL */
1240 sys_div
= (blizzard_read_reg(BLIZZARD_PLL_DIV
) & 0x3f) + 1;
1241 sys_mul
= blizzard_read_reg(BLIZZARD_PLL_CLOCK_SYNTH_0
);
1242 sys_mul
|= ((blizzard_read_reg(BLIZZARD_PLL_CLOCK_SYNTH_1
)
1244 *sys_clk
= ext_clk
* sys_mul
/ sys_div
;
1245 } else /* else source is ext clk, or oscillator */
1248 *pix_clk
= *sys_clk
/ pix_div
; /* HZ */
1249 dev_dbg(blizzard
.fbdev
->dev
,
1250 "ext_clk %ld pix_src %d pix_div %d sys_div %d sys_mul %d\n",
1251 ext_clk
, pix_clk_src
& (0x3 << 1), pix_div
, sys_div
, sys_mul
);
1252 dev_dbg(blizzard
.fbdev
->dev
, "sys_clk %ld pix_clk %ld\n",
1253 *sys_clk
, *pix_clk
);
1256 static int setup_tearsync(unsigned long pix_clk
, int extif_div
)
1262 int hs_pol_inv
, vs_pol_inv
;
1263 int use_hsvs
, use_ndp
;
1266 hsw
= blizzard_read_reg(BLIZZARD_HSW
);
1267 vsw
= blizzard_read_reg(BLIZZARD_VSW
);
1268 hs_pol_inv
= !(hsw
& 0x80);
1269 vs_pol_inv
= !(vsw
& 0x80);
1273 hdisp
= blizzard_read_reg(BLIZZARD_HDISP
) * 8;
1274 vdisp
= blizzard_read_reg(BLIZZARD_VDISP0
) +
1275 ((blizzard_read_reg(BLIZZARD_VDISP1
) & 0x3) << 8);
1277 hndp
= blizzard_read_reg(BLIZZARD_HNDP
) & 0x3f;
1278 vndp
= blizzard_read_reg(BLIZZARD_VNDP
);
1280 /* time to transfer one pixel (16bpp) in ps */
1281 blizzard
.pix_tx_time
= blizzard
.reg_timings
.we_cycle_time
;
1282 if (blizzard
.extif
->get_max_tx_rate
!= NULL
) {
1283 /* The external interface might have a rate limitation,
1284 * if so, we have to maximize our transfer rate.
1286 unsigned long min_tx_time
;
1287 unsigned long max_tx_rate
= blizzard
.extif
->get_max_tx_rate();
1289 dev_dbg(blizzard
.fbdev
->dev
, "max_tx_rate %ld HZ\n",
1291 min_tx_time
= 1000000000 / (max_tx_rate
/ 1000); /* ps */
1292 if (blizzard
.pix_tx_time
< min_tx_time
)
1293 blizzard
.pix_tx_time
= min_tx_time
;
1296 /* time to update one line in ps */
1297 blizzard
.line_upd_time
= (hdisp
+ hndp
) * 1000000 / (pix_clk
/ 1000);
1298 blizzard
.line_upd_time
*= 1000;
1299 if (hdisp
* blizzard
.pix_tx_time
> blizzard
.line_upd_time
)
1300 /* transfer speed too low, we might have to use both
1304 /* decent transfer speed, we'll always use only VS */
1307 if (use_hsvs
&& (hs_pol_inv
|| vs_pol_inv
)) {
1308 /* HS or'ed with VS doesn't work, use the active high
1309 * TE signal based on HNDP / VNDP */
1316 /* Use HS or'ed with VS as a TE signal if both are needed
1317 * or VNDP if only vsync is needed. */
1327 hs
= hs
* 1000000 / (pix_clk
/ 1000); /* ps */
1330 vs
= vs
* (hdisp
+ hndp
) * 1000000 / (pix_clk
/ 1000); /* ps */
1335 /* set VS to 120% of HS to minimize VS detection time */
1337 /* minimize HS too */
1341 b
= blizzard_read_reg(BLIZZARD_NDISP_CTRL_STATUS
);
1343 b
|= use_hsvs
? 1 : 0;
1344 b
|= (use_ndp
&& use_hsvs
) ? 0 : 2;
1345 blizzard_write_reg(BLIZZARD_NDISP_CTRL_STATUS
, b
);
1347 blizzard
.vsync_only
= !use_hsvs
;
1349 dev_dbg(blizzard
.fbdev
->dev
,
1350 "pix_clk %ld HZ pix_tx_time %ld ps line_upd_time %ld ps\n",
1351 pix_clk
, blizzard
.pix_tx_time
, blizzard
.line_upd_time
);
1352 dev_dbg(blizzard
.fbdev
->dev
,
1353 "hs %d ps vs %d ps mode %d vsync_only %d\n",
1354 hs
, vs
, b
& 0x3, !use_hsvs
);
1356 return blizzard
.extif
->setup_tearsync(1, hs
, vs
,
1357 hs_pol_inv
, vs_pol_inv
,
1361 static void blizzard_get_caps(int plane
, struct omapfb_caps
*caps
)
1363 blizzard
.int_ctrl
->get_caps(plane
, caps
);
1364 caps
->ctrl
|= OMAPFB_CAPS_MANUAL_UPDATE
|
1365 OMAPFB_CAPS_WINDOW_PIXEL_DOUBLE
|
1366 OMAPFB_CAPS_WINDOW_SCALE
|
1367 OMAPFB_CAPS_WINDOW_OVERLAY
|
1368 OMAPFB_CAPS_WINDOW_ROTATE
;
1369 if (blizzard
.te_connected
)
1370 caps
->ctrl
|= OMAPFB_CAPS_TEARSYNC
;
1371 caps
->wnd_color
|= (1 << OMAPFB_COLOR_RGB565
) |
1372 (1 << OMAPFB_COLOR_YUV420
);
1375 static void _save_regs(const struct blizzard_reg_list
*list
, int cnt
)
1379 for (i
= 0; i
< cnt
; i
++, list
++) {
1381 for (reg
= list
->start
; reg
<= list
->end
; reg
+= 2)
1382 blizzard_reg_cache
[reg
/ 2] = blizzard_read_reg(reg
);
1386 static void _restore_regs(const struct blizzard_reg_list
*list
, int cnt
)
1390 for (i
= 0; i
< cnt
; i
++, list
++) {
1392 for (reg
= list
->start
; reg
<= list
->end
; reg
+= 2)
1393 blizzard_write_reg(reg
, blizzard_reg_cache
[reg
/ 2]);
1397 static void blizzard_save_all_regs(void)
1399 _save_regs(blizzard_pll_regs
, ARRAY_SIZE(blizzard_pll_regs
));
1400 _save_regs(blizzard_gen_regs
, ARRAY_SIZE(blizzard_gen_regs
));
1403 static void blizzard_restore_pll_regs(void)
1405 _restore_regs(blizzard_pll_regs
, ARRAY_SIZE(blizzard_pll_regs
));
1408 static void blizzard_restore_gen_regs(void)
1410 _restore_regs(blizzard_gen_regs
, ARRAY_SIZE(blizzard_gen_regs
));
1413 static void blizzard_suspend(void)
1418 if (blizzard
.last_color_mode
) {
1419 update_full_screen();
1422 blizzard
.update_mode_before_suspend
= blizzard
.update_mode
;
1423 /* the following will disable clocks as well */
1424 blizzard_set_update_mode(OMAPFB_UPDATE_DISABLED
);
1426 blizzard_save_all_regs();
1428 blizzard_stop_sdram();
1430 l
= blizzard_read_reg(BLIZZARD_POWER_SAVE
);
1431 /* Standby, Sleep. We assume we use an external clock. */
1433 blizzard_write_reg(BLIZZARD_POWER_SAVE
, l
);
1435 tmo
= jiffies
+ msecs_to_jiffies(100);
1436 while (!(blizzard_read_reg(BLIZZARD_PLL_MODE
) & (1 << 1))) {
1437 if (time_after(jiffies
, tmo
)) {
1438 dev_err(blizzard
.fbdev
->dev
,
1439 "s1d1374x: sleep timeout, stopping PLL manually\n");
1440 l
= blizzard_read_reg(BLIZZARD_PLL_MODE
);
1442 /* Disable PLL, counter function */
1444 blizzard_write_reg(BLIZZARD_PLL_MODE
, l
);
1450 if (blizzard
.power_down
!= NULL
)
1451 blizzard
.power_down(blizzard
.fbdev
->dev
);
1454 static void blizzard_resume(void)
1458 if (blizzard
.power_up
!= NULL
)
1459 blizzard
.power_up(blizzard
.fbdev
->dev
);
1461 l
= blizzard_read_reg(BLIZZARD_POWER_SAVE
);
1462 /* Standby, Sleep */
1464 blizzard_write_reg(BLIZZARD_POWER_SAVE
, l
);
1466 blizzard_restore_pll_regs();
1467 l
= blizzard_read_reg(BLIZZARD_PLL_MODE
);
1469 /* Enable PLL, counter function */
1471 blizzard_write_reg(BLIZZARD_PLL_MODE
, l
);
1473 while (!(blizzard_read_reg(BLIZZARD_PLL_DIV
) & (1 << 7)))
1476 blizzard_restart_sdram();
1478 blizzard_restore_gen_regs();
1480 /* Enable display */
1481 blizzard_write_reg(BLIZZARD_DISPLAY_MODE
, 0x01);
1483 /* the following will enable clocks as necessary */
1484 blizzard_set_update_mode(blizzard
.update_mode_before_suspend
);
1486 /* Force a background update */
1487 blizzard
.zoom_on
= 1;
1488 update_full_screen();
1492 static int blizzard_init(struct omapfb_device
*fbdev
, int ext_mode
,
1493 struct omapfb_mem_desc
*req_vram
)
1497 unsigned long ext_clk
;
1499 unsigned long sys_clk
, pix_clk
;
1500 struct omapfb_platform_data
*omapfb_conf
;
1501 struct blizzard_platform_data
*ctrl_conf
;
1503 blizzard
.fbdev
= fbdev
;
1505 BUG_ON(!fbdev
->ext_if
|| !fbdev
->int_ctrl
);
1507 blizzard
.fbdev
= fbdev
;
1508 blizzard
.extif
= fbdev
->ext_if
;
1509 blizzard
.int_ctrl
= fbdev
->int_ctrl
;
1511 omapfb_conf
= fbdev
->dev
->platform_data
;
1512 ctrl_conf
= omapfb_conf
->ctrl_platform_data
;
1513 if (ctrl_conf
== NULL
|| ctrl_conf
->get_clock_rate
== NULL
) {
1514 dev_err(fbdev
->dev
, "s1d1374x: missing platform data\n");
1519 blizzard
.power_down
= ctrl_conf
->power_down
;
1520 blizzard
.power_up
= ctrl_conf
->power_up
;
1522 spin_lock_init(&blizzard
.req_lock
);
1524 if ((r
= blizzard
.int_ctrl
->init(fbdev
, 1, req_vram
)) < 0)
1527 if ((r
= blizzard
.extif
->init(fbdev
)) < 0)
1530 blizzard_ctrl
.set_color_key
= blizzard
.int_ctrl
->set_color_key
;
1531 blizzard_ctrl
.get_color_key
= blizzard
.int_ctrl
->get_color_key
;
1532 blizzard_ctrl
.setup_mem
= blizzard
.int_ctrl
->setup_mem
;
1533 blizzard_ctrl
.mmap
= blizzard
.int_ctrl
->mmap
;
1535 ext_clk
= ctrl_conf
->get_clock_rate(fbdev
->dev
);
1536 if ((r
= calc_extif_timings(ext_clk
, &extif_div
)) < 0)
1539 set_extif_timings(&blizzard
.reg_timings
);
1541 if (blizzard
.power_up
!= NULL
)
1542 blizzard
.power_up(fbdev
->dev
);
1544 calc_blizzard_clk_rates(ext_clk
, &sys_clk
, &pix_clk
);
1546 if ((r
= calc_extif_timings(sys_clk
, &extif_div
)) < 0)
1548 set_extif_timings(&blizzard
.reg_timings
);
1550 if (!(blizzard_read_reg(BLIZZARD_PLL_DIV
) & 0x80)) {
1552 "controller not initialized by the bootloader\n");
1557 if (ctrl_conf
->te_connected
) {
1558 if ((r
= setup_tearsync(pix_clk
, extif_div
)) < 0)
1560 blizzard
.te_connected
= 1;
1563 rev
= blizzard_read_reg(BLIZZARD_REV_CODE
);
1564 conf
= blizzard_read_reg(BLIZZARD_CONFIG
);
1566 switch (rev
& 0xfc) {
1568 blizzard
.version
= BLIZZARD_VERSION_S1D13744
;
1569 pr_info("omapfb: s1d13744 LCD controller rev %d "
1570 "initialized (CNF pins %x)\n", rev
& 0x03, conf
& 0x07);
1573 blizzard
.version
= BLIZZARD_VERSION_S1D13745
;
1574 pr_info("omapfb: s1d13745 LCD controller rev %d "
1575 "initialized (CNF pins %x)\n", rev
& 0x03, conf
& 0x07);
1578 dev_err(fbdev
->dev
, "invalid s1d1374x revision %02x\n",
1584 blizzard
.max_transmit_size
= blizzard
.extif
->max_transmit_size
;
1586 blizzard
.update_mode
= OMAPFB_UPDATE_DISABLED
;
1588 blizzard
.auto_update_window
.x
= 0;
1589 blizzard
.auto_update_window
.y
= 0;
1590 blizzard
.auto_update_window
.width
= fbdev
->panel
->x_res
;
1591 blizzard
.auto_update_window
.height
= fbdev
->panel
->y_res
;
1592 blizzard
.auto_update_window
.out_x
= 0;
1593 blizzard
.auto_update_window
.out_x
= 0;
1594 blizzard
.auto_update_window
.out_width
= fbdev
->panel
->x_res
;
1595 blizzard
.auto_update_window
.out_height
= fbdev
->panel
->y_res
;
1596 blizzard
.auto_update_window
.format
= 0;
1598 blizzard
.screen_width
= fbdev
->panel
->x_res
;
1599 blizzard
.screen_height
= fbdev
->panel
->y_res
;
1601 init_timer(&blizzard
.auto_update_timer
);
1602 blizzard
.auto_update_timer
.function
= blizzard_update_window_auto
;
1603 blizzard
.auto_update_timer
.data
= 0;
1605 INIT_LIST_HEAD(&blizzard
.free_req_list
);
1606 INIT_LIST_HEAD(&blizzard
.pending_req_list
);
1607 for (i
= 0; i
< ARRAY_SIZE(blizzard
.req_pool
); i
++)
1608 list_add(&blizzard
.req_pool
[i
].entry
, &blizzard
.free_req_list
);
1609 BUG_ON(i
<= IRQ_REQ_POOL_SIZE
);
1610 sema_init(&blizzard
.req_sema
, i
- IRQ_REQ_POOL_SIZE
);
1614 if (blizzard
.power_down
!= NULL
)
1615 blizzard
.power_down(fbdev
->dev
);
1616 blizzard
.extif
->cleanup();
1618 blizzard
.int_ctrl
->cleanup();
1623 static void blizzard_cleanup(void)
1625 blizzard_set_update_mode(OMAPFB_UPDATE_DISABLED
);
1626 blizzard
.extif
->cleanup();
1627 blizzard
.int_ctrl
->cleanup();
1628 if (blizzard
.power_down
!= NULL
)
1629 blizzard
.power_down(blizzard
.fbdev
->dev
);
1632 struct lcd_ctrl blizzard_ctrl
= {
1634 .init
= blizzard_init
,
1635 .cleanup
= blizzard_cleanup
,
1636 .bind_client
= blizzard_bind_client
,
1637 .get_caps
= blizzard_get_caps
,
1638 .set_update_mode
= blizzard_set_update_mode
,
1639 .get_update_mode
= blizzard_get_update_mode
,
1640 .setup_plane
= blizzard_setup_plane
,
1641 .set_scale
= blizzard_set_scale
,
1642 .enable_plane
= blizzard_enable_plane
,
1643 .set_rotate
= blizzard_set_rotate
,
1644 .update_window
= blizzard_update_window_async
,
1645 .sync
= blizzard_sync
,
1646 .suspend
= blizzard_suspend
,
1647 .resume
= blizzard_resume
,