aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/video/Kconfig5
-rw-r--r--drivers/video/pxafb.c307
-rw-r--r--drivers/video/pxafb.h12
-rw-r--r--include/asm-arm/arch-pxa/pxafb.h26
-rw-r--r--include/asm-arm/arch-pxa/regs-lcd.h34
5 files changed, 337 insertions, 47 deletions
diff --git a/drivers/video/Kconfig b/drivers/video/Kconfig
index a576dc261732..20a2bbd14ed7 100644
--- a/drivers/video/Kconfig
+++ b/drivers/video/Kconfig
@@ -1774,6 +1774,11 @@ config FB_PXA
1774 1774
1775 If unsure, say N. 1775 If unsure, say N.
1776 1776
1777config FB_PXA_SMARTPANEL
1778 bool "PXA Smartpanel LCD support"
1779 default y
1780 depends on FB_PXA
1781
1777config FB_PXA_PARAMETERS 1782config FB_PXA_PARAMETERS
1778 bool "PXA LCD command line parameters" 1783 bool "PXA LCD command line parameters"
1779 default n 1784 default n
diff --git a/drivers/video/pxafb.c b/drivers/video/pxafb.c
index d11ea78a06d5..a4d656497e9d 100644
--- a/drivers/video/pxafb.c
+++ b/drivers/video/pxafb.c
@@ -40,6 +40,8 @@
40#include <linux/clk.h> 40#include <linux/clk.h>
41#include <linux/err.h> 41#include <linux/err.h>
42#include <linux/completion.h> 42#include <linux/completion.h>
43#include <linux/kthread.h>
44#include <linux/freezer.h>
43 45
44#include <asm/hardware.h> 46#include <asm/hardware.h>
45#include <asm/io.h> 47#include <asm/io.h>
@@ -455,7 +457,7 @@ static int pxafb_mmap(struct fb_info *info,
455 unsigned long off = vma->vm_pgoff << PAGE_SHIFT; 457 unsigned long off = vma->vm_pgoff << PAGE_SHIFT;
456 458
457 if (off < info->fix.smem_len) { 459 if (off < info->fix.smem_len) {
458 vma->vm_pgoff += 1; 460 vma->vm_pgoff += fbi->video_offset / PAGE_SIZE;
459 return dma_mmap_writecombine(fbi->dev, vma, fbi->map_cpu, 461 return dma_mmap_writecombine(fbi->dev, vma, fbi->map_cpu,
460 fbi->map_dma, fbi->map_size); 462 fbi->map_dma, fbi->map_size);
461 } 463 }
@@ -594,6 +596,183 @@ static int setup_frame_dma(struct pxafb_info *fbi, int dma, int pal,
594 return 0; 596 return 0;
595} 597}
596 598
599#ifdef CONFIG_FB_PXA_SMARTPANEL
600static int setup_smart_dma(struct pxafb_info *fbi)
601{
602 struct pxafb_dma_descriptor *dma_desc;
603 unsigned long dma_desc_off, cmd_buff_off;
604
605 dma_desc = &fbi->dma_buff->dma_desc[DMA_CMD];
606 dma_desc_off = offsetof(struct pxafb_dma_buff, dma_desc[DMA_CMD]);
607 cmd_buff_off = offsetof(struct pxafb_dma_buff, cmd_buff);
608
609 dma_desc->fdadr = fbi->dma_buff_phys + dma_desc_off;
610 dma_desc->fsadr = fbi->dma_buff_phys + cmd_buff_off;
611 dma_desc->fidr = 0;
612 dma_desc->ldcmd = fbi->n_smart_cmds * sizeof(uint16_t);
613
614 fbi->fdadr[DMA_CMD] = dma_desc->fdadr;
615 return 0;
616}
617
618int pxafb_smart_flush(struct fb_info *info)
619{
620 struct pxafb_info *fbi = container_of(info, struct pxafb_info, fb);
621 uint32_t prsr;
622 int ret = 0;
623
624 /* disable controller until all registers are set up */
625 lcd_writel(fbi, LCCR0, fbi->reg_lccr0 & ~LCCR0_ENB);
626
627 /* 1. make it an even number of commands to align on 32-bit boundary
628 * 2. add the interrupt command to the end of the chain so we can
629 * keep track of the end of the transfer
630 */
631
632 while (fbi->n_smart_cmds & 1)
633 fbi->smart_cmds[fbi->n_smart_cmds++] = SMART_CMD_NOOP;
634
635 fbi->smart_cmds[fbi->n_smart_cmds++] = SMART_CMD_INTERRUPT;
636 fbi->smart_cmds[fbi->n_smart_cmds++] = SMART_CMD_WAIT_FOR_VSYNC;
637 setup_smart_dma(fbi);
638
639 /* continue to execute next command */
640 prsr = lcd_readl(fbi, PRSR) | PRSR_ST_OK | PRSR_CON_NT;
641 lcd_writel(fbi, PRSR, prsr);
642
643 /* stop the processor in case it executed "wait for sync" cmd */
644 lcd_writel(fbi, CMDCR, 0x0001);
645
646 /* don't send interrupts for fifo underruns on channel 6 */
647 lcd_writel(fbi, LCCR5, LCCR5_IUM(6));
648
649 lcd_writel(fbi, LCCR1, fbi->reg_lccr1);
650 lcd_writel(fbi, LCCR2, fbi->reg_lccr2);
651 lcd_writel(fbi, LCCR3, fbi->reg_lccr3);
652 lcd_writel(fbi, FDADR0, fbi->fdadr[0]);
653 lcd_writel(fbi, FDADR6, fbi->fdadr[6]);
654
655 /* begin sending */
656 lcd_writel(fbi, LCCR0, fbi->reg_lccr0 | LCCR0_ENB);
657
658 if (wait_for_completion_timeout(&fbi->command_done, HZ/2) == 0) {
659 pr_warning("%s: timeout waiting for command done\n",
660 __func__);
661 ret = -ETIMEDOUT;
662 }
663
664 /* quick disable */
665 prsr = lcd_readl(fbi, PRSR) & ~(PRSR_ST_OK | PRSR_CON_NT);
666 lcd_writel(fbi, PRSR, prsr);
667 lcd_writel(fbi, LCCR0, fbi->reg_lccr0 & ~LCCR0_ENB);
668 lcd_writel(fbi, FDADR6, 0);
669 fbi->n_smart_cmds = 0;
670 return ret;
671}
672
673int pxafb_smart_queue(struct fb_info *info, uint16_t *cmds, int n_cmds)
674{
675 int i;
676 struct pxafb_info *fbi = container_of(info, struct pxafb_info, fb);
677
678 /* leave 2 commands for INTERRUPT and WAIT_FOR_SYNC */
679 for (i = 0; i < n_cmds; i++) {
680 if (fbi->n_smart_cmds == CMD_BUFF_SIZE - 8)
681 pxafb_smart_flush(info);
682
683 fbi->smart_cmds[fbi->n_smart_cmds++] = *cmds++;
684 }
685
686 return 0;
687}
688
689static unsigned int __smart_timing(unsigned time_ns, unsigned long lcd_clk)
690{
691 unsigned int t = (time_ns * (lcd_clk / 1000000) / 1000);
692 return (t == 0) ? 1 : t;
693}
694
695static void setup_smart_timing(struct pxafb_info *fbi,
696 struct fb_var_screeninfo *var)
697{
698 struct pxafb_mach_info *inf = fbi->dev->platform_data;
699 struct pxafb_mode_info *mode = &inf->modes[0];
700 unsigned long lclk = clk_get_rate(fbi->clk);
701 unsigned t1, t2, t3, t4;
702
703 t1 = max(mode->a0csrd_set_hld, mode->a0cswr_set_hld);
704 t2 = max(mode->rd_pulse_width, mode->wr_pulse_width);
705 t3 = mode->op_hold_time;
706 t4 = mode->cmd_inh_time;
707
708 fbi->reg_lccr1 =
709 LCCR1_DisWdth(var->xres) |
710 LCCR1_BegLnDel(__smart_timing(t1, lclk)) |
711 LCCR1_EndLnDel(__smart_timing(t2, lclk)) |
712 LCCR1_HorSnchWdth(__smart_timing(t3, lclk));
713
714 fbi->reg_lccr2 = LCCR2_DisHght(var->yres);
715 fbi->reg_lccr3 = LCCR3_PixClkDiv(__smart_timing(t4, lclk));
716
717 /* FIXME: make this configurable */
718 fbi->reg_cmdcr = 1;
719}
720
721static int pxafb_smart_thread(void *arg)
722{
723 struct pxafb_info *fbi = (struct pxafb_info *) arg;
724 struct pxafb_mach_info *inf = fbi->dev->platform_data;
725
726 if (!fbi || !inf->smart_update) {
727 pr_err("%s: not properly initialized, thread terminated\n",
728 __func__);
729 return -EINVAL;
730 }
731
732 pr_debug("%s(): task starting\n", __func__);
733
734 set_freezable();
735 while (!kthread_should_stop()) {
736
737 if (try_to_freeze())
738 continue;
739
740 if (fbi->state == C_ENABLE) {
741 inf->smart_update(&fbi->fb);
742 complete(&fbi->refresh_done);
743 }
744
745 set_current_state(TASK_INTERRUPTIBLE);
746 schedule_timeout(30 * HZ / 1000);
747 }
748
749 pr_debug("%s(): task ending\n", __func__);
750 return 0;
751}
752
753static int pxafb_smart_init(struct pxafb_info *fbi)
754{
755 fbi->smart_thread = kthread_run(pxafb_smart_thread, fbi,
756 "lcd_refresh");
757 if (IS_ERR(fbi->smart_thread)) {
758 printk(KERN_ERR "%s: unable to create kernel thread\n",
759 __func__);
760 return PTR_ERR(fbi->smart_thread);
761 }
762 return 0;
763}
764#else
765int pxafb_smart_queue(struct fb_info *info, uint16_t *cmds, int n_cmds)
766{
767 return 0;
768}
769
770int pxafb_smart_flush(struct fb_info *info)
771{
772 return 0;
773}
774#endif /* CONFIG_FB_SMART_PANEL */
775
597static void setup_parallel_timing(struct pxafb_info *fbi, 776static void setup_parallel_timing(struct pxafb_info *fbi,
598 struct fb_var_screeninfo *var) 777 struct fb_var_screeninfo *var)
599{ 778{
@@ -643,47 +822,55 @@ static int pxafb_activate_var(struct fb_var_screeninfo *var,
643 size_t nbytes; 822 size_t nbytes;
644 823
645#if DEBUG_VAR 824#if DEBUG_VAR
646 if (var->xres < 16 || var->xres > 1024) 825 if (!(fbi->lccr0 & LCCR0_LCDT)) {
647 printk(KERN_ERR "%s: invalid xres %d\n", 826 if (var->xres < 16 || var->xres > 1024)
648 fbi->fb.fix.id, var->xres); 827 printk(KERN_ERR "%s: invalid xres %d\n",
649 switch (var->bits_per_pixel) { 828 fbi->fb.fix.id, var->xres);
650 case 1: 829 switch (var->bits_per_pixel) {
651 case 2: 830 case 1:
652 case 4: 831 case 2:
653 case 8: 832 case 4:
654 case 16: 833 case 8:
655 break; 834 case 16:
656 default: 835 break;
657 printk(KERN_ERR "%s: invalid bit depth %d\n", 836 default:
658 fbi->fb.fix.id, var->bits_per_pixel); 837 printk(KERN_ERR "%s: invalid bit depth %d\n",
659 break; 838 fbi->fb.fix.id, var->bits_per_pixel);
839 break;
840 }
841
842 if (var->hsync_len < 1 || var->hsync_len > 64)
843 printk(KERN_ERR "%s: invalid hsync_len %d\n",
844 fbi->fb.fix.id, var->hsync_len);
845 if (var->left_margin < 1 || var->left_margin > 255)
846 printk(KERN_ERR "%s: invalid left_margin %d\n",
847 fbi->fb.fix.id, var->left_margin);
848 if (var->right_margin < 1 || var->right_margin > 255)
849 printk(KERN_ERR "%s: invalid right_margin %d\n",
850 fbi->fb.fix.id, var->right_margin);
851 if (var->yres < 1 || var->yres > 1024)
852 printk(KERN_ERR "%s: invalid yres %d\n",
853 fbi->fb.fix.id, var->yres);
854 if (var->vsync_len < 1 || var->vsync_len > 64)
855 printk(KERN_ERR "%s: invalid vsync_len %d\n",
856 fbi->fb.fix.id, var->vsync_len);
857 if (var->upper_margin < 0 || var->upper_margin > 255)
858 printk(KERN_ERR "%s: invalid upper_margin %d\n",
859 fbi->fb.fix.id, var->upper_margin);
860 if (var->lower_margin < 0 || var->lower_margin > 255)
861 printk(KERN_ERR "%s: invalid lower_margin %d\n",
862 fbi->fb.fix.id, var->lower_margin);
660 } 863 }
661 if (var->hsync_len < 1 || var->hsync_len > 64)
662 printk(KERN_ERR "%s: invalid hsync_len %d\n",
663 fbi->fb.fix.id, var->hsync_len);
664 if (var->left_margin < 1 || var->left_margin > 255)
665 printk(KERN_ERR "%s: invalid left_margin %d\n",
666 fbi->fb.fix.id, var->left_margin);
667 if (var->right_margin < 1 || var->right_margin > 255)
668 printk(KERN_ERR "%s: invalid right_margin %d\n",
669 fbi->fb.fix.id, var->right_margin);
670 if (var->yres < 1 || var->yres > 1024)
671 printk(KERN_ERR "%s: invalid yres %d\n",
672 fbi->fb.fix.id, var->yres);
673 if (var->vsync_len < 1 || var->vsync_len > 64)
674 printk(KERN_ERR "%s: invalid vsync_len %d\n",
675 fbi->fb.fix.id, var->vsync_len);
676 if (var->upper_margin < 0 || var->upper_margin > 255)
677 printk(KERN_ERR "%s: invalid upper_margin %d\n",
678 fbi->fb.fix.id, var->upper_margin);
679 if (var->lower_margin < 0 || var->lower_margin > 255)
680 printk(KERN_ERR "%s: invalid lower_margin %d\n",
681 fbi->fb.fix.id, var->lower_margin);
682#endif 864#endif
683 /* Update shadow copy atomically */ 865 /* Update shadow copy atomically */
684 local_irq_save(flags); 866 local_irq_save(flags);
685 867
686 setup_parallel_timing(fbi, var); 868#ifdef CONFIG_FB_PXA_SMARTPANEL
869 if (fbi->lccr0 & LCCR0_LCDT)
870 setup_smart_timing(fbi, var);
871 else
872#endif
873 setup_parallel_timing(fbi, var);
687 874
688 fbi->reg_lccr0 = fbi->lccr0 | 875 fbi->reg_lccr0 = fbi->lccr0 |
689 (LCCR0_LDM | LCCR0_SFM | LCCR0_IUM | LCCR0_EFM | 876 (LCCR0_LDM | LCCR0_SFM | LCCR0_IUM | LCCR0_EFM |
@@ -698,7 +885,7 @@ static int pxafb_activate_var(struct fb_var_screeninfo *var,
698 setup_frame_dma(fbi, DMA_LOWER, PAL_NONE, nbytes, nbytes); 885 setup_frame_dma(fbi, DMA_LOWER, PAL_NONE, nbytes, nbytes);
699 } 886 }
700 887
701 if (var->bits_per_pixel >= 16) 888 if ((var->bits_per_pixel >= 16) || (fbi->lccr0 & LCCR0_LCDT))
702 setup_frame_dma(fbi, DMA_BASE, PAL_NONE, 0, nbytes); 889 setup_frame_dma(fbi, DMA_BASE, PAL_NONE, 0, nbytes);
703 else 890 else
704 setup_frame_dma(fbi, DMA_BASE, PAL_BASE, 0, nbytes); 891 setup_frame_dma(fbi, DMA_BASE, PAL_BASE, 0, nbytes);
@@ -801,6 +988,9 @@ static void pxafb_enable_controller(struct pxafb_info *fbi)
801 /* enable LCD controller clock */ 988 /* enable LCD controller clock */
802 clk_enable(fbi->clk); 989 clk_enable(fbi->clk);
803 990
991 if (fbi->lccr0 & LCCR0_LCDT)
992 return;
993
804 /* Sequence from 11.7.10 */ 994 /* Sequence from 11.7.10 */
805 lcd_writel(fbi, LCCR3, fbi->reg_lccr3); 995 lcd_writel(fbi, LCCR3, fbi->reg_lccr3);
806 lcd_writel(fbi, LCCR2, fbi->reg_lccr2); 996 lcd_writel(fbi, LCCR2, fbi->reg_lccr2);
@@ -816,6 +1006,14 @@ static void pxafb_disable_controller(struct pxafb_info *fbi)
816{ 1006{
817 uint32_t lccr0; 1007 uint32_t lccr0;
818 1008
1009#ifdef CONFIG_FB_PXA_SMARTPANEL
1010 if (fbi->lccr0 & LCCR0_LCDT) {
1011 wait_for_completion_timeout(&fbi->refresh_done,
1012 200 * HZ / 1000);
1013 return;
1014 }
1015#endif
1016
819 /* Clear LCD Status Register */ 1017 /* Clear LCD Status Register */
820 lcd_writel(fbi, LCSR, 0xffffffff); 1018 lcd_writel(fbi, LCSR, 0xffffffff);
821 1019
@@ -843,6 +1041,11 @@ static irqreturn_t pxafb_handle_irq(int irq, void *dev_id)
843 complete(&fbi->disable_done); 1041 complete(&fbi->disable_done);
844 } 1042 }
845 1043
1044#ifdef CONFIG_FB_PXA_SMARTPANEL
1045 if (lcsr & LCSR_CMD_INT)
1046 complete(&fbi->command_done);
1047#endif
1048
846 lcd_writel(fbi, LCSR, lcsr); 1049 lcd_writel(fbi, LCSR, lcsr);
847 return IRQ_HANDLED; 1050 return IRQ_HANDLED;
848} 1051}
@@ -1050,15 +1253,17 @@ static int __init pxafb_map_video_memory(struct pxafb_info *fbi)
1050 * We reserve one page for the palette, plus the size 1253 * We reserve one page for the palette, plus the size
1051 * of the framebuffer. 1254 * of the framebuffer.
1052 */ 1255 */
1053 fbi->map_size = PAGE_ALIGN(fbi->fb.fix.smem_len + PAGE_SIZE); 1256 fbi->video_offset = PAGE_ALIGN(sizeof(struct pxafb_dma_buff));
1257 fbi->map_size = PAGE_ALIGN(fbi->fb.fix.smem_len + fbi->video_offset);
1054 fbi->map_cpu = dma_alloc_writecombine(fbi->dev, fbi->map_size, 1258 fbi->map_cpu = dma_alloc_writecombine(fbi->dev, fbi->map_size,
1055 &fbi->map_dma, GFP_KERNEL); 1259 &fbi->map_dma, GFP_KERNEL);
1056 1260
1057 if (fbi->map_cpu) { 1261 if (fbi->map_cpu) {
1058 /* prevent initial garbage on screen */ 1262 /* prevent initial garbage on screen */
1059 memset(fbi->map_cpu, 0, fbi->map_size); 1263 memset(fbi->map_cpu, 0, fbi->map_size);
1060 fbi->fb.screen_base = fbi->map_cpu + PAGE_SIZE; 1264 fbi->fb.screen_base = fbi->map_cpu + fbi->video_offset;
1061 fbi->screen_dma = fbi->map_dma + PAGE_SIZE; 1265 fbi->screen_dma = fbi->map_dma + fbi->video_offset;
1266
1062 /* 1267 /*
1063 * FIXME: this is actually the wrong thing to place in 1268 * FIXME: this is actually the wrong thing to place in
1064 * smem_start. But fbdev suffers from the problem that 1269 * smem_start. But fbdev suffers from the problem that
@@ -1068,9 +1273,14 @@ static int __init pxafb_map_video_memory(struct pxafb_info *fbi)
1068 fbi->fb.fix.smem_start = fbi->screen_dma; 1273 fbi->fb.fix.smem_start = fbi->screen_dma;
1069 fbi->palette_size = fbi->fb.var.bits_per_pixel == 8 ? 256 : 16; 1274 fbi->palette_size = fbi->fb.var.bits_per_pixel == 8 ? 256 : 16;
1070 1275
1071 fbi->dma_buff = (void *)fbi->map_cpu; 1276 fbi->dma_buff = (void *) fbi->map_cpu;
1072 fbi->dma_buff_phys = fbi->map_dma; 1277 fbi->dma_buff_phys = fbi->map_dma;
1073 fbi->palette_cpu = (u16 *)&fbi->dma_buff->palette[0]; 1278 fbi->palette_cpu = (u16 *) fbi->dma_buff->palette;
1279
1280#ifdef CONFIG_FB_PXA_SMARTPANEL
1281 fbi->smart_cmds = (uint16_t *) fbi->dma_buff->cmd_buff;
1282 fbi->n_smart_cmds = 0;
1283#endif
1074 } 1284 }
1075 1285
1076 return fbi->map_cpu ? 0 : -ENOMEM; 1286 return fbi->map_cpu ? 0 : -ENOMEM;
@@ -1191,6 +1401,10 @@ static struct pxafb_info * __init pxafb_init_fbinfo(struct device *dev)
1191 INIT_WORK(&fbi->task, pxafb_task); 1401 INIT_WORK(&fbi->task, pxafb_task);
1192 init_MUTEX(&fbi->ctrlr_sem); 1402 init_MUTEX(&fbi->ctrlr_sem);
1193 init_completion(&fbi->disable_done); 1403 init_completion(&fbi->disable_done);
1404#ifdef CONFIG_FB_PXA_SMARTPANEL
1405 init_completion(&fbi->command_done);
1406 init_completion(&fbi->refresh_done);
1407#endif
1194 1408
1195 return fbi; 1409 return fbi;
1196} 1410}
@@ -1510,6 +1724,13 @@ static int __init pxafb_probe(struct platform_device *dev)
1510 goto failed_free_mem; 1724 goto failed_free_mem;
1511 } 1725 }
1512 1726
1727#ifdef CONFIG_FB_PXA_SMARTPANEL
1728 ret = pxafb_smart_init(fbi);
1729 if (ret) {
1730 dev_err(&dev->dev, "failed to initialize smartpanel\n");
1731 goto failed_free_irq;
1732 }
1733#endif
1513 /* 1734 /*
1514 * This makes sure that our colour bitfield 1735 * This makes sure that our colour bitfield
1515 * descriptors are correctly initialised. 1736 * descriptors are correctly initialised.
diff --git a/drivers/video/pxafb.h b/drivers/video/pxafb.h
index c627b83497da..8238dc826429 100644
--- a/drivers/video/pxafb.h
+++ b/drivers/video/pxafb.h
@@ -52,9 +52,11 @@ enum {
52 52
53/* maximum palette size - 256 entries, each 4 bytes long */ 53/* maximum palette size - 256 entries, each 4 bytes long */
54#define PALETTE_SIZE (256 * 4) 54#define PALETTE_SIZE (256 * 4)
55#define CMD_BUFF_SIZE (1024 * 50)
55 56
56struct pxafb_dma_buff { 57struct pxafb_dma_buff {
57 unsigned char palette[PAL_MAX * PALETTE_SIZE]; 58 unsigned char palette[PAL_MAX * PALETTE_SIZE];
59 uint16_t cmd_buff[CMD_BUFF_SIZE];
58 struct pxafb_dma_descriptor pal_desc[PAL_MAX]; 60 struct pxafb_dma_descriptor pal_desc[PAL_MAX];
59 struct pxafb_dma_descriptor dma_desc[DMA_MAX]; 61 struct pxafb_dma_descriptor dma_desc[DMA_MAX];
60}; 62};
@@ -84,6 +86,7 @@ struct pxafb_info {
84 dma_addr_t screen_dma; /* physical address of frame buffer */ 86 dma_addr_t screen_dma; /* physical address of frame buffer */
85 u16 * palette_cpu; /* virtual address of palette memory */ 87 u16 * palette_cpu; /* virtual address of palette memory */
86 u_int palette_size; 88 u_int palette_size;
89 ssize_t video_offset;
87 90
88 u_int lccr0; 91 u_int lccr0;
89 u_int lccr3; 92 u_int lccr3;
@@ -97,6 +100,7 @@ struct pxafb_info {
97 u_int reg_lccr2; 100 u_int reg_lccr2;
98 u_int reg_lccr3; 101 u_int reg_lccr3;
99 u_int reg_lccr4; 102 u_int reg_lccr4;
103 u_int reg_cmdcr;
100 104
101 unsigned long hsync_time; 105 unsigned long hsync_time;
102 106
@@ -108,6 +112,14 @@ struct pxafb_info {
108 112
109 struct completion disable_done; 113 struct completion disable_done;
110 114
115#ifdef CONFIG_FB_PXA_SMARTPANEL
116 uint16_t *smart_cmds;
117 size_t n_smart_cmds;
118 struct completion command_done;
119 struct completion refresh_done;
120 struct task_struct *smart_thread;
121#endif
122
111#ifdef CONFIG_CPU_FREQ 123#ifdef CONFIG_CPU_FREQ
112 struct notifier_block freq_transition; 124 struct notifier_block freq_transition;
113 struct notifier_block freq_policy; 125 struct notifier_block freq_policy;
diff --git a/include/asm-arm/arch-pxa/pxafb.h b/include/asm-arm/arch-pxa/pxafb.h
index 41a6c2297f9f..bbd22396841a 100644
--- a/include/asm-arm/arch-pxa/pxafb.h
+++ b/include/asm-arm/arch-pxa/pxafb.h
@@ -48,6 +48,7 @@
48#define LCD_COLOR_DSTN_16BPP ((16 << 4) | LCD_TYPE_COLOR_DSTN) 48#define LCD_COLOR_DSTN_16BPP ((16 << 4) | LCD_TYPE_COLOR_DSTN)
49#define LCD_COLOR_TFT_16BPP ((16 << 4) | LCD_TYPE_COLOR_TFT) 49#define LCD_COLOR_TFT_16BPP ((16 << 4) | LCD_TYPE_COLOR_TFT)
50#define LCD_COLOR_TFT_18BPP ((18 << 4) | LCD_TYPE_COLOR_TFT) 50#define LCD_COLOR_TFT_18BPP ((18 << 4) | LCD_TYPE_COLOR_TFT)
51#define LCD_SMART_PANEL_8BPP ((8 << 4) | LCD_TYPE_SMART_PANEL)
51#define LCD_SMART_PANEL_16BPP ((16 << 4) | LCD_TYPE_SMART_PANEL) 52#define LCD_SMART_PANEL_16BPP ((16 << 4) | LCD_TYPE_SMART_PANEL)
52#define LCD_SMART_PANEL_18BPP ((18 << 4) | LCD_TYPE_SMART_PANEL) 53#define LCD_SMART_PANEL_18BPP ((18 << 4) | LCD_TYPE_SMART_PANEL)
53 54
@@ -69,6 +70,10 @@ struct pxafb_mode_info {
69 u_short yres; 70 u_short yres;
70 71
71 u_char bpp; 72 u_char bpp;
73 u_int cmap_greyscale:1,
74 unused:31;
75
76 /* Parallel Mode Timing */
72 u_char hsync_len; 77 u_char hsync_len;
73 u_char left_margin; 78 u_char left_margin;
74 u_char right_margin; 79 u_char right_margin;
@@ -78,8 +83,20 @@ struct pxafb_mode_info {
78 u_char lower_margin; 83 u_char lower_margin;
79 u_char sync; 84 u_char sync;
80 85
81 u_int cmap_greyscale:1, 86 /* Smart Panel Mode Timing - see PXA27x DM 7.4.15.0.3 for details
82 unused:31; 87 * Note:
88 * 1. all parameters in nanosecond (ns)
89 * 2. a0cs{rd,wr}_set_hld are controlled by the same register bits
90 * in pxa27x and pxa3xx, initialize them to the same value or
91 * the larger one will be used
92 * 3. same to {rd,wr}_pulse_width
93 */
94 unsigned a0csrd_set_hld; /* A0 and CS Setup/Hold Time before/after L_FCLK_RD */
95 unsigned a0cswr_set_hld; /* A0 and CS Setup/Hold Time before/after L_PCLK_WR */
96 unsigned wr_pulse_width; /* L_PCLK_WR pulse width */
97 unsigned rd_pulse_width; /* L_FCLK_RD pulse width */
98 unsigned cmd_inh_time; /* Command Inhibit time between two writes */
99 unsigned op_hold_time; /* Output Hold time from L_FCLK_RD negation */
83}; 100};
84 101
85struct pxafb_mach_info { 102struct pxafb_mach_info {
@@ -123,8 +140,11 @@ struct pxafb_mach_info {
123 u_int lccr4; 140 u_int lccr4;
124 void (*pxafb_backlight_power)(int); 141 void (*pxafb_backlight_power)(int);
125 void (*pxafb_lcd_power)(int, struct fb_var_screeninfo *); 142 void (*pxafb_lcd_power)(int, struct fb_var_screeninfo *);
126 143 void (*smart_update)(struct fb_info *);
127}; 144};
128void set_pxa_fb_info(struct pxafb_mach_info *hard_pxa_fb_info); 145void set_pxa_fb_info(struct pxafb_mach_info *hard_pxa_fb_info);
129void set_pxa_fb_parent(struct device *parent_dev); 146void set_pxa_fb_parent(struct device *parent_dev);
130unsigned long pxafb_get_hsync_time(struct device *dev); 147unsigned long pxafb_get_hsync_time(struct device *dev);
148
149extern int pxafb_smart_queue(struct fb_info *info, uint16_t *cmds, int);
150extern int pxafb_smart_flush(struct fb_info *info);
diff --git a/include/asm-arm/arch-pxa/regs-lcd.h b/include/asm-arm/arch-pxa/regs-lcd.h
index f84dd47be28a..f762493f5141 100644
--- a/include/asm-arm/arch-pxa/regs-lcd.h
+++ b/include/asm-arm/arch-pxa/regs-lcd.h
@@ -7,7 +7,8 @@
7#define LCCR1 (0x004) /* LCD Controller Control Register 1 */ 7#define LCCR1 (0x004) /* LCD Controller Control Register 1 */
8#define LCCR2 (0x008) /* LCD Controller Control Register 2 */ 8#define LCCR2 (0x008) /* LCD Controller Control Register 2 */
9#define LCCR3 (0x00C) /* LCD Controller Control Register 3 */ 9#define LCCR3 (0x00C) /* LCD Controller Control Register 3 */
10#define LCCR4 (0x010) /* LCD Controller Control Register 3 */ 10#define LCCR4 (0x010) /* LCD Controller Control Register 4 */
11#define LCCR5 (0x014) /* LCD Controller Control Register 5 */
11#define DFBR0 (0x020) /* DMA Channel 0 Frame Branch Register */ 12#define DFBR0 (0x020) /* DMA Channel 0 Frame Branch Register */
12#define DFBR1 (0x024) /* DMA Channel 1 Frame Branch Register */ 13#define DFBR1 (0x024) /* DMA Channel 1 Frame Branch Register */
13#define LCSR (0x038) /* LCD Controller Status Register */ 14#define LCSR (0x038) /* LCD Controller Status Register */
@@ -15,6 +16,9 @@
15#define TMEDRGBR (0x040) /* TMED RGB Seed Register */ 16#define TMEDRGBR (0x040) /* TMED RGB Seed Register */
16#define TMEDCR (0x044) /* TMED Control Register */ 17#define TMEDCR (0x044) /* TMED Control Register */
17 18
19#define CMDCR (0x100) /* Command Control Register */
20#define PRSR (0x104) /* Panel Read Status Register */
21
18#define LCCR3_1BPP (0 << 24) 22#define LCCR3_1BPP (0 << 24)
19#define LCCR3_2BPP (1 << 24) 23#define LCCR3_2BPP (1 << 24)
20#define LCCR3_4BPP (2 << 24) 24#define LCCR3_4BPP (2 << 24)
@@ -39,6 +43,9 @@
39#define FSADR1 (0x214) /* DMA Channel 1 Frame Source Address Register */ 43#define FSADR1 (0x214) /* DMA Channel 1 Frame Source Address Register */
40#define FIDR1 (0x218) /* DMA Channel 1 Frame ID Register */ 44#define FIDR1 (0x218) /* DMA Channel 1 Frame ID Register */
41#define LDCMD1 (0x21C) /* DMA Channel 1 Command Register */ 45#define LDCMD1 (0x21C) /* DMA Channel 1 Command Register */
46#define FDADR6 (0x260) /* DMA Channel 6 Frame Descriptor Address Register */
47#define FSADR6 (0x264) /* DMA Channel 6 Frame Source Address Register */
48#define FIDR6 (0x268) /* DMA Channel 6 Frame ID Register */
42 49
43#define LCCR0_ENB (1 << 0) /* LCD Controller enable */ 50#define LCCR0_ENB (1 << 0) /* LCD Controller enable */
44#define LCCR0_CMS (1 << 1) /* Color/Monochrome Display Select */ 51#define LCCR0_CMS (1 << 1) /* Color/Monochrome Display Select */
@@ -122,6 +129,11 @@
122#define LCCR3_VrtSnchH (LCCR3_VSP*0) /* VSP Active High */ 129#define LCCR3_VrtSnchH (LCCR3_VSP*0) /* VSP Active High */
123#define LCCR3_VrtSnchL (LCCR3_VSP*1) /* VSP Active Low */ 130#define LCCR3_VrtSnchL (LCCR3_VSP*1) /* VSP Active Low */
124 131
132#define LCCR5_IUM(x) (1 << ((x) + 23)) /* input underrun mask */
133#define LCCR5_BSM(x) (1 << ((x) + 15)) /* branch mask */
134#define LCCR5_EOFM(x) (1 << ((x) + 7)) /* end of frame mask */
135#define LCCR5_SOFM(x) (1 << ((x) + 0)) /* start of frame mask */
136
125#define LCSR_LDD (1 << 0) /* LCD Disable Done */ 137#define LCSR_LDD (1 << 0) /* LCD Disable Done */
126#define LCSR_SOF (1 << 1) /* Start of frame */ 138#define LCSR_SOF (1 << 1) /* Start of frame */
127#define LCSR_BER (1 << 2) /* Bus error */ 139#define LCSR_BER (1 << 2) /* Bus error */
@@ -133,7 +145,27 @@
133#define LCSR_EOF (1 << 8) /* end of frame */ 145#define LCSR_EOF (1 << 8) /* end of frame */
134#define LCSR_BS (1 << 9) /* branch status */ 146#define LCSR_BS (1 << 9) /* branch status */
135#define LCSR_SINT (1 << 10) /* subsequent interrupt */ 147#define LCSR_SINT (1 << 10) /* subsequent interrupt */
148#define LCSR_RD_ST (1 << 11) /* read status */
149#define LCSR_CMD_INT (1 << 12) /* command interrupt */
136 150
137#define LDCMD_PAL (1 << 26) /* instructs DMA to load palette buffer */ 151#define LDCMD_PAL (1 << 26) /* instructs DMA to load palette buffer */
138 152
153/* smartpanel related */
154#define PRSR_DATA(x) ((x) & 0xff) /* Panel Data */
155#define PRSR_A0 (1 << 8) /* Read Data Source */
156#define PRSR_ST_OK (1 << 9) /* Status OK */
157#define PRSR_CON_NT (1 << 10) /* Continue to Next Command */
158
159#define SMART_CMD_A0 (0x1 << 8)
160#define SMART_CMD_READ_STATUS_REG (0x0 << 9)
161#define SMART_CMD_READ_FRAME_BUFFER ((0x0 << 9) | SMART_CMD_A0)
162#define SMART_CMD_WRITE_COMMAND (0x1 << 9)
163#define SMART_CMD_WRITE_DATA ((0x1 << 9) | SMART_CMD_A0)
164#define SMART_CMD_WRITE_FRAME ((0x2 << 9) | SMART_CMD_A0)
165#define SMART_CMD_WAIT_FOR_VSYNC (0x3 << 9)
166#define SMART_CMD_NOOP (0x4 << 9)
167#define SMART_CMD_INTERRUPT (0x5 << 9)
168
169#define SMART_CMD(x) (SMART_CMD_WRITE_COMMAND | ((x) & 0xff))
170#define SMART_DAT(x) (SMART_CMD_WRITE_DATA | ((x) & 0xff))
139#endif /* __ASM_ARCH_REGS_LCD_H */ 171#endif /* __ASM_ARCH_REGS_LCD_H */