1
0
mirror of git://projects.qi-hardware.com/openwrt-xburst.git synced 2025-01-19 12:51:07 +02:00

2994 lines
89 KiB
C
Raw Normal View History

/*
* linux/drivers/video/jz4760_lcd.c -- Ingenic Jz4760 LCD frame buffer device
*
* Copyright (C) 2005-2008, Ingenic Semiconductor Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/*
* --------------------------------
* NOTE:
* This LCD driver support TFT16 TFT32 LCD, not support STN and Special TFT LCD
* now.
* It seems not necessory to support STN and Special TFT.
* If it's necessary, update this driver in the future.
* <Wolfgang Wang, Jun 10 2008>
*/
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/string.h>
#include <linux/mm.h>
#include <linux/tty.h>
#include <linux/slab.h>
#include <linux/delay.h>
#include <linux/fb.h>
#include <linux/init.h>
#include <linux/dma-mapping.h>
#include <linux/platform_device.h>
#include <linux/suspend.h>
#include <linux/pm.h>
#include <linux/leds.h>
#include <asm/irq.h>
#include <asm/pgtable.h>
#include <asm/system.h>
#include <asm/uaccess.h>
#include <asm/processor.h>
#include <asm/jzsoc.h>
#include "console/fbcon.h"
#include "jz4760_lcd.h"
//#include "jz4760_tve.h"
#define DRIVER_NAME "jz-lcd"
#ifdef CONFIG_JZ4760_SLCD_KGM701A3_TFT_SPFD5420A
#include "jz_kgm_spfd5420a.h"
#endif
MODULE_DESCRIPTION("Jz4760 LCD Controller driver");
MODULE_AUTHOR("Wolfgang Wang <lgwang@ingenic.cn>, Lemon Liu <zyliu@ingenic.cn>");
MODULE_LICENSE("GPL");
//#define LCD_DEBUG
#undef LCD_DEBUG
#ifdef LCD_DEBUG
#define dprintk(x...) printk(x)
#define print_dbg(f, arg...) printk("dbg::" __FILE__ ",LINE(%d): " f "\n", __LINE__, ## arg)
#else
#define dprintk(x...)
#define print_dbg(f, arg...) do {} while (0)
#endif
#define print_err(f, arg...) printk(KERN_ERR DRIVER_NAME ": " f "\n", ## arg)
#define print_warn(f, arg...) printk(KERN_WARNING DRIVER_NAME ": " f "\n", ## arg)
#define print_info(f, arg...) printk(KERN_INFO DRIVER_NAME ": " f "\n", ## arg)
#define JZ_LCD_ID "jz-lcd"
#define ANDROID_NUMBER_OF_BUFFERS 2
struct lcd_cfb_info {
struct fb_info fb0; /* foreground 0 */
struct fb_info fb; /* foreground 1 */
struct display_switch *dispsw;
signed int currcon;
int func_use_count;
struct {
u16 red, green, blue;
} palette[NR_PALETTE];
#ifdef CONFIG_PM
struct pm_dev *pm;
#endif
};
static struct lcd_cfb_info *jz4760fb_info;
static int current_dma0_id, current_dma1_id;
static struct jz4760_lcd_dma_desc *dma_desc_base;
static struct jz4760_lcd_dma_desc *dma0_desc_palette, *dma0_desc0, *dma0_desc1, *dma1_desc0, *dma1_desc1;
static struct jz4760_lcd_dma_desc *dma0_desc0_change, *dma1_desc0_change, *dma0_desc1_change, *dma1_desc1_change;
#define DMA_DESC_NUM 9
static unsigned char *lcd_palette;
static unsigned char *lcd_frame0;
static unsigned char *lcd_frame;
/* APP */
static void jz4760fb_set_mode(struct jz4760lcd_osd_t * lcd_osd_info );
static void jz4760fb_deep_set_mode( struct jz4760lcd_info * lcd_info );
static void print_lcdc_registers(void);
#ifdef CONFIG_FB_JZ4760_TVE
static void jz4760lcd_info_switch_to_TVE(int mode);
static void jz4760lcd_info_switch_to_lcd(void);
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
static int jz4760fb0_foreground_resize(struct jz4760lcd_osd_t *lcd_osd_info);
static int jz4760fb0_foreground_move(struct jz4760lcd_osd_t *lcd_osd_info);
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
static int jz4760fb_foreground_resize(struct jz4760lcd_osd_t *lcd_osd_info);
static int jz4760fb_foreground_move(struct jz4760lcd_osd_t *lcd_osd_info);
#endif
struct jz4760lcd_info jz4760_lcd_panel = {
#if defined(CONFIG_JZ4760_LCD_SAMSUNG_LTP400WQF02)
.panel = {
.cfg = LCD_CFG_LCDPIN_LCD | LCD_CFG_RECOVER | /* Underrun recover */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_GENERIC_TFT | /* General TFT panel */
LCD_CFG_MODE_TFT_18BIT | /* output 18bpp */
LCD_CFG_HSP | /* Hsync polarity: active low */
LCD_CFG_VSP, /* Vsync polarity: leading edge is falling edge */
.slcd_cfg = 0,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
480, 272, 60, 41, 10, 2, 2, 2, 2,
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN, /* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 480, 272}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 480, 272}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_LCD_AUO_A043FL01V2)
.panel = {
.cfg = LCD_CFG_LCDPIN_LCD | LCD_CFG_RECOVER | /* Underrun recover */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_GENERIC_TFT | /* General TFT panel */
LCD_CFG_MODE_TFT_24BIT | /* output 18bpp */
LCD_CFG_HSP | /* Hsync polarity: active low */
LCD_CFG_VSP, /* Vsync polarity: leading edge is falling edge */
.slcd_cfg = 0,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
480, 272, 60, 41, 10, 8, 4, 4, 2,
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN | LCD_OSDC_ALPHAEN,// | /* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0x80000000, /* disable colorkey */
// .colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xff, /* alpha value */
// .alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {16, 0, 0, 480, 272}, /* bpp, x, y, w, h */
.fg1 = {16, 0, 0, 480, 272}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_LCD_TRULY_TFT_GG1P0319LTSW_W)
.panel = {
.cfg = LCD_CFG_LCDPIN_SLCD | /* Underrun recover*/
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_SLCD, /* TFT Smart LCD panel */
.slcd_cfg = SLCD_CFG_DWIDTH_16BIT | SLCD_CFG_CWIDTH_16BIT | SLCD_CFG_CS_ACTIVE_LOW | SLCD_CFG_RS_CMD_LOW | SLCD_CFG_CLK_ACTIVE_FALLING | SLCD_CFG_TYPE_PARALLEL,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
240, 320, 60, 0, 0, 0, 0, 0, 0,
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN,/* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 240, 320}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 240, 320}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_LCD_FOXCONN_PT035TN01)
.panel = {
.cfg = LCD_CFG_LCDPIN_LCD | LCD_CFG_RECOVER | /* Underrun recover */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_GENERIC_TFT | /* General TFT panel */
LCD_CFG_MODE_TFT_24BIT | /* output 24bpp */
LCD_CFG_HSP | /* Hsync polarity: active low */
LCD_CFG_VSP | /* Vsync polarity: leading edge is falling edge */
LCD_CFG_PCP, /* Pix-CLK polarity: data translations at falling edge */
.slcd_cfg = 0,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
320, 240, 80, 1, 1, 10, 50, 10, 13
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN, /* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 320, 240}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 320, 240}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_LCD_INNOLUX_PT035TN01_SERIAL)
.panel = {
.cfg = LCD_CFG_LCDPIN_LCD | LCD_CFG_RECOVER | /* Underrun recover */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_SERIAL_TFT | /* Serial TFT panel */
LCD_CFG_MODE_TFT_18BIT | /* output 18bpp */
LCD_CFG_HSP | /* Hsync polarity: active low */
LCD_CFG_VSP | /* Vsync polarity: leading edge is falling edge */
LCD_CFG_PCP, /* Pix-CLK polarity: data translations at falling edge */
.slcd_cfg = 0,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
320, 240, 60, 1, 1, 10, 50, 10, 13
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN, /* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 320, 240}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 320, 240}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_SLCD_KGM701A3_TFT_SPFD5420A)
.panel = {
.cfg = LCD_CFG_LCDPIN_SLCD | /* Underrun recover*/
// LCD_CFG_DITHER | /* dither */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_SLCD, /* TFT Smart LCD panel */
.slcd_cfg = SLCD_CFG_DWIDTH_18BIT | SLCD_CFG_CWIDTH_18BIT | SLCD_CFG_CS_ACTIVE_LOW | SLCD_CFG_RS_CMD_LOW | SLCD_CFG_CLK_ACTIVE_FALLING | SLCD_CFG_TYPE_PARALLEL,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
400, 240, 60, 0, 0, 0, 0, 0, 0,
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN, /* Use OSD mode */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 400, 240}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 400, 240}, /* bpp, x, y, w, h */
},
#elif defined(CONFIG_JZ4760_VGA_DISPLAY)
.panel = {
.cfg = LCD_CFG_LCDPIN_LCD | LCD_CFG_RECOVER |/* Underrun recover */
LCD_CFG_NEWDES | /* 8words descriptor */
LCD_CFG_MODE_GENERIC_TFT | /* General TFT panel */
LCD_CFG_MODE_TFT_24BIT | /* output 18bpp */
LCD_CFG_HSP | /* Hsync polarity: active low */
LCD_CFG_VSP, /* Vsync polarity: leading edge is falling edge */
.slcd_cfg = 0,
.ctrl = LCD_CTRL_OFUM | LCD_CTRL_BST_16, /* 16words burst, enable out FIFO underrun irq */
// 800, 600, 60, 128, 4, 40, 88, 0, 23
640, 480, 54, 96, 2, 16, 48, 10, 33
// 1280, 720, 50, 152, 15, 22, 200, 14, 1
},
.osd = {
.osd_cfg = LCD_OSDC_OSDEN | /* Use OSD mode */
// LCD_OSDC_ALPHAEN | /* enable alpha */
// LCD_OSDC_F1EN | /* enable Foreground1 */
LCD_OSDC_F0EN, /* enable Foreground0 */
.osd_ctrl = 0, /* disable ipu, */
.rgb_ctrl = 0,
.bgcolor = 0x000000, /* set background color Black */
.colorkey0 = 0, /* disable colorkey */
.colorkey1 = 0, /* disable colorkey */
.alpha = 0xA0, /* alpha value */
.ipu_restart = 0x80001000, /* ipu restart */
.fg_change = FG_CHANGE_ALL, /* change all initially */
.fg0 = {32, 0, 0, 640, 480}, /* bpp, x, y, w, h */
.fg1 = {32, 0, 0, 640, 480}, /* bpp, x, y, w, h */
},
#else
#error "Select LCD panel first!!!"
#endif
};
#ifdef CONFIG_FB_JZ4760_TVE
struct jz4760lcd_info jz4760_info_tve = {
.panel = {
.w = TVE_WIDTH_PAL, TVE_HEIGHT_PAL, TVE_FREQ_PAL, 0, 0, 0, 0, 0, 0,
},
.osd = {
.rgb_ctrl = LCD_RGBC_YCC, /* enable RGB => YUV */
.fg0 = {32,}, /* */
.fg1 = {32,},
},
};
#endif
struct jz4760lcd_info *jz4760_lcd_info = &jz4760_lcd_panel; /* default output to lcd panel */
static inline u_int chan_to_field(u_int chan, struct fb_bitfield *bf)
{
chan &= 0xffff;
chan >>= 16 - bf->length;
return chan << bf->offset;
}
static void display_v_color_bar(void *frame, int w, int h, int bpp) {
int i, j, wpl, bpl, data = 0;
int *ptr;
printk("frame_v = 0x%08x\n", (unsigned int)frame);
if(w == 0 || h == 0)
return;
ptr = (int *)frame;
wpl = w*bpp/32;
bpl = w*bpp/8;
if (!(bpp > 8))
switch(bpp){
case 1:
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
*ptr++ = 0x00ff00ff;
}
break;
case 2:
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
data = (i%4)*0x55555555;
*ptr++ = data;
}
break;
case 4:
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
data = (i%16)*0x11111111;
*ptr++ = data;
}
break;
case 8:
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i+=2) {
data = (i%(256))*0x01010101;
*ptr++ = data;
*ptr++ = data;
}
break;
}
else {
switch(bpp) {
case 16:
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
if((i/4)%8==0)
*ptr++ = 0xffffffff;
else if ((i/4)%8==1)
*ptr++ = 0xf800f800;
else if ((i/4)%8==2)
*ptr++ = 0xffe0ffe0;
else if ((i/4)%8==3)
*ptr++ = 0x07e007e0;
else if ((i/4)%8==4)
*ptr++ = 0x07ff07ff;
else if ((i/4)%8==5)
*ptr++ = 0x001f001f;
else if ((i/4)%8==6)
*ptr++ = 0xf81ff81f;
else if ((i/4)%8==7)
*ptr++ = 0x00000000;
}
break;
case 18:
case 24:
case 32:
default:
#ifndef CONFIG_FB_JZ4760_LCD_USE_COMPRESS_24BPP
#if 1
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
if((i/8)%8==7)
*ptr++ = 0xffffff;
else if ((i/8)%8==1)
*ptr++ = 0xff0000;
else if ((i/8)%8==2)
*ptr++ = 0xffff00;
else if ((i/8)%8==3)
*ptr++ = 0x00ff00;
else if ((i/8)%8==4)
*ptr++ = 0x00ffff;
else if ((i/8)%8==5)
*ptr++ = 0x0000ff;
else if ((i/8)%8==6)
*ptr++ = 0xff00ff;
else if ((i/8)%8==0)
*ptr++ = 0x000000;
}
#else
for (j = 0;j < h; j++)
for (i = 0;i < wpl; i++) {
if((i/8)%8==7)
*ptr++ = 0x00ff0000;
else if ((i/8)%8==1)
*ptr++ = 0xffff0000;
else if ((i/8)%8==2)
*ptr++ = 0x20ff0000;
else if ((i/8)%8==3)
*ptr++ = 0x40ff0000;
else if ((i/8)%8==4)
*ptr++ = 0x60ff0000;
else if ((i/8)%8==5)
*ptr++ = 0x80ff0000;
else if ((i/8)%8==6)
*ptr++ = 0xa0ff0000;
else if ((i/8)%8==0)
*ptr++ = 0xc0ff0000;
}
#endif
#else
#if 1
ptrc = (unsigned char *)lcd_frame1;
for (j = 0;j < h; j++)
for (i = 0;i < bpl; i+=3) {
if (i < bpl/8*1 ) {
*ptrc++ = 0xff;
*ptrc++ = 0x0;
*ptrc++ = 0x0;
}
else if (i < bpl/8*2) {
*ptrc++ = 0x00;
*ptrc++ = 0xff;
*ptrc++ = 0x00;
}
else if (i < bpl/8*3) {
*ptrc++ = 0x00;
*ptrc++ = 0x00;
*ptrc++ = 0xff;
}
else if (i < bpl/8*4) {
*ptrc++ = 0xff;
*ptrc++ = 0x00;
*ptrc++ = 0x00;
}
else if (i < bpl/8*5) {
*ptrc++ = 0x00;
*ptrc++ = 0xff;
*ptrc++ = 0x00;
}
else if (i < bpl/8*6) {
*ptrc++ = 0x00;
*ptrc++ = 0x00;
*ptrc++ = 0xff;
}
else if (i < bpl/8*7) {
*ptrc++ = 0xff;
*ptrc++ = 0x00;
*ptrc++ = 0x00;
}
else if (i < bpl) {
*ptrc++ = 0x00;
*ptrc++ = 0xff;
*ptrc++ = 0x00;
}
}
#endif
#endif
break;
}
}
dma_cache_wback_inv((unsigned int)(frame), w*h*bpp/8);
}
static void display_h_color_bar(void *frame, int w, int h, int bpp) {
int i, data = 0;
int *ptr;
int wpl; //word_per_line
printk("frame_h = 0x%08x\n", (unsigned int)frame);
if(w == 0 || h == 0)
return;
ptr = (int *)frame;
wpl = w*bpp/32;
if (!(bpp > 8))
for (i = 0;i < wpl*h;i++) {
switch(bpp){
case 1:
if(i%(wpl*8)==0)
data = ((i/(wpl*8))%2)*0xffffffff;
*ptr++ = data;
break;
case 2:
if(i%(wpl*8)==0)
data = ((i/(wpl*8))%4)*0x55555555;
*ptr++ = data;
break;
case 4:
if(i%(wpl*8)==0)
data = ((i/(wpl*8))%16)*0x11111111;
*ptr++ = data;
break;
case 8:
if(i%(wpl*8)==0)
data = ((i/(wpl*8))%256)*0x01010101;
*ptr++ = data;
break;
}
}
else {
switch(bpp) {
case 15:
case 16:
for (i = 0;i < wpl*h;i++) {
if (((i/(wpl*8)) % 8) == 0)
*ptr++ = 0xffffffff;
else if (((i/(wpl*8)) % 8) == 1)
*ptr++ = 0xf800f800;
else if (((i/(wpl*8)) % 8) == 2)
*ptr++ = 0xffe0ffe0;
else if (((i/(wpl*8)) % 8) == 3)
*ptr++ = 0x07e007e0;
else if (((i/(wpl*8)) % 8) == 4)
*ptr++ = 0x07ff07ff;
else if (((i/(wpl*8)) % 8) == 5)
*ptr++ = 0x001f001f;
else if (((i/(wpl*8)) % 8) == 6)
*ptr++ = 0xf81ff81f;
else if (((i/(wpl*8)) % 8) == 7)
*ptr++ = 0x00000000;
}
break;
case 18:
case 24:
case 32:
default:
for (i = 0;i < wpl*h;i++) {
if (((i/(wpl*8)) % 8) == 7)
*ptr++ = 0xffffff;
else if (((i/(wpl*8)) % 8) == 2)
*ptr++ = 0xff0000;
else if (((i/(wpl*8)) % 8) == 4)
*ptr++ = 0xffff00;
else if (((i/(wpl*8)) % 8) == 6)
*ptr++ = 0x00ff00;
else if (((i/(wpl*8)) % 8) == 1)
*ptr++ = 0x00ffff;
else if (((i/(wpl*8)) % 8) == 3)
*ptr++ = 0x0000ff;
else if (((i/(wpl*8)) % 8) == 5)
*ptr++ = 0x000000;
else if (((i/(wpl*8)) % 8) == 0)
*ptr++ = 0xff00ff;
}
break;
}
}
dma_cache_wback_inv((unsigned int)(frame), w*h*bpp/8);
}
/************************************
* Jz475X Framebuffer ops
************************************/
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
static int jz4760fb_setcolreg(u_int regno, u_int red, u_int green, u_int blue,
u_int transp, struct fb_info *info)
{
struct fb_info *fb = info;
if (regno >= NR_PALETTE)
return 1;
if (fb->var.bits_per_pixel <= 16) {
red >>= 8;
green >>= 8;
blue >>= 8;
red &= 0xff;
green &= 0xff;
blue &= 0xff;
}
switch (fb->var.bits_per_pixel) {
case 15:
if (regno < 16)
((u32 *)fb->pseudo_palette)[regno] =
((red >> 3) << 10) |
((green >> 3) << 5) |
(blue >> 3);
break;
case 16:
if (regno < 16) {
((u32 *)fb->pseudo_palette)[regno] =
((red >> 3) << 11) |
((green >> 2) << 5) |
(blue >> 3);
}
break;
case 17 ... 32:
if (regno < 16)
((u32 *)fb->pseudo_palette)[regno] =
(red << 16) |
(green << 8) |
(blue << 0);
/* if (regno < 16) {
unsigned val;
val = chan_to_field(red, &fb->var.red);
val |= chan_to_field(green, &fb->var.green);
val |= chan_to_field(blue, &fb->var.blue);
((u32 *)fb->pseudo_palette)[regno] = val;
}
*/
break;
}
return 0;
}
static int jz4760fb_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg)
{
int ret = 0;
void __user *argp = (void __user *)arg;
struct jz4760lcd_fg_t fg1;
#ifdef CONFIG_FB_JZ4760_TVE
struct jz4760tve_mode jz4760_tve_mode;
#endif
switch (cmd) {
case FBIOSETBACKLIGHT:
__lcd_set_backlight_level(arg); /* We support 8 levels here. */
break;
case FBIODISPON:
REG_LCD_STATE = 0; /* clear lcdc status */
__lcd_slcd_special_on();
REG_LCD_DA1 = virt_to_phys(dma1_desc0);
__lcd_clr_dis();
__lcd_set_ena(); /* enable lcdc */
__lcd_display_on();
break;
case FBIODISPOFF:
__lcd_display_off();
if ( jz4760_lcd_info->panel.cfg & LCD_CFG_LCDPIN_SLCD ||
jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) /* */
__lcd_clr_ena(); /* Smart lcd and TVE mode only support quick disable */
else
__lcd_set_dis(); /* regular disable */
break;
case FBIOPRINT_REG:
print_lcdc_registers();
break;
case FBIO_GET_MODE:
print_dbg("fbio get mode\n");
if (copy_to_user(argp, jz4760_lcd_info, sizeof(struct jz4760lcd_info)))
return -EFAULT;
break;
case FBIO_SET_MODE:
print_dbg("fbio set mode\n");
if (copy_from_user(jz4760_lcd_info, argp, sizeof(struct jz4760lcd_info)))
return -EFAULT;
/* set mode */
jz4760fb_set_mode(&jz4760_lcd_info->osd);
break;
case FBIO_DEEP_SET_MODE:
print_dbg("fbio deep set mode\n");
if (copy_from_user(jz4760_lcd_info, argp, sizeof(struct jz4760lcd_info)))
return -EFAULT;
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
#ifdef CONFIG_FB_JZ4760_TVE
case FBIO_MODE_SWITCH:
print_dbg("lcd mode switch between tve and lcd, arg=%lu\n", arg);
switch ( arg ) {
case PANEL_MODE_TVE_PAL: /* switch to TVE_PAL mode */
case PANEL_MODE_TVE_NTSC: /* switch to TVE_NTSC mode */
jz4760lcd_info_switch_to_TVE(arg);
jz4760tve_init(arg); /* tve controller init */
jz4760tve_enable_tve();
/* turn off lcd backlight */
__lcd_display_off();
break;
case PANEL_MODE_LCD_PANEL: /* switch to LCD mode */
default :
/* turn off TVE, turn off DACn... */
jz4760tve_disable_tve();
jz4760_lcd_info = &jz4760_lcd_panel;
/* turn on lcd backlight */
__lcd_display_on();
break;
}
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
case FBIO_GET_TVE_MODE:
print_dbg("fbio get TVE mode\n");
if (copy_to_user(argp, jz4760_tve_info, sizeof(struct jz4760tve_info)))
return -EFAULT;
break;
case FBIO_SET_TVE_MODE:
print_dbg("fbio set TVE mode\n");
if (copy_from_user(jz4760_tve_info, argp, sizeof(struct jz4760tve_info)))
return -EFAULT;
/* set tve mode */
jz4760tve_set_tve_mode(jz4760_tve_info);
break;
#endif
#if 1
case FBIODISON_FG: //pass
/*lcdc_enable_fg1();*/
print_dbg("lcdc_disable_fg1()\n");
jz4760_lcd_info->osd.osd_cfg |= LCD_OSDC_F1EN;
__lcd_enable_f1();
break;
case FBIODISOFF_FG://pass
/*lcdc_disable_fg1();*/
jz4760_lcd_info->osd.osd_cfg &= ~LCD_OSDC_F1EN;
__lcd_disable_f1();
break;
#ifdef CONFIG_FB_JZ4760_TVE
case FBIO_SET_LCD_TO_TVE:
/*tve PAL NTSC, LCD, */
//jz4760_tve_info
/* 1. display off 2. enable ipu_restart 3. use ipu as input 4. after run_ipu, display on lcd*/
if (copy_from_user(&jz4760_tve_mode, argp, sizeof(struct jz4760tve_mode)))
return -EFAULT;
if (jz4760_tve_mode.mode == PANEL_MODE_LCD_PANEL) {
jz4760tve_disable_tve();
jz4760lcd_info_switch_to_lcd();
__lcd_display_on();
}
else {
jz4760lcd_info_switch_to_TVE(jz4760_tve_mode.mode);
jz4760tve_outfmt_init(jz4760_tve_mode.out_fmt);
jz4760tve_init(jz4760_tve_mode.mode);
jz4760tve_enable_tve();
__lcd_display_off();
}
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
#endif
case FBIO_SET_IPU_TO_LCD:
/* 1. display off 2. enable ipu_restart 3. use ipu as input 4. after run_ipu, display on lcd*/
__lcd_set_dis();
__lcd_fg1_use_ipu();
REG_LCD_BGC = jz4760_lcd_info->osd.ipu_restart;
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
case FBIO_SET_FRM_TO_LCD:
__lcd_fg1_use_dma_chan1();
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
case FBIO_CHANGE_SIZE:
/*fg1_change_size();*/
if(!(REG_LCD_OSDC & LCD_OSDC_F1EN))
return -EFAULT;
if (copy_from_user(&fg1, argp, sizeof(struct jz4760lcd_fg_t)))
return -EFAULT;
jz4760_lcd_info->osd.fg1.w = fg1.w;
jz4760_lcd_info->osd.fg1.h = fg1.h;
jz4760fb_foreground_resize(&jz4760_lcd_info->osd);
break;
case FBIO_CHANGE_POSITION:
if (copy_from_user(&fg1, argp, sizeof(struct jz4760lcd_fg_t)))
return -EFAULT;
jz4760_lcd_info->osd.fg1.x = fg1.x;
jz4760_lcd_info->osd.fg1.y = fg1.y;
jz4760fb_foreground_move(&jz4760_lcd_info->osd);
break;
case FBIO_SET_BG_COLOR://pass
jz4760_lcd_info->osd.bgcolor = arg;
/*lcdc_set_bgcolor(arg);*/
REG_LCD_BGC = jz4760_lcd_info->osd.bgcolor;
break;
case FBIO_SET_IPU_RESTART_VAL:
/*lcdc_set_ipu_restart_val(arg);*/
jz4760_lcd_info->osd.ipu_restart &= ~LCD_IPUR_IPURMASK;
jz4760_lcd_info->osd.ipu_restart |= (arg & LCD_IPUR_IPURMASK);
__lcd_set_ipu_restart_triger(arg);
break;
case FBIO_SET_IPU_RESTART_ON:
/*lcdc_enable_ipu_restart();*/
__lcd_enable_ipu_restart();
break;
case FBIO_SET_IPU_RESTART_OFF:
/*lcdc_disable_ipu_restart();*/
__lcd_disable_ipu_restart();
break;
#endif
default:
printk("%s, unknown command(0x%x)", __FILE__, cmd);
break;
}
return ret;
}
/* Use mmap /dev/fb can only get a non-cacheable Virtual Address. */
static int jz4760fb_mmap(struct fb_info *info, struct vm_area_struct *vma)
{
struct fb_info *fb = info;
unsigned long start;
unsigned long off;
u32 len;
off = vma->vm_pgoff << PAGE_SHIFT;
//fb->fb_get_fix(&fix, PROC_CONSOLE(info), info);
/* frame buffer memory */
start = fb->fix.smem_start;
len = PAGE_ALIGN((start & ~PAGE_MASK) + fb->fix.smem_len);
start &= PAGE_MASK;
if ((vma->vm_end - vma->vm_start + off) > len)
return -EINVAL;
off += start;
vma->vm_pgoff = off >> PAGE_SHIFT;
vma->vm_flags |= VM_IO;
vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); /* Uncacheable */
#if 1
pgprot_val(vma->vm_page_prot) &= ~_CACHE_MASK;
pgprot_val(vma->vm_page_prot) |= _CACHE_UNCACHED; /* Uncacheable */
// pgprot_val(vma->vm_page_prot) |= _CACHE_CACHABLE_NONCOHERENT; /* Write-Back */
#endif
if (io_remap_pfn_range(vma, vma->vm_start, off >> PAGE_SHIFT,
vma->vm_end - vma->vm_start,
vma->vm_page_prot)) {
return -EAGAIN;
}
return 0;
}
#endif /*end of Foreground 1 ops*/
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
static int jz4760fb0_setcolreg(u_int regno, u_int red, u_int green, u_int blue,
u_int transp, struct fb_info *info)
{
struct lcd_cfb_info *cfb = (struct lcd_cfb_info *)info;
unsigned short *ptr, ctmp;
if (regno >= NR_PALETTE)
return 1;
cfb->palette[regno].red = red ;
cfb->palette[regno].green = green;
cfb->palette[regno].blue = blue;
if (cfb->fb0.var.bits_per_pixel <= 16) {
red >>= 8;
green >>= 8;
blue >>= 8;
red &= 0xff;
green &= 0xff;
blue &= 0xff;
}
switch (cfb->fb0.var.bits_per_pixel) {
case 1:
case 2:
case 4:
case 8:
if (((jz4760_lcd_info->panel.cfg & LCD_CFG_MODE_MASK) == LCD_CFG_MODE_SINGLE_MSTN ) ||
((jz4760_lcd_info->panel.cfg & LCD_CFG_MODE_MASK) == LCD_CFG_MODE_DUAL_MSTN )) {
ctmp = (77L * red + 150L * green + 29L * blue) >> 8;
ctmp = ((ctmp >> 3) << 11) | ((ctmp >> 2) << 5) |
(ctmp >> 3);
} else {
/* RGB 565 */
if (((red >> 3) == 0) && ((red >> 2) != 0))
red = 1 << 3;
if (((blue >> 3) == 0) && ((blue >> 2) != 0))
blue = 1 << 3;
ctmp = ((red >> 3) << 11)
| ((green >> 2) << 5) | (blue >> 3);
}
ptr = (unsigned short *)lcd_palette;
ptr = (unsigned short *)(((u32)ptr)|0xa0000000);
ptr[regno] = ctmp;
break;
case 15:
if (regno < 16)
((u32 *)cfb->fb0.pseudo_palette)[regno] =
((red >> 3) << 10) |
((green >> 3) << 5) |
(blue >> 3);
break;
case 16:
if (regno < 16) {
((u32 *)cfb->fb0.pseudo_palette)[regno] =
((red >> 3) << 11) |
((green >> 2) << 5) |
(blue >> 3);
}
break;
case 17 ... 32:
if (regno < 16)
((u32 *)cfb->fb0.pseudo_palette)[regno] =
(red << 16) |
(green << 8) |
(blue << 0);
/* if (regno < 16) {
unsigned val;
val = chan_to_field(red, &cfb->fb0.var.red);
val |= chan_to_field(green, &cfb->fb0.var.green);
val |= chan_to_field(blue, &cfb->fb0.var.blue);
((u32 *)cfb->fb.pseudo_palette)[regno] = val;
}
*/
break;
}
return 0;
}
static int jz4760fb0_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg)
{
int ret = 0;
void __user *argp = (void __user *)arg;
struct jz4760lcd_fg_t fg0;
switch (cmd) {
case FBIOSETBACKLIGHT:
__lcd_set_backlight_level(arg); /* We support 8 levels here. */
break;
case FBIODISPON:
REG_LCD_STATE = 0; /* clear lcdc status */
__lcd_slcd_special_on();
__lcd_clr_dis();
__lcd_set_ena(); /* enable lcdc */
__lcd_display_on();
break;
case FBIODISPOFF:
__lcd_display_off();
if ( jz4760_lcd_info->panel.cfg & LCD_CFG_LCDPIN_SLCD ||
jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) /* */
__lcd_clr_ena(); /* Smart lcd and TVE mode only support quick disable */
else
__lcd_set_dis(); /* regular disable */
break;
case FBIOPRINT_REG:
print_lcdc_registers();
break;
case FBIO_GET_MODE:
print_dbg("fbio get mode\n");
if (copy_to_user(argp, jz4760_lcd_info, sizeof(struct jz4760lcd_info)))
// if (copy_to_user(argp, &jz4760_lcd_info->osd, sizeof(struct jz4760lcd_osd_t)))
return -EFAULT;
break;
case FBIO_SET_MODE:
print_dbg("fbio set mode\n");
if (copy_from_user(jz4760_lcd_info, argp, sizeof(struct jz4760lcd_info)))
// if (copy_to_user(argp, &jz4760_lcd_info->osd, sizeof(struct jz4760lcd_osd_t)))
return -EFAULT;
/* set mode */
jz4760fb_set_mode(&jz4760_lcd_info->osd);
// jz4760fb_set_mode(jz4760_lcd_info);
break;
case FBIO_DEEP_SET_MODE:
print_dbg("fbio deep set mode\n");
if (copy_from_user(jz4760_lcd_info, argp, sizeof(struct jz4760lcd_info)))
return -EFAULT;
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
#ifdef CONFIG_FB_JZ4760_TVE
case FBIO_MODE_SWITCH:
print_dbg("lcd mode switch between tve and lcd, arg=%lu\n", arg);
switch ( arg ) {
case PANEL_MODE_TVE_PAL: /* switch to TVE_PAL mode */
case PANEL_MODE_TVE_NTSC: /* switch to TVE_NTSC mode */
jz4760lcd_info_switch_to_TVE(arg);
jz4760tve_init(arg); /* tve controller init */
jz4760tve_enable_tve();
/* turn off lcd backlight */
__lcd_display_off();
break;
case PANEL_MODE_LCD_PANEL: /* switch to LCD mode */
default :
/* turn off TVE, turn off DACn... */
jz4760tve_disable_tve();
jz4760_lcd_info = &jz4760_lcd_panel;
/* turn on lcd backlight */
__lcd_display_on();
break;
}
jz4760fb_deep_set_mode(jz4760_lcd_info);
break;
case FBIO_GET_TVE_MODE:
print_dbg("fbio get TVE mode\n");
if (copy_to_user(argp, jz4760_tve_info, sizeof(struct jz4760tve_info)))
return -EFAULT;
break;
case FBIO_SET_TVE_MODE:
print_dbg("fbio set TVE mode\n");
if (copy_from_user(jz4760_tve_info, argp, sizeof(struct jz4760tve_info)))
return -EFAULT;
/* set tve mode */
jz4760tve_set_tve_mode(jz4760_tve_info);
break;
#endif
#if 1
case FBIODISON_FG: //pass
/*lcdc_enable_fg0();*/
jz4760_lcd_info->osd.osd_cfg |= LCD_OSDC_F0EN;
__lcd_enable_f0();
break;
case FBIODISOFF_FG://pass
/*lcdc_disable_fg0();*/
print_dbg("lcdc_disable_fg0()\n");
jz4760_lcd_info->osd.osd_cfg &= ~LCD_OSDC_F0EN;
__lcd_disable_f0();
break;
case FBIO_CHANGE_SIZE:
/*fg0_change_size();*/
if(!(REG_LCD_OSDC & LCD_OSDC_F0EN))
return -EFAULT;
if (copy_from_user(&fg0, argp, sizeof(struct jz4760lcd_fg_t)))
return -EFAULT;
jz4760_lcd_info->osd.fg0.w = fg0.w;
jz4760_lcd_info->osd.fg0.h = fg0.h;
jz4760fb0_foreground_resize(&jz4760_lcd_info->osd);
break;
case FBIO_CHANGE_POSITION:
if (copy_from_user(&fg0, argp, sizeof(struct jz4760lcd_fg_t)))
return -EFAULT;
jz4760_lcd_info->osd.fg0.x = fg0.x;
jz4760_lcd_info->osd.fg0.y = fg0.y;
jz4760fb0_foreground_move(&jz4760_lcd_info->osd);
break;
case FBIO_SET_BG_COLOR://pass
jz4760_lcd_info->osd.bgcolor = arg;
/*lcdc_set_bgcolor(arg);*/
REG_LCD_BGC = jz4760_lcd_info->osd.bgcolor;
break;
case FBIO_ALPHA_ON://pass
/*lcdc_enable_alpha();*/
jz4760_lcd_info->osd.osd_cfg |= LCD_OSDC_ALPHAEN;
__lcd_enable_alpha();
break;
case FBIO_ALPHA_OFF://pass
/*lcdc_disable_alpha();*/
jz4760_lcd_info->osd.osd_cfg &= ~LCD_OSDC_ALPHAEN;
__lcd_disable_alpha();
break;
case FBIO_SET_ALPHA_VAL://pass
jz4760_lcd_info->osd.alpha = arg;
/*lcdc_set_alpha(arg);*/
REG_LCD_ALPHA = jz4760_lcd_info->osd.alpha;
break;
#endif
default:
printk("FG0:%s, unknown command(0x%x)", __FILE__, cmd);
break;
}
return ret;
}
static int jz4760fb0_mmap(struct fb_info *info, struct vm_area_struct *vma)
{
struct lcd_cfb_info *cfb = (struct lcd_cfb_info *)info;
unsigned long start;
unsigned long off;
u32 len;
off = vma->vm_pgoff << PAGE_SHIFT;
//fb->fb_get_fix(&fix, PROC_CONSOLE(info), info);
/* frame buffer memory */
start = cfb->fb0.fix.smem_start;
len = PAGE_ALIGN((start & ~PAGE_MASK) + cfb->fb0.fix.smem_len);
start &= PAGE_MASK;
if ((vma->vm_end - vma->vm_start + off) > len)
return -EINVAL;
off += start;
vma->vm_pgoff = off >> PAGE_SHIFT;
vma->vm_flags |= VM_IO;
vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); /* Uncacheable */
#if 1
pgprot_val(vma->vm_page_prot) &= ~_CACHE_MASK;
pgprot_val(vma->vm_page_prot) |= _CACHE_UNCACHED; /* Uncacheable */
// pgprot_val(vma->vm_page_prot) |= _CACHE_CACHABLE_NONCOHERENT; /* Write-Back */
#endif
if (io_remap_pfn_range(vma, vma->vm_start, off >> PAGE_SHIFT,
vma->vm_end - vma->vm_start,
vma->vm_page_prot)) {
return -EAGAIN;
}
return 0;
}
#endif /* end of Foreground 0 ops*/
/* checks var and eventually tweaks it to something supported,
* DO NOT MODIFY PAR */
static int jz4760fb_check_var(struct fb_var_screeninfo *var, struct fb_info *info)
{
if((var->rotate & 1) != (info->var.rotate & 1)) {
if((var->xres != info->var.yres) ||
(var->yres != info->var.xres) ||
(var->xres_virtual != info->var.yres) ||
(var->yres_virtual >
info->var.xres * ANDROID_NUMBER_OF_BUFFERS) ||
(var->yres_virtual < info->var.xres )) {
return -EINVAL;
}
}
else {
if((var->xres != info->var.xres) ||
(var->yres != info->var.yres) ||
(var->xres_virtual != info->var.xres) ||
(var->yres_virtual >
info->var.yres * ANDROID_NUMBER_OF_BUFFERS) ||
(var->yres_virtual < info->var.yres )) {
return -EINVAL;
}
}
if((var->xoffset != info->var.xoffset) ||
(var->bits_per_pixel != info->var.bits_per_pixel)) {// ||
// (var->grayscale != info->var.grayscale)) {
return -EINVAL;
}
return 0;
}
/*
* set the video mode according to info->var
*/
static int jz4760fb_set_par(struct fb_info *info)
{
dprintk("jz4760fb_set_par, not implemented\n");
return 0;
}
/*
* (Un)Blank the display.
* Fix me: should we use VESA value?
*/
static int jz4760fb_blank(int blank_mode, struct fb_info *info)
{
printk("jz4760 fb_blank %d %p", blank_mode, info);
switch (blank_mode) {
case FB_BLANK_UNBLANK:
//case FB_BLANK_NORMAL:
/* Turn on panel */
__lcd_set_ena();
__lcd_display_on();
break;
case FB_BLANK_NORMAL:
case FB_BLANK_VSYNC_SUSPEND:
case FB_BLANK_HSYNC_SUSPEND:
case FB_BLANK_POWERDOWN:
/* Turn off panel */
#if 0
__lcd_display_off();
__lcd_set_dis();
#endif
break;
default:
break;
}
return 0;
}
/*
* pan display
*/
static int jz4760fb_pan_display(struct fb_var_screeninfo *var, struct fb_info *info)
{
struct fb_info *fb = info;
int dy;
if (!var || !fb) {
return -EINVAL;
}
if (var->xoffset - fb->var.xoffset) {
/* No support for X panning for now! */
return -EINVAL;
}
/* TODO: Wait for current frame to finished */
dy = var->yoffset;// - fb->var.yoffset;
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY)
if (dy) {
dma1_desc0->databuf = (unsigned int)virt_to_phys((void *)lcd_frame + (fb->fix.line_length * dy));
dma_cache_wback((unsigned int)(dma1_desc0), sizeof(struct jz4760_lcd_dma_desc));
}
else {
dma1_desc0->databuf = (unsigned int)virt_to_phys((void *)lcd_frame);
dma_cache_wback((unsigned int)(dma1_desc0), sizeof(struct jz4760_lcd_dma_desc));
}
#else
if (dy) {
dma0_desc0->databuf = (unsigned int)virt_to_phys((void *)lcd_frame0 + (fb->fix.line_length * dy));
dma_cache_wback((unsigned int)(dma0_desc0), sizeof(struct jz4760_lcd_dma_desc));
}
else {
dma0_desc0->databuf = (unsigned int)virt_to_phys((void *)lcd_frame0);
dma_cache_wback((unsigned int)(dma0_desc0), sizeof(struct jz4760_lcd_dma_desc));
}
#endif
return 0;
}
/* use default function cfb_fillrect, cfb_copyarea, cfb_imageblit */
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
static struct fb_ops jz4760fb_ops = {
.owner = THIS_MODULE,
.fb_setcolreg = jz4760fb_setcolreg,
.fb_check_var = jz4760fb_check_var,
.fb_set_par = jz4760fb_set_par,
.fb_blank = jz4760fb_blank,
.fb_pan_display = jz4760fb_pan_display,
.fb_fillrect = cfb_fillrect,
.fb_copyarea = cfb_copyarea,
.fb_imageblit = cfb_imageblit,
.fb_mmap = jz4760fb_mmap,
.fb_ioctl = jz4760fb_ioctl,
};
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* use default function cfb_fillrect, cfb_copyarea, cfb_imageblit */
static struct fb_ops jz4760fb0_ops = {
.owner = THIS_MODULE,
.fb_setcolreg = jz4760fb0_setcolreg,
.fb_check_var = jz4760fb_check_var,
.fb_set_par = jz4760fb_set_par,
.fb_blank = jz4760fb_blank,
.fb_pan_display = jz4760fb_pan_display,
.fb_fillrect = cfb_fillrect,
.fb_copyarea = cfb_copyarea,
.fb_imageblit = cfb_imageblit,
.fb_mmap = jz4760fb0_mmap,
.fb_ioctl = jz4760fb0_ioctl,
};
#endif
static int jz4760fb_set_var(struct fb_var_screeninfo *var, int con,
struct fb_info *info)
{
struct fb_info *fb = info;
struct jz4760lcd_info *lcd_info = jz4760_lcd_info;
int chgvar = 0;
if (con == 0) {
var->height = lcd_info->osd.fg0.h; /* tve mode */
var->width = lcd_info->osd.fg0.w;
var->bits_per_pixel = lcd_info->osd.fg0.bpp;
}
else {
var->height = lcd_info->osd.fg1.h;
var->width = lcd_info->osd.fg1.w;
var->bits_per_pixel = lcd_info->osd.fg1.bpp;
}
var->vmode = FB_VMODE_NONINTERLACED;
// var->vmode = FB_VMODE_DOUBLE
var->activate = fb->var.activate;
var->xres = var->width;
var->yres = var->height;
var->xres_virtual = var->width;
var->yres_virtual = var->height * ANDROID_NUMBER_OF_BUFFERS;
var->xoffset = 0;
var->yoffset = 0;
var->pixclock = KHZ2PICOS(jz_clocks.pixclk/1000);
var->left_margin = lcd_info->panel.elw;
var->right_margin = lcd_info->panel.blw;
var->upper_margin = lcd_info->panel.efw;
var->lower_margin = lcd_info->panel.bfw;
var->hsync_len = lcd_info->panel.hsw;
var->vsync_len = lcd_info->panel.vsw;
var->sync = 0;
var->activate = FB_ACTIVATE_NOW;
/*
* CONUPDATE and SMOOTH_XPAN are equal. However,
* SMOOTH_XPAN is only used internally by fbcon.
*/
if (var->vmode & FB_VMODE_CONUPDATE) {
var->vmode |= FB_VMODE_YWRAP;
var->xoffset = fb->var.xoffset;
var->yoffset = fb->var.yoffset;
}
if (var->activate & FB_ACTIVATE_TEST)
return 0;
if ((var->activate & FB_ACTIVATE_MASK) != FB_ACTIVATE_NOW)
return -EINVAL;
if (fb->var.xres != var->xres)
chgvar = 1;
if (fb->var.yres != var->yres)
chgvar = 1;
if (fb->var.xres_virtual != var->xres_virtual)
chgvar = 1;
if (fb->var.yres_virtual != var->yres_virtual)
chgvar = 1;
if (fb->var.bits_per_pixel != var->bits_per_pixel)
chgvar = 1;
//display = fb_display + con;
var->red.msb_right = 0;
var->green.msb_right = 0;
var->blue.msb_right = 0;
switch(var->bits_per_pixel){
case 1: /* Mono */
fb->fix.visual = FB_VISUAL_MONO01;
fb->fix.line_length = (var->xres * var->bits_per_pixel) / 8;
break;
case 2: /* Mono */
var->red.offset = 0;
var->red.length = 2;
var->green.offset = 0;
var->green.length = 2;
var->blue.offset = 0;
var->blue.length = 2;
fb->fix.visual = FB_VISUAL_PSEUDOCOLOR;
fb->fix.line_length = (var->xres * var->bits_per_pixel) / 8;
break;
case 4: /* PSEUDOCOLOUR*/
var->red.offset = 0;
var->red.length = 4;
var->green.offset = 0;
var->green.length = 4;
var->blue.offset = 0;
var->blue.length = 4;
fb->fix.visual = FB_VISUAL_PSEUDOCOLOR;
fb->fix.line_length = var->xres / 2;
break;
case 8: /* PSEUDOCOLOUR, 256 */
var->red.offset = 0;
var->red.length = 8;
var->green.offset = 0;
var->green.length = 8;
var->blue.offset = 0;
var->blue.length = 8;
fb->fix.visual = FB_VISUAL_PSEUDOCOLOR;
fb->fix.line_length = var->xres ;
break;
case 15: /* DIRECTCOLOUR, 32k */
var->bits_per_pixel = 15;
var->red.offset = 10;
var->red.length = 5;
var->green.offset = 5;
var->green.length = 5;
var->blue.offset = 0;
var->blue.length = 5;
fb->fix.visual = FB_VISUAL_DIRECTCOLOR;
fb->fix.line_length = var->xres_virtual * 2;
break;
case 16: /* DIRECTCOLOUR, 64k */
var->bits_per_pixel = 16;
var->red.offset = 11;
var->red.length = 5;
var->green.offset = 5;
var->green.length = 6;
var->blue.offset = 0;
var->blue.length = 5;
fb->fix.visual = FB_VISUAL_TRUECOLOR;
fb->fix.line_length = var->xres_virtual * 2;
break;
case 17 ... 32:
/* DIRECTCOLOUR, 256 */
var->bits_per_pixel = 32;
var->red.offset = 16;
var->red.length = 8;
var->green.offset = 8;
var->green.length = 8;
var->blue.offset = 0;
var->blue.length = 8;
var->transp.offset = 24;
var->transp.length = 8;
fb->fix.visual = FB_VISUAL_TRUECOLOR;
fb->fix.line_length = var->xres_virtual * 4;
break;
default: /* in theory this should never happen */
printk(KERN_WARNING "%s: don't support for %dbpp\n",
fb->fix.id, var->bits_per_pixel);
break;
}
fb->var = *var;
fb->var.activate &= ~FB_ACTIVATE_ALL;
/*
* Update the old var. The fbcon drivers still use this.
* Once they are using cfb->fb.var, this can be dropped.
* --rmk
*/
//display->var = cfb->fb.var;
/*
* If we are setting all the virtual consoles, also set the
* defaults used to create new consoles.
*/
fb_set_cmap(&fb->cmap, fb);
return 0;
}
static struct lcd_cfb_info * jz4760fb_alloc_fb_info(void)
{
struct lcd_cfb_info *cfb;
cfb = kmalloc(sizeof(struct lcd_cfb_info) + sizeof(u32) * 16, GFP_KERNEL);
if (!cfb)
return NULL;
jz4760fb_info = cfb;
memset(cfb, 0, sizeof(struct lcd_cfb_info) );
cfb->currcon = -1;
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* Foreground 1 -- fb */
strcpy(cfb->fb.fix.id, "jzlcd-fg1");
cfb->fb.flags = FBINFO_FLAG_DEFAULT;
cfb->fb.fix.type = FB_TYPE_PACKED_PIXELS;
cfb->fb.fix.type_aux = 0;
cfb->fb.fix.xpanstep = 1;
cfb->fb.fix.ypanstep = 1;
cfb->fb.fix.ywrapstep = 0;
cfb->fb.fix.accel = FB_ACCEL_NONE;
cfb->fb.var.nonstd = 0;
cfb->fb.var.activate = FB_ACTIVATE_NOW;
cfb->fb.var.height = -1;
cfb->fb.var.width = -1;
cfb->fb.var.accel_flags = FB_ACCELF_TEXT;
cfb->fb.fbops = &jz4760fb_ops;
cfb->fb.flags = FBINFO_FLAG_DEFAULT;
cfb->fb.pseudo_palette = (void *)(cfb + 1);
switch (jz4760_lcd_info->osd.fg1.bpp) {
case 1:
fb_alloc_cmap(&cfb->fb.cmap, 4, 0);
break;
case 2:
fb_alloc_cmap(&cfb->fb.cmap, 8, 0);
break;
case 4:
fb_alloc_cmap(&cfb->fb.cmap, 32, 0);
break;
case 8:
default:
fb_alloc_cmap(&cfb->fb.cmap, 256, 0);
break;
}
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* Foreground 0 -- fb0 */
strcpy(cfb->fb0.fix.id, "jzlcd-fg0");
cfb->fb0.fix.type = FB_TYPE_PACKED_PIXELS;
cfb->fb0.fix.type_aux = 0;
cfb->fb0.fix.xpanstep = 1;
cfb->fb0.fix.ypanstep = 1;
cfb->fb0.fix.ywrapstep = 0;
cfb->fb0.fix.accel = FB_ACCEL_NONE;
cfb->fb0.var.nonstd = 0;
cfb->fb0.var.activate = FB_ACTIVATE_NOW;
cfb->fb0.var.height = -1;
cfb->fb0.var.width = -1;
cfb->fb0.var.accel_flags = FB_ACCELF_TEXT;
cfb->fb0.fbops = &jz4760fb0_ops;
cfb->fb0.flags = FBINFO_FLAG_DEFAULT;
cfb->fb0.pseudo_palette = (void *)(cfb + 1);
switch (jz4760_lcd_info->osd.fg0.bpp) {
case 1:
fb_alloc_cmap(&cfb->fb0.cmap, 4, 0);
break;
case 2:
fb_alloc_cmap(&cfb->fb0.cmap, 8, 0);
break;
case 4:
fb_alloc_cmap(&cfb->fb0.cmap, 32, 0);
break;
case 8:
default:
fb_alloc_cmap(&cfb->fb0.cmap, 256, 0);
break;
}
#endif
dprintk("fb_alloc_cmap, fb.cmap.len:%d, fb0.cmap.len:%d....\n", cfb->fb.cmap.len, cfb->fb0.cmap.len);
return cfb;
}
/*
* Map screen memory
*/
static int jz4760fb_map_smem(struct lcd_cfb_info *cfb)
{
unsigned long page;
unsigned int page_shift, needroom = 0, needroom1=0, bpp, w, h;
unsigned char *fb_palette, *fb_frame;
/* caculate the mem size of Foreground 1 */
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
bpp = jz4760_lcd_info->osd.fg1.bpp;
if (bpp == 18 || bpp == 24)
bpp = 32;
if (bpp == 15)
bpp = 16;
#ifndef CONFIG_FB_JZ4760_TVE
w = jz4760_lcd_info->osd.fg1.w;
h = jz4760_lcd_info->osd.fg1.h;
#else /* CONFIG_FB_JZ4760_TVE */
w = (jz4760_lcd_info->osd.fg1.w > TVE_WIDTH_PAL) ? jz4760_lcd_info->osd.fg1.w : TVE_WIDTH_PAL;
h = (jz4760_lcd_info->osd.fg1.h > TVE_HEIGHT_PAL) ? jz4760_lcd_info->osd.fg1.h : TVE_HEIGHT_PAL;
#endif
needroom1 = needroom = ((w * bpp + 7) >> 3) * h * ANDROID_NUMBER_OF_BUFFERS;
#endif /* end of alloc Foreground 1 mem */
/* caculate the mem size of Foreground 1 */
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
bpp = jz4760_lcd_info->osd.fg0.bpp;
if (bpp == 18 || bpp == 24)
bpp = 32;
if (bpp == 15)
bpp = 16;
#ifndef CONFIG_FB_JZ4760_TVE
w = jz4760_lcd_info->osd.fg0.w;
h = jz4760_lcd_info->osd.fg0.h;
#else
w = (jz4760_lcd_info->osd.fg0.w > TVE_WIDTH_PAL) ? jz4760_lcd_info->osd.fg0.w : TVE_WIDTH_PAL;
h = (jz4760_lcd_info->osd.fg0.h > TVE_HEIGHT_PAL) ? jz4760_lcd_info->osd.fg0.h : TVE_HEIGHT_PAL;
#endif
// needroom += ((w * bpp + 7) >> 3) * h;
needroom += ((w * bpp + 7) >> 3) * h * ANDROID_NUMBER_OF_BUFFERS;
#endif /* end of alloc Foreground 1 mem */
/* Alloc memory */
for (page_shift = 0; page_shift < 12; page_shift++)
if ((PAGE_SIZE << page_shift) >= needroom)
break;
fb_palette = (unsigned char *)__get_free_pages(GFP_KERNEL, 0);
fb_frame = (unsigned char *)__get_free_pages(GFP_KERNEL, page_shift);
if ((!fb_palette) || (!fb_frame))
return -ENOMEM;
memset((void *)fb_palette, 0, PAGE_SIZE);
memset((void *)fb_frame, 0, PAGE_SIZE << page_shift);
lcd_palette = fb_palette;
dma_desc_base = (struct jz4760_lcd_dma_desc *)((void*)lcd_palette + ((PALETTE_SIZE+3)/4)*4);
/*
#if defined(CONFIG_FB_JZ4760_SLCD)
lcd_cmdbuf = (unsigned char *)__get_free_pages(GFP_KERNEL, 0);
memset((void *)lcd_cmdbuf, 0, PAGE_SIZE);
int data, i, *ptr;
ptr = (unsigned int *)lcd_cmdbuf;
data = WR_GRAM_CMD;
data = ((data & 0xff) << 1) | ((data & 0xff00) << 2);
for(i = 0; i < 3; i++)
ptr[i] = data;
#endif
*/
/*
* Set page reserved so that mmap will work. This is necessary
* since we'll be remapping normal memory.
*/
page = (unsigned long)lcd_palette;
SetPageReserved(virt_to_page((void*)page));
#if 1
for (page = (unsigned long)fb_frame;
page < PAGE_ALIGN((unsigned long)fb_frame + (PAGE_SIZE<<page_shift));
page += PAGE_SIZE) {
SetPageReserved(virt_to_page((void*)page));
}
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
// lcd_frame = fb_frame;
lcd_frame = fb_frame + needroom1;
cfb->fb.fix.smem_start = virt_to_phys((void *)lcd_frame);
cfb->fb.fix.smem_len = needroom - needroom1; /* page_shift/2 ??? */
cfb->fb.screen_base =
(unsigned char *)(((unsigned int)lcd_frame&0x1fffffff) | 0xa0000000);
if (!cfb->fb.screen_base) {
printk("jz4760fb, %s: unable to map screen memory\n", cfb->fb.fix.id);
return -ENOMEM;
}
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
lcd_frame0 = fb_frame;
cfb->fb0.fix.smem_start = virt_to_phys((void *)lcd_frame0);
cfb->fb0.fix.smem_len = needroom1; /* page_shift/2 ??? */
cfb->fb0.screen_base =
(unsigned char *)(((unsigned int)lcd_frame0&0x1fffffff) | 0xa0000000);
if (!cfb->fb0.screen_base) {
printk("jz4760fb0, %s: unable to map screen memory\n", cfb->fb0.fix.id);
return -ENOMEM;
}
#endif
return 0;
}
static void jz4760fb_free_fb_info(struct lcd_cfb_info *cfb)
{
if (cfb) {
fb_alloc_cmap(&cfb->fb.cmap, 0, 0);
kfree(cfb);
}
}
static void jz4760fb_unmap_smem(struct lcd_cfb_info *cfb)
{
struct page * map = NULL;
unsigned char *tmp;
unsigned int page_shift, needroom, bpp, w, h;
bpp = jz4760_lcd_info->osd.fg0.bpp;
if ( bpp == 18 || bpp == 24)
bpp = 32;
if ( bpp == 15 )
bpp = 16;
w = jz4760_lcd_info->osd.fg0.w;
h = jz4760_lcd_info->osd.fg0.h;
needroom = ((w * bpp + 7) >> 3) * h;
#if defined(JZ4760_LCD_USE_2LAYER_FG)
bpp = jz4760_lcd_info->osd.fg1.bpp;
if ( bpp == 18 || bpp == 24)
bpp = 32;
if ( bpp == 15 )
bpp = 16;
w = jz4760_lcd_info->osd.fg1.w;
h = jz4760_lcd_info->osd.fg1.h;
needroom += ((w * bpp + 7) >> 3) * h;
#endif
for (page_shift = 0; page_shift < 12; page_shift++)
if ((PAGE_SIZE << page_shift) >= needroom)
break;
if (cfb && cfb->fb.screen_base) {
iounmap(cfb->fb.screen_base);
cfb->fb.screen_base = NULL;
release_mem_region(cfb->fb.fix.smem_start,
cfb->fb.fix.smem_len);
}
if (lcd_palette) {
map = virt_to_page(lcd_palette);
clear_bit(PG_reserved, &map->flags);
free_pages((int)lcd_palette, 0);
}
if (lcd_frame0) {
for (tmp=(unsigned char *)lcd_frame0;
tmp < lcd_frame0 + (PAGE_SIZE << page_shift);
tmp += PAGE_SIZE) {
map = virt_to_page(tmp);
clear_bit(PG_reserved, &map->flags);
}
free_pages((int)lcd_frame0, page_shift);
}
}
/************************************
* Jz475X Chipset OPS
************************************/
/*
* switch to tve mode from lcd mode
* mode:
* PANEL_MODE_TVE_PAL: switch to TVE_PAL mode
* PANEL_MODE_TVE_NTSC: switch to TVE_NTSC mode
*/
static void print_lcdc_registers(void) /* debug */
{
#ifdef DEBUG
/* LCD Controller Resgisters */
printk("REG_LCD_CFG:\t0x%08x\n", REG_LCD_CFG);
printk("REG_LCD_CTRL:\t0x%08x\n", REG_LCD_CTRL);
printk("REG_LCD_STATE:\t0x%08x\n", REG_LCD_STATE);
printk("REG_LCD_OSDC:\t0x%08x\n", REG_LCD_OSDC);
printk("REG_LCD_OSDCTRL:\t0x%08x\n", REG_LCD_OSDCTRL);
printk("REG_LCD_OSDS:\t0x%08x\n", REG_LCD_OSDS);
printk("REG_LCD_BGC:\t0x%08x\n", REG_LCD_BGC);
printk("REG_LCD_KEK0:\t0x%08x\n", REG_LCD_KEY0);
printk("REG_LCD_KEY1:\t0x%08x\n", REG_LCD_KEY1);
printk("REG_LCD_ALPHA:\t0x%08x\n", REG_LCD_ALPHA);
printk("REG_LCD_IPUR:\t0x%08x\n", REG_LCD_IPUR);
printk("REG_LCD_VAT:\t0x%08x\n", REG_LCD_VAT);
printk("REG_LCD_DAH:\t0x%08x\n", REG_LCD_DAH);
printk("REG_LCD_DAV:\t0x%08x\n", REG_LCD_DAV);
printk("REG_LCD_XYP0:\t0x%08x\n", REG_LCD_XYP0);
printk("REG_LCD_XYP1:\t0x%08x\n", REG_LCD_XYP1);
printk("REG_LCD_SIZE0:\t0x%08x\n", REG_LCD_SIZE0);
printk("REG_LCD_SIZE1:\t0x%08x\n", REG_LCD_SIZE1);
printk("REG_LCD_RGBC\t0x%08x\n", REG_LCD_RGBC);
printk("REG_LCD_VSYNC:\t0x%08x\n", REG_LCD_VSYNC);
printk("REG_LCD_HSYNC:\t0x%08x\n", REG_LCD_HSYNC);
printk("REG_LCD_PS:\t0x%08x\n", REG_LCD_PS);
printk("REG_LCD_CLS:\t0x%08x\n", REG_LCD_CLS);
printk("REG_LCD_SPL:\t0x%08x\n", REG_LCD_SPL);
printk("REG_LCD_REV:\t0x%08x\n", REG_LCD_REV);
printk("REG_LCD_IID:\t0x%08x\n", REG_LCD_IID);
printk("REG_LCD_DA0:\t0x%08x\n", REG_LCD_DA0);
printk("REG_LCD_SA0:\t0x%08x\n", REG_LCD_SA0);
printk("REG_LCD_FID0:\t0x%08x\n", REG_LCD_FID0);
printk("REG_LCD_CMD0:\t0x%08x\n", REG_LCD_CMD0);
printk("REG_LCD_OFFS0:\t0x%08x\n", REG_LCD_OFFS0);
printk("REG_LCD_PW0:\t0x%08x\n", REG_LCD_PW0);
printk("REG_LCD_CNUM0:\t0x%08x\n", REG_LCD_CNUM0);
printk("REG_LCD_DESSIZE0:\t0x%08x\n", REG_LCD_DESSIZE0);
printk("REG_LCD_DA1:\t0x%08x\n", REG_LCD_DA1);
printk("REG_LCD_SA1:\t0x%08x\n", REG_LCD_SA1);
printk("REG_LCD_FID1:\t0x%08x\n", REG_LCD_FID1);
printk("REG_LCD_CMD1:\t0x%08x\n", REG_LCD_CMD1);
printk("REG_LCD_OFFS1:\t0x%08x\n", REG_LCD_OFFS1);
printk("REG_LCD_PW1:\t0x%08x\n", REG_LCD_PW1);
printk("REG_LCD_CNUM1:\t0x%08x\n", REG_LCD_CNUM1);
printk("REG_LCD_DESSIZE1:\t0x%08x\n", REG_LCD_DESSIZE1);
printk("==================================\n");
printk("REG_LCD_VSYNC:\t%d:%d\n", REG_LCD_VSYNC>>16, REG_LCD_VSYNC&0xfff);
printk("REG_LCD_HSYNC:\t%d:%d\n", REG_LCD_HSYNC>>16, REG_LCD_HSYNC&0xfff);
printk("REG_LCD_VAT:\t%d:%d\n", REG_LCD_VAT>>16, REG_LCD_VAT&0xfff);
printk("REG_LCD_DAH:\t%d:%d\n", REG_LCD_DAH>>16, REG_LCD_DAH&0xfff);
printk("REG_LCD_DAV:\t%d:%d\n", REG_LCD_DAV>>16, REG_LCD_DAV&0xfff);
printk("==================================\n");
/* Smart LCD Controller Resgisters */
printk("REG_SLCD_CFG:\t0x%08x\n", REG_SLCD_CFG);
printk("REG_SLCD_CTRL:\t0x%08x\n", REG_SLCD_CTRL);
printk("REG_SLCD_STATE:\t0x%08x\n", REG_SLCD_STATE);
printk("==================================\n");
/* TVE Controller Resgisters */
printk("REG_TVE_CTRL:\t0x%08x\n", REG_TVE_CTRL);
printk("REG_TVE_FRCFG:\t0x%08x\n", REG_TVE_FRCFG);
printk("REG_TVE_SLCFG1:\t0x%08x\n", REG_TVE_SLCFG1);
printk("REG_TVE_SLCFG2:\t0x%08x\n", REG_TVE_SLCFG2);
printk("REG_TVE_SLCFG3:\t0x%08x\n", REG_TVE_SLCFG3);
printk("REG_TVE_LTCFG1:\t0x%08x\n", REG_TVE_LTCFG1);
printk("REG_TVE_LTCFG2:\t0x%08x\n", REG_TVE_LTCFG2);
printk("REG_TVE_CFREQ:\t0x%08x\n", REG_TVE_CFREQ);
printk("REG_TVE_CPHASE:\t0x%08x\n", REG_TVE_CPHASE);
printk("REG_TVE_CBCRCFG:\t0x%08x\n", REG_TVE_CBCRCFG);
printk("REG_TVE_WSSCR:\t0x%08x\n", REG_TVE_WSSCR);
printk("REG_TVE_WSSCFG1:\t0x%08x\n", REG_TVE_WSSCFG1);
printk("REG_TVE_WSSCFG2:\t0x%08x\n", REG_TVE_WSSCFG2);
printk("REG_TVE_WSSCFG3:\t0x%08x\n", REG_TVE_WSSCFG3);
printk("==================================\n");
if ( 1 ) {
unsigned int * pii = (unsigned int *)dma_desc_base;
int i, j;
for (j=0;j< DMA_DESC_NUM ; j++) {
printk("dma_desc%d(0x%08x):\n", j, (unsigned int)pii);
for (i =0; i<8; i++ ) {
printk("\t\t0x%08x\n", *pii++);
}
}
}
#endif
}
#ifdef CONFIG_FB_JZ4760_TVE
static void jz4760lcd_info_switch_to_TVE(int mode)
{
struct jz4760lcd_info *info;
struct jz4760lcd_osd_t *osd_lcd;
struct jz4760lcd_panel_t *panel_lcd;
int x, y, w, h;
info = &jz4760_info_tve;
/* Set to tve mode */
info->panel.cfg = jz4760_lcd_panel.panel.cfg;
info->panel.cfg |= LCD_CFG_TVEN | LCD_CFG_MODE_INTER_CCIR656; /* Interlace CCIR656 mode */
info->panel.ctrl = jz4760_lcd_panel.panel.ctrl;
info->osd.rgb_ctrl = LCD_RGBC_YCC; /* enable RGB => YUV */
/* Copy current to keep the old style */
osd_lcd = &jz4760_lcd_panel.osd;
panel_lcd = &jz4760_lcd_panel.panel;
switch ( mode ) {
case PANEL_MODE_TVE_PAL:
info->panel.cfg |= LCD_CFG_TVEPEH; /* TVE PAL enable extra halfline signal */
info->panel.w = TVE_WIDTH_PAL;
info->panel.h = TVE_HEIGHT_PAL;
info->panel.fclk = TVE_FREQ_PAL;
/* set Foreground 0 */
w = osd_lcd->fg0.w / panel_lcd->w * TVE_WIDTH_PAL;
h = osd_lcd->fg0.h / panel_lcd->h * TVE_HEIGHT_PAL;
x = osd_lcd->fg0.x / panel_lcd->w * TVE_WIDTH_PAL;
y = osd_lcd->fg0.y / panel_lcd->h * TVE_HEIGHT_PAL;
info->osd.fg0.bpp = osd_lcd->fg0.bpp;
info->osd.fg0.x = x;
info->osd.fg0.y = y;
info->osd.fg0.w = w;
info->osd.fg0.h = h;
/* set Foreground 1 */
w = osd_lcd->fg1.w / panel_lcd->w * TVE_WIDTH_PAL;
h = osd_lcd->fg1.h / panel_lcd->h * TVE_HEIGHT_PAL;
x = osd_lcd->fg1.x / panel_lcd->w * TVE_WIDTH_PAL;
y = osd_lcd->fg1.y / panel_lcd->h * TVE_HEIGHT_PAL;
info->osd.fg1.bpp = 32; /* use RGB888 in TVE mode*/
info->osd.fg1.x = x;
info->osd.fg1.y = y;
info->osd.fg1.w = w;
info->osd.fg1.h = h;
break;
case PANEL_MODE_TVE_NTSC:
info->panel.cfg &= ~LCD_CFG_TVEPEH; /* TVE NTSC disable extra halfline signal */
info->panel.w = TVE_WIDTH_NTSC;
info->panel.h = TVE_HEIGHT_NTSC;
info->panel.fclk = TVE_FREQ_NTSC;
w = osd_lcd->fg0.w / panel_lcd->w * TVE_WIDTH_PAL;
h = osd_lcd->fg0.h / panel_lcd->h * TVE_HEIGHT_NTSC;
x = osd_lcd->fg0.x / panel_lcd->w * TVE_WIDTH_NTSC;
y = osd_lcd->fg0.y / panel_lcd->h * TVE_HEIGHT_NTSC;
info->osd.fg0.bpp = osd_lcd->fg0.bpp;
info->osd.fg0.x = x;
info->osd.fg0.y = y;
info->osd.fg0.w = w;
info->osd.fg0.h = h;
w = osd_lcd->fg1.w / panel_lcd->w * TVE_WIDTH_PAL;
h = osd_lcd->fg1.h / panel_lcd->h * TVE_HEIGHT_NTSC;
x = osd_lcd->fg1.x / panel_lcd->w * TVE_WIDTH_NTSC;
y = osd_lcd->fg1.y / panel_lcd->h * TVE_HEIGHT_NTSC;
info->osd.fg1.bpp = 32; /* use RGB888 int TVE mode */
info->osd.fg1.x = x;
info->osd.fg1.y = y;
info->osd.fg1.w = w;
info->osd.fg1.h = h;
break;
default:
printk("%s, %s: Unknown tve mode\n", __FILE__, __FUNCTION__);
}
jz4760_lcd_info = &jz4760_info_tve;
}
/*
* switch to lcd mode from TVE mode
*/
static void jz4760lcd_info_switch_to_lcd(void)
{
struct jz4760lcd_info *info;
struct jz4760lcd_osd_t *osd_lcd;
struct jz4760lcd_panel_t *panel_lcd;
int x, y, w, h;
info = &jz4760_lcd_panel;
/* set to tve mode */
info->panel.cfg = jz4760_info_tve.panel.cfg;
info->panel.cfg &= ~(LCD_CFG_TVEN | LCD_CFG_MODE_INTER_CCIR656); /* Interlace CCIR656 mode */
info->panel.ctrl = jz4760_info_tve.panel.ctrl;
info->osd.rgb_ctrl &= ~LCD_RGBC_YCC; /* enable YUV => RGB*/
/* */
osd_lcd = &jz4760_info_tve.osd;
panel_lcd = &jz4760_info_tve.panel;
/* set Foreground 0 */
w = osd_lcd->fg0.w / panel_lcd->w * info->panel.w;
h = osd_lcd->fg0.h / panel_lcd->h * info->panel.h;
x = osd_lcd->fg0.x / panel_lcd->w * info->panel.w;
y = osd_lcd->fg0.y / panel_lcd->h * info->panel.h;
info->osd.fg0.x = x;
info->osd.fg0.y = y;
info->osd.fg0.w = w;
info->osd.fg0.h = h;
/* set Foreground 1 */
w = osd_lcd->fg1.w / panel_lcd->w * info->panel.w;
h = osd_lcd->fg1.h / panel_lcd->h * info->panel.h;
x = osd_lcd->fg1.x / panel_lcd->w * info->panel.w;
y = osd_lcd->fg1.y / panel_lcd->h * info->panel.h;
info->osd.fg1.x = x;
info->osd.fg1.y = y;
info->osd.fg1.w = w;
info->osd.fg1.h = h;
jz4760_lcd_info = &jz4760_lcd_panel;
}
#endif
/* initial dma descriptors */
static void jz4760fb_descriptor_init( struct jz4760lcd_info * lcd_info )
{
unsigned int pal_size;
int fg0_line_size, fg0_frm_size, fg1_line_size, fg1_frm_size;
int size0, size1;
switch ( lcd_info->osd.fg0.bpp ) {
case 1:
pal_size = 4;
break;
case 2:
pal_size = 8;
break;
case 4:
pal_size = 32;
break;
case 8:
default:
pal_size = 512;
}
pal_size /= 4;
/*
* Normal TFT panel's DMA Chan0:
* TO LCD Panel:
* no palette: dma0_desc0 <<==>> dma0_desc0
* palette : dma0_desc_palette <<==>> dma0_desc0
* TO TV Encoder:
* no palette: dma0_desc0 <<==>> dma0_desc1
* palette: dma0_desc_palette --> dma0_desc0
* --> dma0_desc1 --> dma0_desc_palette --> ...
* DMA Chan1:
* TO LCD Panel:
* dma1_desc0 <<==>> dma1_desc0
* TO TV Encoder:
* dma1_desc0 <<==>> dma1_desc1
*/
dma0_desc_palette = dma_desc_base + 0;
#if 1//defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
dma0_desc0 = dma_desc_base + 1;
dma0_desc1 = dma_desc_base + 2;
dma0_desc0_change = dma_desc_base + 3;
dma0_desc1_change = dma_desc_base + 4;
/* Foreground 0, caculate size */
if ( lcd_info->osd.fg0.x >= lcd_info->panel.w )
lcd_info->osd.fg0.x = lcd_info->panel.w - 1;
if ( lcd_info->osd.fg0.y >= lcd_info->panel.h )
lcd_info->osd.fg0.y = lcd_info->panel.h - 1;
if ( lcd_info->osd.fg0.x + lcd_info->osd.fg0.w > lcd_info->panel.w )
lcd_info->osd.fg0.w = lcd_info->panel.w - lcd_info->osd.fg0.x;
if ( lcd_info->osd.fg0.y + lcd_info->osd.fg0.h > lcd_info->panel.h )
lcd_info->osd.fg0.h = lcd_info->panel.h - lcd_info->osd.fg0.y;
size0 = lcd_info->osd.fg0.h << 16 | lcd_info->osd.fg0.w;
fg0_line_size = (lcd_info->osd.fg0.w * (lcd_info->osd.fg0.bpp) / 8);
fg0_line_size = ((fg0_line_size + 3) >> 2) << 2; /* word aligned */
fg0_frm_size = fg0_line_size * lcd_info->osd.fg0.h;
/* Palette Descriptor */
dma0_desc_palette->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma0_desc_palette->databuf = (unsigned int)virt_to_phys((void *)lcd_palette);
dma0_desc_palette->frame_id = (unsigned int)0xaaaaaaaa;
dma0_desc_palette->cmd = LCD_CMD_PAL | pal_size; /* Palette Descriptor */
/* DMA0 Descriptor */
if ( lcd_info->panel.cfg & LCD_CFG_TVEN ) { /* TVE mode */
/* Next */
dma0_desc0->next_desc = (unsigned int)virt_to_phys(dma0_desc1);
if (lcd_info->osd.fg0.bpp <= 8) /* load palette only once at setup */
dma0_desc1->next_desc = (unsigned int)virt_to_phys(dma0_desc_palette);
else
dma0_desc1->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma0_desc0_change->next_desc = (unsigned int)virt_to_phys(dma0_desc1_change);
dma0_desc1_change->next_desc = (unsigned int)virt_to_phys(dma0_desc0_change);
/* frame phys addr */
dma0_desc0->databuf = dma0_desc0_change->databuf = virt_to_phys((void *)lcd_frame0);
dma0_desc1->databuf = virt_to_phys((void *)(lcd_frame0 + fg0_line_size));
/* frame id */
dma0_desc0->frame_id = (unsigned int)0x0da00000; /* DMA0'0 */
dma0_desc1->frame_id = (unsigned int)0x0da00001; /* DMA0'1 */
dma0_desc0_change->frame_id = (unsigned int)0x0da000c0; /* DMA0'2 */
dma0_desc1_change->frame_id = (unsigned int)0x0da000c1;
/* others */
dma0_desc0->cmd = dma0_desc1->cmd = (fg0_frm_size/4)/2;
dma0_desc0->offsize = dma0_desc1->offsize = fg0_line_size/4;
dma0_desc0->page_width = dma0_desc1->page_width = fg0_line_size/4;
dma0_desc0->desc_size = dma0_desc1->desc_size = size0;
}
else { /* Normal TFT LCD */
/* next */
dma0_desc0->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma0_desc0_change->next_desc = (unsigned int)virt_to_phys(dma0_desc0_change);
/* frame phys addr */
dma0_desc0->databuf = dma0_desc0_change->databuf = virt_to_phys((void *)lcd_frame0);
/* frame id */
dma0_desc0->frame_id = (unsigned int)0x0000da00; /* DMA0'0 */
dma0_desc0_change->frame_id = (unsigned int)0x0da000c0; /* DMA0'2 */
/* others */
dma0_desc0->cmd = fg0_frm_size/4;
dma0_desc0->offsize =0;
dma0_desc0->page_width = 0;
dma0_desc0->desc_size = size0;
}
if (lcd_info->osd.fg0.bpp <= 8) /* load palette only once at setup */
REG_LCD_DA0 = virt_to_phys(dma0_desc_palette);
else
REG_LCD_DA0 = virt_to_phys(dma0_desc0); //tft
REG_LCD_SIZE0 = size0;
current_dma0_id = 0;//dma0_desc0;
#endif
#if 1//defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
dma1_desc0 = dma_desc_base + 5;
dma1_desc1 = dma_desc_base + 6;
dma1_desc0_change = dma_desc_base + 7;
dma1_desc1_change = dma_desc_base + 8;
/* Foreground 1, caculate size */
if ( lcd_info->osd.fg1.x >= lcd_info->panel.w )
lcd_info->osd.fg1.x = lcd_info->panel.w - 1;
if ( lcd_info->osd.fg1.y >= lcd_info->panel.h )
lcd_info->osd.fg1.y = lcd_info->panel.h - 1;
if ( lcd_info->osd.fg1.x + lcd_info->osd.fg1.w > lcd_info->panel.w )
lcd_info->osd.fg1.w = lcd_info->panel.w - lcd_info->osd.fg1.x;
if ( lcd_info->osd.fg1.y + lcd_info->osd.fg1.h > lcd_info->panel.h )
lcd_info->osd.fg1.h = lcd_info->panel.h - lcd_info->osd.fg1.y;
size1 = lcd_info->osd.fg1.h << 16 | lcd_info->osd.fg1.w;
fg1_line_size = lcd_info->osd.fg1.w*lcd_info->osd.fg1.bpp/8;
fg1_line_size = ((fg1_line_size+3)>>2)<<2; /* word aligned */
fg1_frm_size = fg1_line_size * lcd_info->osd.fg1.h;
/* DMA1 Descriptor */
if ( lcd_info->panel.cfg & LCD_CFG_TVEN ) {/* TVE mode */
/* Next */
dma1_desc0->next_desc = (unsigned int)virt_to_phys(dma1_desc1);
dma1_desc1->next_desc = (unsigned int)virt_to_phys(dma1_desc0);
dma1_desc0_change->next_desc = (unsigned int)virt_to_phys(dma1_desc1_change);
dma1_desc1_change->next_desc = (unsigned int)virt_to_phys(dma1_desc0_change);
/* frame phys addr */
dma1_desc0->databuf = dma1_desc0_change->databuf = virt_to_phys((void *)lcd_frame);
dma1_desc1->databuf = virt_to_phys((void *)(lcd_frame + fg1_line_size));
/* frame id */
dma1_desc0->frame_id = (unsigned int)0x0da10000; /* DMA1'0 */
dma1_desc1->frame_id = (unsigned int)0x0da10001; /* DMA1'1 */
dma1_desc0_change->frame_id = (unsigned int)0x0da100c0; /* DMA1'C0 */
dma1_desc1_change->frame_id = (unsigned int)0x0da100c1; /* DMA1'C1 */
/* other*/
dma1_desc0->cmd = dma1_desc1->cmd = (fg1_frm_size/4)/2;
dma1_desc0->offsize = dma1_desc1->offsize = fg1_line_size/4;
dma1_desc0->page_width = dma1_desc1->page_width = fg1_line_size/4;
dma1_desc0->desc_size = dma1_desc1->desc_size = size1;
}
else { /* Normal TFT LCD */
/* Next */
dma1_desc0->next_desc = (unsigned int)virt_to_phys(dma1_desc0);
dma1_desc0_change->next_desc = (unsigned int)virt_to_phys(dma1_desc0_change);
/* frame phys addr */
dma1_desc0->databuf = dma1_desc0_change->databuf = virt_to_phys((void *)lcd_frame);
/* frame id */
dma1_desc0->frame_id = (unsigned int)0x0da10000; /* DMA1'0 */
dma1_desc0_change->frame_id = (unsigned int)0x0da100c0; /* DMA1'C0 */
/* other */
dma1_desc0->cmd = fg1_frm_size/4;
dma1_desc0->offsize = 0;
dma1_desc0->page_width = 0;
dma1_desc0->desc_size = size1;
}
REG_LCD_DA1 = virt_to_phys(dma1_desc0); /* set Dma-chan1's Descripter Addrress */
REG_LCD_SIZE1 = size1;
current_dma1_id = 0;//dma1_desc0;
#endif
dma_cache_wback((unsigned int)(dma_desc_base), (DMA_DESC_NUM)*sizeof(struct jz4760_lcd_dma_desc));
}
static void jz4760fb_set_panel_mode(struct jz4760lcd_info * lcd_info)
{
struct jz4760lcd_panel_t *panel = &lcd_info->panel;
/* set bpp */
lcd_info->panel.ctrl &= ~LCD_CTRL_BPP_MASK;
if ( lcd_info->osd.fg0.bpp == 1 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_1;
else if ( lcd_info->osd.fg0.bpp == 2 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_2;
else if ( lcd_info->osd.fg0.bpp == 4 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_4;
else if ( lcd_info->osd.fg0.bpp == 8 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_8;
else if ( lcd_info->osd.fg0.bpp == 15 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_16 | LCD_CTRL_RGB555;
else if ( lcd_info->osd.fg0.bpp == 16 )
lcd_info->panel.ctrl |= LCD_CTRL_BPP_16 | LCD_CTRL_RGB565;
else if ( lcd_info->osd.fg0.bpp > 16 && lcd_info->osd.fg0.bpp < 32+1 ) {
lcd_info->osd.fg0.bpp = 32;
lcd_info->panel.ctrl |= LCD_CTRL_BPP_18_24;
}
else {
printk("The BPP %d is not supported\n", lcd_info->osd.fg0.bpp);
lcd_info->osd.fg0.bpp = 32;
lcd_info->panel.ctrl |= LCD_CTRL_BPP_18_24;
}
lcd_info->panel.cfg |= LCD_CFG_NEWDES; /* use 8words descriptor always */
REG_LCD_CTRL = lcd_info->panel.ctrl; /* LCDC Controll Register */
REG_LCD_CFG = lcd_info->panel.cfg; /* LCDC Configure Register */
REG_SLCD_CFG = lcd_info->panel.slcd_cfg; /* Smart LCD Configure Register */
if ( lcd_info->panel.cfg & LCD_CFG_LCDPIN_SLCD ) /* enable Smart LCD DMA */
REG_SLCD_CTRL = SLCD_CTRL_DMA_EN;
switch ( lcd_info->panel.cfg & LCD_CFG_MODE_MASK ) {
case LCD_CFG_MODE_GENERIC_TFT:
case LCD_CFG_MODE_INTER_CCIR656:
case LCD_CFG_MODE_NONINTER_CCIR656:
case LCD_CFG_MODE_SLCD:
default: /* only support TFT16 TFT32, not support STN and Special TFT by now(10-06-2008)*/
REG_LCD_VAT = (((panel->blw + panel->w + panel->elw + panel->hsw)) << 16) | (panel->vsw + panel->bfw + panel->h + panel->efw);
REG_LCD_DAH = ((panel->hsw + panel->blw) << 16) | (panel->hsw + panel->blw + panel->w);
REG_LCD_DAV = ((panel->vsw + panel->bfw) << 16) | (panel->vsw + panel->bfw + panel->h);
REG_LCD_HSYNC = (0 << 16) | panel->hsw;
REG_LCD_VSYNC = (0 << 16) | panel->vsw;
break;
}
}
//static void jz4760fb_set_osd_mode( struct jz4760lcd_info * lcd_info )
static void jz4760fb_set_osd_mode(struct jz4760lcd_osd_t *lcd_osd_info)
{
lcd_osd_info->osd_ctrl &= ~(LCD_OSDCTRL_OSDBPP_MASK);
if ( lcd_osd_info->fg1.bpp == 15 )
lcd_osd_info->osd_ctrl |= LCD_OSDCTRL_OSDBPP_15_16|LCD_OSDCTRL_RGB555;
else if ( lcd_osd_info->fg1.bpp == 16 )
lcd_osd_info->osd_ctrl |= LCD_OSDCTRL_OSDBPP_15_16|LCD_OSDCTRL_RGB565;
else {
lcd_osd_info->fg1.bpp = 32;
lcd_osd_info->osd_ctrl |= LCD_OSDCTRL_OSDBPP_18_24;
}
REG_LCD_OSDC = lcd_osd_info->osd_cfg; /* F0, F1, alpha, */
REG_LCD_OSDCTRL = lcd_osd_info->osd_ctrl; /* IPUEN, bpp */
REG_LCD_RGBC = lcd_osd_info->rgb_ctrl;
REG_LCD_BGC = lcd_osd_info->bgcolor;
REG_LCD_KEY0 = lcd_osd_info->colorkey0;
REG_LCD_KEY1 = lcd_osd_info->colorkey1;
REG_LCD_ALPHA = lcd_osd_info->alpha;
REG_LCD_IPUR = lcd_osd_info->ipu_restart;
}
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* Change Position of Foreground 0 */
static int jz4760fb0_foreground_move(struct jz4760lcd_osd_t *lcd_osd_info)
{
int pos;
int j, count;
/*
* Foreground, only one of the following can be change at one time:
* 1. F0 size, 2. F0 position, 3. F1 size, 4. F1 position
*
* The rules of fg0 position:
* fg0.x + fg0.w <= panel.w;
* fg0.y + fg0.h <= panel.h;
*
* When output is LCD panel, fg.y can be odd number or even number.
* When output is TVE, as the TVE has odd frame and even frame,
* to simplified operation, fg.y should be even number always.
*
*/
/* Foreground 0 */
if (lcd_osd_info->fg0.x + lcd_osd_info->fg0.w > jz4760_lcd_info->panel.w)
lcd_osd_info->fg0.x = jz4760_lcd_info->panel.w - lcd_osd_info->fg0.w;
if (lcd_osd_info->fg0.y + lcd_osd_info->fg0.h > jz4760_lcd_info->panel.h)
lcd_osd_info->fg0.y = jz4760_lcd_info->panel.h - lcd_osd_info->fg0.h;
if (lcd_osd_info->fg0.x >= jz4760_lcd_info->panel.w)
lcd_osd_info->fg0.x = jz4760_lcd_info->panel.w - 1;
if (lcd_osd_info->fg0.y >= jz4760_lcd_info->panel.h)
lcd_osd_info->fg0.y = jz4760_lcd_info->panel.h - 1;
pos = lcd_osd_info->fg0.y << 16 | lcd_osd_info->fg0.x;
if (REG_LCD_XYP0 == pos){
printk("FG0: same position\n");
return 0;
}
#if 0//defined(CONFIG_SOC_JZ4760)
/****jz4760****/
// REG_LCD_OSDC &= ~LCD_OSDC_F0EN;
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
REG_LCD_XYP0 = pos;
// REG_LCD_OSDC |= LCD_OSDC_F0EN;
/*********************************************/
#endif
REG_LCD_XYP0 = pos;
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
while(!(REG_LCD_OSDS & LCD_OSDS_READY));
j = count;
msleep(40);
while((REG_LCD_OSDCTRL & LCD_OSDCTRL_CHANGES) && j--);
if(j == 0) {
printk("Error FG0 Position: Wait change fail.\n");
return -EFAULT;
}
return 0;
}
/* Change Window size of Foreground 0 */
static int jz4760fb0_foreground_resize(struct jz4760lcd_osd_t *lcd_osd_info)
{
struct lcd_cfb_info *cfb = jz4760fb_info;
int size, fg0_line_size, fg0_frm_size;
// int desc_len = sizeof(struct jz4760_lcd_dma_desc);
/*
* NOTE:
* Foreground change sequence:
* 1. Change Position Registers -> LCD_OSDCTL.Change;
* 2. LCD_OSDCTRL.Change -> descripter->Size
* Foreground, only one of the following can be change at one time:
* 1. F0 size;
* 2. F0 position
* 3. F1 size
* 4. F1 position
*/
/*
* The rules of f0, f1's position:
* f0.x + f0.w <= panel.w;
* f0.y + f0.h <= panel.h;
*
* When output is LCD panel, fg.y and fg.h can be odd number or even number.
* When output is TVE, as the TVE has odd frame and even frame,
* to simplified operation, fg.y and fg.h should be even number always.
*
*/
/* Foreground 0 */
if (lcd_osd_info->fg0.x + lcd_osd_info->fg0.w > jz4760_lcd_info->panel.w)
lcd_osd_info->fg0.w = jz4760_lcd_info->panel.w - lcd_osd_info->fg0.x;
if (lcd_osd_info->fg0.y + lcd_osd_info->fg0.h > jz4760_lcd_info->panel.h)
lcd_osd_info->fg0.h = jz4760_lcd_info->panel.h - lcd_osd_info->fg0.y;
size = lcd_osd_info->fg0.h << 16 | lcd_osd_info->fg0.w;
if (REG_LCD_SIZE0 == size) {
printk("FG0: same size\n");
return 0;
}
fg0_line_size = lcd_osd_info->fg0.w * lcd_osd_info->fg0.bpp / 8;
fg0_line_size = ((fg0_line_size + 3) >> 2) << 2; /* word aligned */
fg0_frm_size = fg0_line_size * lcd_osd_info->fg0.h;
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
#if 0 // For 4760d
if (current_dma0_id == 0) {
printk("Change to dma0_desc0_change\n");
// REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma0_desc0_change->cmd = dma0_desc1_change->cmd = (fg0_frm_size/4)/2;
dma0_desc0_change->offsize = dma0_desc1_change->offsize
= fg0_line_size/4;
dma0_desc0_change->page_width = dma0_desc1_change->page_width
= fg0_line_size/4;
dma0_desc1_change->databuf = virt_to_phys((void *)(lcd_frame0 + fg0_line_size));
dma0_desc0_change->desc_size = dma0_desc1->desc_size = size;
dma_cache_wback_inv((unsigned int)(dma0_desc0_change), desc_len);
dma_cache_wback_inv((unsigned int)(dma0_desc1_change), desc_len);
}
else {
dma0_desc0_change->cmd = fg0_frm_size/4;
dma0_desc0_change->offsize = 0;
dma0_desc0_change->page_width = 0;
dma0_desc0_change->desc_size = size;
dma0_desc0_change->next_desc = (unsigned int)virt_to_phys(dma0_desc0_change);
dma_cache_wback_inv((unsigned int)(dma0_desc0_change),desc_len);
}
REG_LCD_SIZE0 = size;
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma0_desc1->next_desc = (unsigned int)virt_to_phys(dma0_desc0_change);
dma_cache_wback_inv((unsigned int)(dma0_desc1_change), desc_len);
}
else {
dma0_desc0->next_desc = (unsigned int)virt_to_phys(dma0_desc0_change);
dma_cache_wback_inv((unsigned int)(dma0_desc0_change), desc_len);
}
current_dma0_id = 1;//dma0_desc0_change;
}
else {
printk("Change to dma0_desc0\n");
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma0_desc0->cmd = dma0_desc1->cmd = (fg0_frm_size/4)/2;
dma0_desc0->offsize = dma0_desc1->offsize
= fg0_line_size/4;
dma0_desc0->page_width = dma0_desc1->page_width
= fg0_line_size/4;
dma0_desc1->databuf = virt_to_phys((void *)(lcd_frame0 + fg0_line_size));
dma0_desc0->desc_size = dma0_desc1->desc_size = size;
}
else {
dma0_desc0->cmd = fg0_frm_size/4;
dma0_desc0->offsize =0;
dma0_desc0->page_width = 0;
dma0_desc0->desc_size = size;
dma0_desc0->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma_cache_wback_inv((unsigned int)(dma0_desc0), desc_len);
}
REG_LCD_SIZE0 = size;
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma0_desc1_change->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma_cache_wback_inv((unsigned int)(dma0_desc1_change), desc_len);
}
else {
dma0_desc0_change->next_desc = (unsigned int)virt_to_phys(dma0_desc0);
dma_cache_wback_inv((unsigned int)(dma0_desc0_change), desc_len);
}
current_dma0_id = 0;//dma0_desc0;
}
#else
/* set change bit */
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) { /* output to TV */
dma0_desc0->cmd = dma0_desc1->cmd = (fg0_frm_size/4)/2;
dma0_desc0->offsize = dma0_desc1->offsize
= fg0_line_size/4;
dma0_desc0->page_width = dma0_desc1->page_width
= fg0_line_size/4;
dma0_desc1->databuf = virt_to_phys((void *)(lcd_frame0 + fg0_line_size));
}
else {
dma0_desc0->cmd = dma0_desc1->cmd = fg0_frm_size/4;
dma0_desc0->offsize = dma0_desc1->offsize =0;
dma0_desc0->page_width = dma0_desc1->page_width = 0;
}
dma0_desc0->desc_size = dma0_desc1->desc_size = size;
// = lcd_osd_info->fg0.h << 16 | lcd_osd_info->fg0.w;
REG_LCD_SIZE0 = size;
// REG_LCD_SIZE0 = (lcd_osd_info->fg0.h << 16) | lcd_osd_info->fg0.w;
dma_cache_wback((unsigned int)(dma_desc_base), (DMA_DESC_NUM)*sizeof(struct jz4760_lcd_dma_desc));
#endif
jz4760fb_set_var(&cfb->fb0.var, 0, &cfb->fb0);
return 0;
}
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* Change Position of Foreground 1 */
static int jz4760fb_foreground_move(struct jz4760lcd_osd_t *lcd_osd_info)
{
int pos;
int j, count = 100000;
/*
* Foreground, only one of the following can be change at one time:
* 1. F0 size, 2. F0 position, 3. F1 size, 4. F1 position
*
* The rules of fg1 position:
* fg1.x + fg1.w <= panel.w;
* fg1.y + fg1.h <= panel.h;
*
* When output is LCD panel, fg.y can be odd number or even number.
* When output is TVE, as the TVE has odd frame and even frame,
* to simplified operation, fg.y should be even number always.
*
*/
/* Foreground 0 */
if (lcd_osd_info->fg1.x + lcd_osd_info->fg1.w > jz4760_lcd_info->panel.w)
lcd_osd_info->fg1.x = jz4760_lcd_info->panel.w - lcd_osd_info->fg1.w;
if (lcd_osd_info->fg1.y + lcd_osd_info->fg1.h > jz4760_lcd_info->panel.h)
lcd_osd_info->fg1.y = jz4760_lcd_info->panel.h - lcd_osd_info->fg1.h;
if (lcd_osd_info->fg1.x >= jz4760_lcd_info->panel.w)
lcd_osd_info->fg1.x = jz4760_lcd_info->panel.w - 1;
if (lcd_osd_info->fg1.y >= jz4760_lcd_info->panel.h)
lcd_osd_info->fg1.y = jz4760_lcd_info->panel.h - 1;
pos = lcd_osd_info->fg1.y << 16 | lcd_osd_info->fg1.x;
if (REG_LCD_XYP1 == pos){
printk("FG1: same position\n");
return 0;
}
#if 0//defined(CONFIG_SOC_JZ4760)
/****jz4760****/
// REG_LCD_OSDC &= ~LCD_OSDC_F0EN;
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
REG_LCD_XYP1 = pos;
// REG_LCD_OSDC |= LCD_OSDC_F0EN;
#endif
/*********************************************/
REG_LCD_XYP1 = pos;
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
while(!(REG_LCD_OSDS & LCD_OSDS_READY));
j = count;
msleep(40);
while((REG_LCD_OSDCTRL & LCD_OSDCTRL_CHANGES) && j--);
if(j == 0) {
printk("Error FG1 Position: Wait change fail.\n");
return -EFAULT;
}
return 0;
}
/* Change window size of Foreground 1 */
static int jz4760fb_foreground_resize(struct jz4760lcd_osd_t *lcd_osd_info)
{
struct lcd_cfb_info *cfb = jz4760fb_info;
int size, fg1_line_size, fg1_frm_size;
// int desc_len = sizeof(struct jz4760_lcd_dma_desc);
/*
* NOTE:
* Foreground change sequence:
* 1. Change Position Registers -> LCD_OSDCTL.Change;
* 2. LCD_OSDCTRL.Change -> descripter->Size
* Foreground, only one of the following can be change at one time:
* 1. F0 size;
* 2. F0 position
* 3. F1 size
* 4. F1 position
*/
/*
* The rules of f0, f1's position:
* f0.x + f0.w <= panel.w;
* f0.y + f0.h <= panel.h;
*
* When output is LCD panel, fg.y and fg.h can be odd number or even number.
* When output is TVE, as the TVE has odd frame and even frame,
* to simplified operation, fg.y and fg.h should be even number always.
*
*/
/* Foreground 0 */
if (lcd_osd_info->fg1.x + lcd_osd_info->fg1.w > jz4760_lcd_info->panel.w)
lcd_osd_info->fg1.w = jz4760_lcd_info->panel.w - lcd_osd_info->fg1.x;
if (lcd_osd_info->fg1.y + lcd_osd_info->fg1.h > jz4760_lcd_info->panel.h)
lcd_osd_info->fg1.h = jz4760_lcd_info->panel.h - lcd_osd_info->fg1.y;
// size = lcd_info->osd.fg1.h << 16|lcd_info->osd.fg1.w;
size = lcd_osd_info->fg1.h << 16|lcd_osd_info->fg1.w;
if (REG_LCD_SIZE1 == size) {
printk("FG1: same size\n");
return 0;// -EFAULT;
}
// fg1_line_size = lcd_osd_info->fg1.w * ((lcd_osd_info->fg1.bpp + 7) / 8);
fg1_line_size = lcd_osd_info->fg1.w * lcd_osd_info->fg1.bpp / 8;
fg1_line_size = ((fg1_line_size + 3) >> 2) << 2; /* word aligned */
fg1_frm_size = fg1_line_size * lcd_osd_info->fg1.h;
#if 0
if (current_dma1_id == 0) {
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma1_desc0_change->cmd = dma1_desc1_change->cmd = (fg1_frm_size/4)/2;
dma1_desc0_change->offsize = dma1_desc1_change->offsize
= fg1_line_size/4;
dma1_desc0_change->page_width = dma1_desc1_change->page_width
= fg1_line_size/4;
dma1_desc1_change->databuf = virt_to_phys((void *)(lcd_frame + fg1_line_size));
dma1_desc0_change->desc_size = dma1_desc1->desc_size = size;
dma_cache_wback_inv((unsigned int)(dma1_desc0_change), desc_len);
dma_cache_wback_inv((unsigned int)(dma1_desc1_change), desc_len);
}
else {
dma1_desc0_change->cmd = fg1_frm_size/4;
dma1_desc0_change->offsize = 0;
dma1_desc0_change->page_width = 0;
dma1_desc0_change->desc_size = size;
dma1_desc0_change->next_desc = (unsigned int)virt_to_phys(dma1_desc0_change);
dma_cache_wback_inv((unsigned int)(dma1_desc0_change),desc_len);
}
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma1_desc1->next_desc = (unsigned int)virt_to_phys(dma1_desc0_change);
dma_cache_wback_inv((unsigned int)(dma1_desc1_change), desc_len);
}
else {
dma1_desc0->next_desc = (unsigned int)virt_to_phys(dma1_desc0_change);
dma_cache_wback_inv((unsigned int)(dma1_desc0_change), desc_len);
}
REG_LCD_SIZE1 = size;
current_dma1_id = 1;//dma1_desc0_change;
}
else {
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma1_desc0->cmd = dma1_desc1->cmd = (fg1_frm_size/4)/2;
dma1_desc0->offsize = dma1_desc1->offsize
= fg1_line_size/4;
dma1_desc0->page_width = dma1_desc1->page_width
= fg1_line_size/4;
dma1_desc1->databuf = virt_to_phys((void *)(lcd_frame + fg1_line_size));
dma1_desc0->desc_size = dma1_desc1->desc_size = size;
}
else {
dma1_desc0->cmd = fg1_frm_size/4;
dma1_desc0->offsize =0;
dma1_desc0->page_width = 0;
dma1_desc0->desc_size = size;
dma1_desc0->next_desc = (unsigned int)virt_to_phys(dma1_desc0);
dma_cache_wback_inv((unsigned int)(dma1_desc0), desc_len);
}
if (jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) {
dma1_desc1_change->next_desc = (unsigned int)virt_to_phys(dma1_desc0);
dma_cache_wback_inv((unsigned int)(dma1_desc1_change), desc_len);
}
else {
dma1_desc0_change->next_desc = (unsigned int)virt_to_phys(dma1_desc0);
dma_cache_wback_inv((unsigned int)(dma1_desc0_change), desc_len);
}
REG_LCD_SIZE1 = size;
current_dma1_id = 0;//dma1_desc0_change;
}
#else
/* set change bit */
REG_LCD_OSDCTRL |= LCD_OSDCTRL_CHANGES;
if ( jz4760_lcd_info->panel.cfg & LCD_CFG_TVEN ) { /* output to TV */
dma1_desc0->cmd = dma1_desc1->cmd = (fg1_frm_size/4)/2;
dma1_desc0->offsize = dma1_desc1->offsize = fg1_line_size/4;
dma1_desc0->page_width = dma1_desc1->page_width = fg1_line_size/4;
dma1_desc1->databuf = virt_to_phys((void *)(lcd_frame + fg1_line_size));
}
else {
dma1_desc0->cmd = dma1_desc1->cmd = fg1_frm_size/4;
dma1_desc0->offsize = dma1_desc1->offsize = 0;
dma1_desc0->page_width = dma1_desc1->page_width = 0;
}
dma1_desc0->desc_size = dma1_desc1->desc_size = size;
REG_LCD_SIZE1 = size;
dma_cache_wback((unsigned int)(dma_desc_base), (DMA_DESC_NUM)*sizeof(struct jz4760_lcd_dma_desc));
#endif
jz4760fb_set_var(&cfb->fb.var, 1, &cfb->fb);
return 0;
}
#endif
/*
* Set lcd pixel clock
*/
static void jz4760fb_change_clock( struct jz4760lcd_info * lcd_info )
{
#if defined(CONFIG_FPGA) /* FPGA test, pixdiv */
REG_LCD_REV = 0x00000004;
printk("Fuwa test, pixclk divide REG_LCD_REV=0x%08x\n", REG_LCD_REV);
printk("Fuwa test, pixclk %d\n", JZ_EXTAL/(((REG_LCD_REV&0xFF)+1)*2));
#else
unsigned int val = 0;
unsigned int pclk;
/* Timing setting */
__cpm_stop_lcd();
val = lcd_info->panel.fclk; /* frame clk */
if ( (lcd_info->panel.cfg & LCD_CFG_MODE_MASK) != LCD_CFG_MODE_SERIAL_TFT) {
pclk = val * (lcd_info->panel.w + lcd_info->panel.hsw + lcd_info->panel.elw + lcd_info->panel.blw) * (lcd_info->panel.h + lcd_info->panel.vsw + lcd_info->panel.efw + lcd_info->panel.bfw); /* Pixclk */
}
else {
/* serial mode: Hsync period = 3*Width_Pixel */
pclk = val * (lcd_info->panel.w*3 + lcd_info->panel.hsw + lcd_info->panel.elw + lcd_info->panel.blw) * (lcd_info->panel.h + lcd_info->panel.vsw + lcd_info->panel.efw + lcd_info->panel.bfw); /* Pixclk */
}
/********* In TVE mode PCLK = 27MHz ***********/
if ( lcd_info->panel.cfg & LCD_CFG_TVEN ) { /* LCDC output to TVE */
pclk = 27000000;
__cpm_select_pixclk_tve();
}
else { /* LCDC output to LCD panel */
__cpm_select_pixclk_lcd();
}
val = __cpm_get_pllout2() / pclk; /* pclk */
val--;
dprintk("ratio: val = %d\n", val);
if ( val > 0x7ff ) {
printk("pixel clock divid is too large, set it to 0x7ff\n");
val = 0x7ff;
}
__cpm_set_pixdiv(val);
dprintk("REG_CPM_LPCDR = 0x%08x\n", REG_CPM_LPCDR);
__cpm_enable_pll_change();
dprintk("REG_CPM_LPCDR=0x%08x\n", REG_CPM_LPCDR);
dprintk("REG_CPM_CPCCR=0x%08x\n", REG_CPM_CPCCR);
jz_clocks.pixclk = __cpm_get_pixclk();
printk("LCDC: PixClock:%d\n", jz_clocks.pixclk);
__cpm_start_lcd();
udelay(1000);
/*
* set lcd device clock and lcd pixel clock.
* what about TVE mode???
*
*/
#endif
}
/*
* jz4760fb_set_mode(), set osd configure, resize foreground
*
*/
static void jz4760fb_set_mode(struct jz4760lcd_osd_t * lcd_osd_info)
//static void jz4760fb_set_mode(struct jz4760lcd_info * lcd_info)
{
// struct jz4760lcd_osd_t * lcd_osd_info = jz4760_lcd_info->osd;
// struct lcd_cfb_info *cfb = jz4760fb_info;
jz4760fb_set_osd_mode(lcd_osd_info);
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
jz4760fb_foreground_resize(lcd_osd_info);
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
jz4760fb0_foreground_resize(lcd_osd_info);
#endif
}
/*
* jz4760fb_deep_set_mode,
*
*/
static void jz4760fb_deep_set_mode( struct jz4760lcd_info * lcd_info )
{
/* configurate sequence:
* 1. disable lcdc.
* 2. init frame descriptor.
* 3. set panel mode
* 4. set osd mode
* 5. start lcd clock in CPM
* 6. enable lcdc.
*/
struct lcd_cfb_info *cfb = jz4760fb_info;
__lcd_clr_ena(); /* Quick Disable */
lcd_info->osd.fg_change = FG_CHANGE_ALL; /* change FG0, FG1 size, postion??? */
jz4760fb_set_osd_mode(&lcd_info->osd);
jz4760fb_set_panel_mode(lcd_info);
jz4760fb_descriptor_init(lcd_info);
jz4760fb_change_clock(lcd_info);
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
jz4760fb_set_var(&cfb->fb0.var, 0, &cfb->fb0);
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
jz4760fb_set_var(&cfb->fb.var, 1, &cfb->fb);
#endif
__lcd_set_ena(); /* enable lcdc */
}
static irqreturn_t jz4760fb_interrupt_handler(int irq, void *dev_id)
{
unsigned int state;
static int irqcnt=0;
state = REG_LCD_STATE;
dprintk("In the lcd interrupt handler, state=0x%x\n", state);
if (state & LCD_STATE_EOF) /* End of frame */
REG_LCD_STATE = state & ~LCD_STATE_EOF;
if (state & LCD_STATE_IFU0) {
printk("%s, InFiFo0 underrun\n", __FUNCTION__);
REG_LCD_STATE = state & ~LCD_STATE_IFU0;
}
if (state & LCD_STATE_IFU1) {
printk("%s, InFiFo1 underrun\n", __FUNCTION__);
REG_LCD_STATE = state & ~LCD_STATE_IFU1;
}
if (state & LCD_STATE_OFU) { /* Out fifo underrun */
REG_LCD_STATE = state & ~LCD_STATE_OFU;
if ( irqcnt++ > 100 ) {
__lcd_disable_ofu_intr();
printk("disable Out FiFo underrun irq.\n");
}
printk("%s, Out FiFo underrun.\n", __FUNCTION__);
}
return IRQ_HANDLED;
}
#ifdef CONFIG_PM
/*
* Suspend the LCDC.
*/
static int jz4760fb_suspend(struct platform_device *pdev, pm_message_t state)
{
__lcd_clr_ena(); /* Quick Disable */
__lcd_display_off();
__cpm_stop_lcd();
return 0;
}
/*
* Resume the LCDC.
*/
static int jz4760fb_resume(struct platform_device *pdev)
{
__cpm_start_lcd();
REG_LCD_DA1 = virt_to_phys(dma1_desc0);
__lcd_set_ena();
__lcd_display_on();
return 0;
}
#else
static int jz4760fb_suspend(struct device *dev, pm_message_t state) {return 0;}
static int jz4760fb_resume(struct device *dev) {return 0;}
#endif /* CONFIG_PM */
/* The following routine is only for test */
static void jz4760_lcd_gpio_init(void)
{
/* gpio init __gpio_as_lcd */
if (jz4760_lcd_info->panel.cfg & LCD_CFG_MODE_TFT_16BIT)
__gpio_as_lcd_16bit();
else if (jz4760_lcd_info->panel.cfg & LCD_CFG_MODE_TFT_24BIT)
__gpio_as_lcd_24bit();
else
__gpio_as_lcd_18bit();
/* Configure SLCD module for setting smart lcd control registers */
#if defined(CONFIG_FB_JZ4760_SLCD)
__lcd_as_smart_lcd();
__slcd_disable_dma();
__init_slcd_bus(); /* Note: modify this depend on you lcd */
#endif
__lcd_display_pin_init();
}
static void jz4760_lcd_init_cfg(void)
{
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
jz4760_lcd_info->osd.osd_cfg |= LCD_OSDC_F0EN; /* only open fg0 */
#endif
// jz4760_lcd_info->osd.osd_cfg |= LCD_OSDC_F1EN; /* only open fg1 */
/* In special mode, we only need init special pin,
* as general lcd pin has init in uboot */
#if defined(CONFIG_SOC_JZ4760)
switch (jz4760_lcd_info->panel.cfg & LCD_CFG_MODE_MASK) {
case LCD_CFG_MODE_SPECIAL_TFT_1:
case LCD_CFG_MODE_SPECIAL_TFT_2:
case LCD_CFG_MODE_SPECIAL_TFT_3:
__gpio_as_lcd_special();
break;
default:
break;
}
#endif
/* Foreground 0 support bpp = 1, 2, 4, 8, 15, 16, 18, 24 */
switch ( jz4760_lcd_info->osd.fg0.bpp ) {
case 17 ... 32:
jz4760_lcd_info->osd.fg1.bpp = 32;
break;
default:
break;
}
/* Foreground 1 support bpp = 15, 16, 18, 24 */
switch ( jz4760_lcd_info->osd.fg1.bpp ) {
case 15:
case 16:
break;
case 17 ... 32:
jz4760_lcd_info->osd.fg1.bpp = 32;
break;
default:
printk("jz4760fb fg1 not support bpp(%d), force to 32bpp\n",
jz4760_lcd_info->osd.fg1.bpp);
jz4760_lcd_info->osd.fg1.bpp = 32;
}
}
#ifdef CONFIG_LEDS_CLASS
static void lcd_set_backlight_level(struct led_classdev *led_cdev, enum led_brightness value) {
__lcd_set_backlight_level((int)value);
}
static struct led_classdev lcd_backlight_led = {
.name = "lcd-backlight",
.brightness_set = lcd_set_backlight_level,
};
#endif
static int __init jz4760fb_probe(struct platform_device *pdev)
{
struct lcd_cfb_info *cfb;
int err = 0;
__lcd_close_backlight();
if (!pdev)
return -EINVAL;
jz4760_lcd_gpio_init(); /* gpio init */
jz4760_lcd_init_cfg(); /* first config of lcd */
__lcd_clr_dis();
__lcd_clr_ena();
/* init clock */
__lcd_slcd_special_on();
cfb = jz4760fb_alloc_fb_info();
if (!cfb)
goto failed;
err = jz4760fb_map_smem(cfb);
if (err)
goto failed;
jz4760fb_deep_set_mode( jz4760_lcd_info );
/* registers frame buffer devices */
#if defined(CONFIG_JZ4760_LCD_USE_FG0_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* register fg0 */
err = register_framebuffer(&cfb->fb0);
if (err < 0) {
dprintk("jz4760fb_init(): register framebuffer err.\n");
goto failed;
}
printk("fb%d: %s frame buffer device, using %dK of video memory\n",
cfb->fb0.node, cfb->fb0.fix.id, cfb->fb0.fix.smem_len>>10);
#endif
#if defined(CONFIG_JZ4760_LCD_USE_FG1_ONLY) || defined(CONFIG_JZ4760_LCD_USE_2LAYER_FG)
/* register fg1 */
err = register_framebuffer(&cfb->fb);
if (err < 0) {
dprintk("jz4760fb_init(): register framebuffer err.\n");
goto failed;
}
printk("fb%d: %s frame buffer device, using %dK of video memory\n",
cfb->fb.node, cfb->fb.fix.id, cfb->fb.fix.smem_len>>10);
#endif
if (request_irq(IRQ_LCD, jz4760fb_interrupt_handler, IRQF_DISABLED,
"lcd", 0)) {
err = -EBUSY;
goto failed;
}
// display_h_color_bar((void *)lcd_frame, jz4760_lcd_info->osd.fg1.w, jz4760_lcd_info->osd.fg1.h, jz4760_lcd_info->osd.fg1.bpp);
#ifdef CONFIG_LEDS_CLASS
err = led_classdev_register(&pdev->dev, &lcd_backlight_led);
if (err < 0)
goto failed;
#endif
__lcd_set_ena(); /* enalbe LCD Controller */
__lcd_display_on();
print_lcdc_registers();
pm_set_vt_switch(0); /*disable VT switch during suspend/resume*/
return 0;
failed:
print_dbg();
jz4760fb_unmap_smem(cfb);
jz4760fb_free_fb_info(cfb);
return err;
}
static int jz4760fb_remove(struct platform_device *pdev)
{
struct lcd_cfb_info *cfb = platform_get_drvdata(pdev);
jz4760fb_unmap_smem(cfb);
jz4760fb_free_fb_info(cfb);
return 0;
}
static struct platform_driver jz_lcd_driver = {
.probe = jz4760fb_probe,
.remove = jz4760fb_remove,
#ifdef CONFIG_PM
.suspend = jz4760fb_suspend,
.resume = jz4760fb_resume,
#endif
.driver = {
.name = DRIVER_NAME,
},
};
static int __init jz4760fb_init(void)
{
return platform_driver_register(&jz_lcd_driver);
}
static void __exit jz4760fb_cleanup(void)
{
platform_driver_unregister(&jz_lcd_driver);
}
module_init(jz4760fb_init);
module_exit(jz4760fb_cleanup);