2008-03-20 03:01:10 +03:00
/*
* linux / drivers / video / metronomefb . c - - FB driver for Metronome controller
*
* Copyright ( C ) 2008 , Jaya Kumar
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of this archive for
* more details .
*
* Layout is based on skeletonfb . c by James Simmons and Geert Uytterhoeven .
*
* This work was made possible by help and equipment support from E - Ink
2010-10-18 13:03:14 +04:00
* Corporation . http : //www.eink.com/
2008-03-20 03:01:10 +03:00
*
* This driver is written to be used with the Metronome display controller .
2008-04-28 13:15:38 +04:00
* It is intended to be architecture independent . A board specific driver
* must be used to perform all the physical IO interactions . An example
* is provided as am200epd . c
2008-03-20 03:01:10 +03:00
*
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/mm.h>
# include <linux/vmalloc.h>
# include <linux/delay.h>
# include <linux/interrupt.h>
# include <linux/fb.h>
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/list.h>
# include <linux/firmware.h>
# include <linux/dma-mapping.h>
# include <linux/uaccess.h>
# include <linux/irq.h>
2008-04-28 13:15:38 +04:00
# include <video/metronomefb.h>
2008-03-20 03:01:10 +03:00
# include <asm/unaligned.h>
/* Display specific information */
# define DPY_W 832
# define DPY_H 622
2008-08-19 14:17:55 +04:00
static int user_wfm_size ;
2008-03-20 03:01:10 +03:00
/* frame differs from image. frame includes non-visible pixels */
struct epd_frame {
int fw ; /* frame width */
int fh ; /* frame height */
2008-08-19 14:17:55 +04:00
u16 config [ 4 ] ;
int wfm_size ;
2008-03-20 03:01:10 +03:00
} ;
static struct epd_frame epd_frame_table [ ] = {
{
2008-08-19 14:17:55 +04:00
. fw = 832 ,
. fh = 622 ,
. config = {
15 /* sdlew */
| 2 < < 8 /* sdosz */
| 0 < < 11 /* sdor */
| 0 < < 12 /* sdces */
| 0 < < 15 , /* sdcer */
42 /* gdspl */
| 1 < < 8 /* gdr1 */
| 1 < < 9 /* sdshr */
| 0 < < 15 , /* gdspp */
18 /* gdspw */
| 0 < < 15 , /* dispc */
599 /* vdlc */
| 0 < < 11 /* dsi */
| 0 < < 12 , /* dsic */
} ,
. wfm_size = 47001 ,
} ,
{
. fw = 1088 ,
. fh = 791 ,
. config = {
0x0104 ,
0x031f ,
0x0088 ,
0x02ff ,
} ,
. wfm_size = 46770 ,
} ,
{
. fw = 1200 ,
. fh = 842 ,
. config = {
0x0101 ,
0x030e ,
0x0012 ,
0x0280 ,
} ,
. wfm_size = 46770 ,
2008-03-20 03:01:10 +03:00
} ,
} ;
2012-12-22 01:07:39 +04:00
static struct fb_fix_screeninfo metronomefb_fix = {
2008-03-20 03:01:10 +03:00
. id = " metronomefb " ,
. type = FB_TYPE_PACKED_PIXELS ,
. visual = FB_VISUAL_STATIC_PSEUDOCOLOR ,
. xpanstep = 0 ,
. ypanstep = 0 ,
. ywrapstep = 0 ,
. line_length = DPY_W ,
. accel = FB_ACCEL_NONE ,
} ;
2012-12-22 01:07:39 +04:00
static struct fb_var_screeninfo metronomefb_var = {
2008-03-20 03:01:10 +03:00
. xres = DPY_W ,
. yres = DPY_H ,
. xres_virtual = DPY_W ,
. yres_virtual = DPY_H ,
. bits_per_pixel = 8 ,
. grayscale = 1 ,
. nonstd = 1 ,
. red = { 4 , 3 , 0 } ,
. green = { 0 , 0 , 0 } ,
. blue = { 0 , 0 , 0 } ,
. transp = { 0 , 0 , 0 } ,
} ;
2008-04-28 13:15:38 +04:00
/* the waveform structure that is coming from userspace firmware */
2008-03-20 03:01:10 +03:00
struct waveform_hdr {
u8 stuff [ 32 ] ;
u8 wmta [ 3 ] ;
u8 fvsn ;
u8 luts ;
u8 mc ;
u8 trc ;
u8 stuff3 ;
u8 endb ;
u8 swtb ;
u8 stuff2a [ 2 ] ;
u8 stuff2b [ 3 ] ;
u8 wfm_cs ;
} __attribute__ ( ( packed ) ) ;
/* main metronomefb functions */
static u8 calc_cksum ( int start , int end , u8 * mem )
{
u8 tmp = 0 ;
int i ;
for ( i = start ; i < end ; i + + )
tmp + = mem [ i ] ;
return tmp ;
}
static u16 calc_img_cksum ( u16 * start , int length )
{
u16 tmp = 0 ;
while ( length - - )
tmp + = * start + + ;
return tmp ;
}
/* here we decode the incoming waveform file and populate metromem */
2012-12-22 01:07:39 +04:00
static int load_waveform ( u8 * mem , size_t size , int m , int t ,
struct metronomefb_par * par )
2008-03-20 03:01:10 +03:00
{
int tta ;
int wmta ;
int trn = 0 ;
int i ;
unsigned char v ;
u8 cksum ;
int cksum_idx ;
int wfm_idx , owfm_idx ;
int mem_idx = 0 ;
struct waveform_hdr * wfm_hdr ;
2008-08-05 16:52:14 +04:00
u8 * metromem = par - > metromem_wfm ;
struct device * dev = par - > info - > dev ;
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
if ( user_wfm_size )
epd_frame_table [ par - > dt ] . wfm_size = user_wfm_size ;
if ( size ! = epd_frame_table [ par - > dt ] . wfm_size ) {
2008-10-15 13:50:32 +04:00
dev_err ( dev , " Error: unexpected size %Zd != %d \n " , size ,
2008-08-19 14:17:55 +04:00
epd_frame_table [ par - > dt ] . wfm_size ) ;
2008-03-20 03:01:10 +03:00
return - EINVAL ;
}
wfm_hdr = ( struct waveform_hdr * ) mem ;
if ( wfm_hdr - > fvsn ! = 1 ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad fvsn %x \n " , wfm_hdr - > fvsn ) ;
2008-03-20 03:01:10 +03:00
return - EINVAL ;
}
if ( wfm_hdr - > luts ! = 0 ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad luts %x \n " , wfm_hdr - > luts ) ;
2008-03-20 03:01:10 +03:00
return - EINVAL ;
}
cksum = calc_cksum ( 32 , 47 , mem ) ;
if ( cksum ! = wfm_hdr - > wfm_cs ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad cksum %x != %x \n " , cksum ,
2008-03-20 03:01:10 +03:00
wfm_hdr - > wfm_cs ) ;
return - EINVAL ;
}
wfm_hdr - > mc + = 1 ;
wfm_hdr - > trc + = 1 ;
for ( i = 0 ; i < 5 ; i + + ) {
if ( * ( wfm_hdr - > stuff2a + i ) ! = 0 ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: unexpected value in padding \n " ) ;
2008-03-20 03:01:10 +03:00
return - EINVAL ;
}
}
/* calculating trn. trn is something used to index into
the waveform . presumably selecting the right one for the
desired temperature . it works out the offset of the first
v that exceeds the specified temperature */
if ( ( sizeof ( * wfm_hdr ) + wfm_hdr - > trc ) > size )
return - EINVAL ;
for ( i = sizeof ( * wfm_hdr ) ; i < = sizeof ( * wfm_hdr ) + wfm_hdr - > trc ; i + + ) {
if ( mem [ i ] > t ) {
trn = i - sizeof ( * wfm_hdr ) - 1 ;
break ;
}
}
/* check temperature range table checksum */
cksum_idx = sizeof ( * wfm_hdr ) + wfm_hdr - > trc + 1 ;
if ( cksum_idx > size )
return - EINVAL ;
cksum = calc_cksum ( sizeof ( * wfm_hdr ) , cksum_idx , mem ) ;
if ( cksum ! = mem [ cksum_idx ] ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad temperature range table cksum "
2008-03-20 03:01:10 +03:00
" %x != %x \n " , cksum , mem [ cksum_idx ] ) ;
return - EINVAL ;
}
/* check waveform mode table address checksum */
2008-04-29 12:03:41 +04:00
wmta = get_unaligned_le32 ( wfm_hdr - > wmta ) & 0x00FFFFFF ;
2008-03-20 03:01:10 +03:00
cksum_idx = wmta + m * 4 + 3 ;
if ( cksum_idx > size )
return - EINVAL ;
cksum = calc_cksum ( cksum_idx - 3 , cksum_idx , mem ) ;
if ( cksum ! = mem [ cksum_idx ] ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad mode table address cksum "
2008-03-20 03:01:10 +03:00
" %x != %x \n " , cksum , mem [ cksum_idx ] ) ;
return - EINVAL ;
}
/* check waveform temperature table address checksum */
2008-04-29 12:03:41 +04:00
tta = get_unaligned_le32 ( mem + wmta + m * 4 ) & 0x00FFFFFF ;
2008-03-20 03:01:10 +03:00
cksum_idx = tta + trn * 4 + 3 ;
if ( cksum_idx > size )
return - EINVAL ;
cksum = calc_cksum ( cksum_idx - 3 , cksum_idx , mem ) ;
if ( cksum ! = mem [ cksum_idx ] ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad temperature table address cksum "
2008-03-20 03:01:10 +03:00
" %x != %x \n " , cksum , mem [ cksum_idx ] ) ;
return - EINVAL ;
}
/* here we do the real work of putting the waveform into the
metromem buffer . this does runlength decoding of the waveform */
2008-04-29 12:03:41 +04:00
wfm_idx = get_unaligned_le32 ( mem + tta + trn * 4 ) & 0x00FFFFFF ;
2008-03-20 03:01:10 +03:00
owfm_idx = wfm_idx ;
if ( wfm_idx > size )
return - EINVAL ;
while ( wfm_idx < size ) {
unsigned char rl ;
v = mem [ wfm_idx + + ] ;
if ( v = = wfm_hdr - > swtb ) {
while ( ( ( v = mem [ wfm_idx + + ] ) ! = wfm_hdr - > swtb ) & &
wfm_idx < size )
metromem [ mem_idx + + ] = v ;
continue ;
}
if ( v = = wfm_hdr - > endb )
break ;
rl = mem [ wfm_idx + + ] ;
for ( i = 0 ; i < = rl ; i + + )
metromem [ mem_idx + + ] = v ;
}
cksum_idx = wfm_idx ;
if ( cksum_idx > size )
return - EINVAL ;
cksum = calc_cksum ( owfm_idx , cksum_idx , mem ) ;
if ( cksum ! = mem [ cksum_idx ] ) {
2008-08-05 16:52:14 +04:00
dev_err ( dev , " Error: bad waveform data cksum "
2008-03-20 03:01:10 +03:00
" %x != %x \n " , cksum , mem [ cksum_idx ] ) ;
return - EINVAL ;
}
2008-08-05 16:52:14 +04:00
par - > frame_count = ( mem_idx / 64 ) ;
2008-03-20 03:01:10 +03:00
return 0 ;
}
static int metronome_display_cmd ( struct metronomefb_par * par )
{
int i ;
u16 cs ;
u16 opcode ;
static u8 borderval ;
/* setup display command
we can ' t immediately set the opcode since the controller
will try parse the command before we ' ve set it all up
so we just set cs here and set the opcode at the end */
if ( par - > metromem_cmd - > opcode = = 0xCC40 )
opcode = cs = 0xCC41 ;
else
opcode = cs = 0xCC40 ;
/* set the args ( 2 bytes ) for display */
i = 0 ;
par - > metromem_cmd - > args [ i ] = 1 < < 3 /* border update */
| ( ( borderval + + % 4 ) & 0x0F ) < < 4
| ( par - > frame_count - 1 ) < < 8 ;
cs + = par - > metromem_cmd - > args [ i + + ] ;
/* the rest are 0 */
memset ( ( u8 * ) ( par - > metromem_cmd - > args + i ) , 0 , ( 32 - i ) * 2 ) ;
par - > metromem_cmd - > csum = cs ;
par - > metromem_cmd - > opcode = opcode ; /* display cmd */
2008-04-28 13:15:38 +04:00
return par - > board - > met_wait_event_intr ( par ) ;
2008-03-20 03:01:10 +03:00
}
2012-12-22 01:07:39 +04:00
static int metronome_powerup_cmd ( struct metronomefb_par * par )
2008-03-20 03:01:10 +03:00
{
int i ;
u16 cs ;
/* setup power up command */
par - > metromem_cmd - > opcode = 0x1234 ; /* pwr up pseudo cmd */
cs = par - > metromem_cmd - > opcode ;
/* set pwr1,2,3 to 1024 */
for ( i = 0 ; i < 3 ; i + + ) {
par - > metromem_cmd - > args [ i ] = 1024 ;
cs + = par - > metromem_cmd - > args [ i ] ;
}
/* the rest are 0 */
memset ( ( u8 * ) ( par - > metromem_cmd - > args + i ) , 0 , ( 32 - i ) * 2 ) ;
par - > metromem_cmd - > csum = cs ;
msleep ( 1 ) ;
2008-04-28 13:15:38 +04:00
par - > board - > set_rst ( par , 1 ) ;
2008-03-20 03:01:10 +03:00
msleep ( 1 ) ;
2008-04-28 13:15:38 +04:00
par - > board - > set_stdby ( par , 1 ) ;
2008-03-20 03:01:10 +03:00
2008-04-28 13:15:38 +04:00
return par - > board - > met_wait_event ( par ) ;
2008-03-20 03:01:10 +03:00
}
2012-12-22 01:07:39 +04:00
static int metronome_config_cmd ( struct metronomefb_par * par )
2008-03-20 03:01:10 +03:00
{
/* setup config command
we can ' t immediately set the opcode since the controller
2008-08-19 14:17:55 +04:00
will try parse the command before we ' ve set it all up */
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
memcpy ( par - > metromem_cmd - > args , epd_frame_table [ par - > dt ] . config ,
sizeof ( epd_frame_table [ par - > dt ] . config ) ) ;
2008-03-20 03:01:10 +03:00
/* the rest are 0 */
2008-08-19 14:17:55 +04:00
memset ( ( u8 * ) ( par - > metromem_cmd - > args + 4 ) , 0 , ( 32 - 4 ) * 2 ) ;
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
par - > metromem_cmd - > csum = 0xCC10 ;
par - > metromem_cmd - > csum + = calc_img_cksum ( par - > metromem_cmd - > args , 4 ) ;
2008-03-20 03:01:10 +03:00
par - > metromem_cmd - > opcode = 0xCC10 ; /* config cmd */
2008-04-28 13:15:38 +04:00
return par - > board - > met_wait_event ( par ) ;
2008-03-20 03:01:10 +03:00
}
2012-12-22 01:07:39 +04:00
static int metronome_init_cmd ( struct metronomefb_par * par )
2008-03-20 03:01:10 +03:00
{
int i ;
u16 cs ;
/* setup init command
we can ' t immediately set the opcode since the controller
will try parse the command before we ' ve set it all up
so we just set cs here and set the opcode at the end */
cs = 0xCC20 ;
/* set the args ( 2 bytes ) for init */
i = 0 ;
par - > metromem_cmd - > args [ i ] = 0 ;
cs + = par - > metromem_cmd - > args [ i + + ] ;
/* the rest are 0 */
memset ( ( u8 * ) ( par - > metromem_cmd - > args + i ) , 0 , ( 32 - i ) * 2 ) ;
par - > metromem_cmd - > csum = cs ;
par - > metromem_cmd - > opcode = 0xCC20 ; /* init cmd */
2008-04-28 13:15:38 +04:00
return par - > board - > met_wait_event ( par ) ;
2008-03-20 03:01:10 +03:00
}
2012-12-22 01:07:39 +04:00
static int metronome_init_regs ( struct metronomefb_par * par )
2008-03-20 03:01:10 +03:00
{
int res ;
2008-08-19 14:17:55 +04:00
res = par - > board - > setup_io ( par ) ;
if ( res )
return res ;
2008-03-20 03:01:10 +03:00
res = metronome_powerup_cmd ( par ) ;
if ( res )
return res ;
res = metronome_config_cmd ( par ) ;
if ( res )
return res ;
res = metronome_init_cmd ( par ) ;
return res ;
}
static void metronomefb_dpy_update ( struct metronomefb_par * par )
{
2008-08-19 14:17:55 +04:00
int fbsize ;
2008-03-20 03:01:10 +03:00
u16 cksum ;
unsigned char * buf = ( unsigned char __force * ) par - > info - > screen_base ;
2008-08-19 14:17:55 +04:00
fbsize = par - > info - > fix . smem_len ;
2008-03-20 03:01:10 +03:00
/* copy from vm to metromem */
2008-08-19 14:17:55 +04:00
memcpy ( par - > metromem_img , buf , fbsize ) ;
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
cksum = calc_img_cksum ( ( u16 * ) par - > metromem_img , fbsize / 2 ) ;
* ( ( u16 * ) ( par - > metromem_img ) + fbsize / 2 ) = cksum ;
2008-03-20 03:01:10 +03:00
metronome_display_cmd ( par ) ;
}
static u16 metronomefb_dpy_update_page ( struct metronomefb_par * par , int index )
{
int i ;
u16 csum = 0 ;
2008-04-28 13:15:38 +04:00
u16 * buf = ( u16 __force * ) ( par - > info - > screen_base + index ) ;
u16 * img = ( u16 * ) ( par - > metromem_img + index ) ;
2008-03-20 03:01:10 +03:00
/* swizzle from vm to metromem and recalc cksum at the same time*/
for ( i = 0 ; i < PAGE_SIZE / 2 ; i + + ) {
* ( img + i ) = ( buf [ i ] < < 5 ) & 0xE0E0 ;
csum + = * ( img + i ) ;
}
return csum ;
}
/* this is called back from the deferred io workqueue */
static void metronomefb_dpy_deferred_io ( struct fb_info * info ,
struct list_head * pagelist )
{
u16 cksum ;
struct page * cur ;
struct fb_deferred_io * fbdefio = info - > fbdefio ;
struct metronomefb_par * par = info - > par ;
/* walk the written page list and swizzle the data */
list_for_each_entry ( cur , & fbdefio - > pagelist , lru ) {
cksum = metronomefb_dpy_update_page ( par ,
( cur - > index < < PAGE_SHIFT ) ) ;
par - > metromem_img_csum - = par - > csum_table [ cur - > index ] ;
par - > csum_table [ cur - > index ] = cksum ;
par - > metromem_img_csum + = cksum ;
}
metronome_display_cmd ( par ) ;
}
static void metronomefb_fillrect ( struct fb_info * info ,
const struct fb_fillrect * rect )
{
struct metronomefb_par * par = info - > par ;
2008-04-28 13:15:36 +04:00
sys_fillrect ( info , rect ) ;
2008-03-20 03:01:10 +03:00
metronomefb_dpy_update ( par ) ;
}
static void metronomefb_copyarea ( struct fb_info * info ,
const struct fb_copyarea * area )
{
struct metronomefb_par * par = info - > par ;
2008-04-28 13:15:36 +04:00
sys_copyarea ( info , area ) ;
2008-03-20 03:01:10 +03:00
metronomefb_dpy_update ( par ) ;
}
static void metronomefb_imageblit ( struct fb_info * info ,
const struct fb_image * image )
{
struct metronomefb_par * par = info - > par ;
2008-04-28 13:15:36 +04:00
sys_imageblit ( info , image ) ;
2008-03-20 03:01:10 +03:00
metronomefb_dpy_update ( par ) ;
}
/*
* this is the slow path from userspace . they can seek and write to
* the fb . it is based on fb_sys_write
*/
static ssize_t metronomefb_write ( struct fb_info * info , const char __user * buf ,
size_t count , loff_t * ppos )
{
struct metronomefb_par * par = info - > par ;
unsigned long p = * ppos ;
void * dst ;
int err = 0 ;
unsigned long total_size ;
if ( info - > state ! = FBINFO_STATE_RUNNING )
return - EPERM ;
total_size = info - > fix . smem_len ;
if ( p > total_size )
return - EFBIG ;
if ( count > total_size ) {
err = - EFBIG ;
count = total_size ;
}
if ( count + p > total_size ) {
if ( ! err )
err = - ENOSPC ;
count = total_size - p ;
}
2008-04-28 13:15:38 +04:00
dst = ( void __force * ) ( info - > screen_base + p ) ;
2008-03-20 03:01:10 +03:00
if ( copy_from_user ( dst , buf , count ) )
err = - EFAULT ;
if ( ! err )
* ppos + = count ;
metronomefb_dpy_update ( par ) ;
return ( err ) ? err : count ;
}
static struct fb_ops metronomefb_ops = {
. owner = THIS_MODULE ,
. fb_write = metronomefb_write ,
. fb_fillrect = metronomefb_fillrect ,
. fb_copyarea = metronomefb_copyarea ,
. fb_imageblit = metronomefb_imageblit ,
} ;
static struct fb_deferred_io metronomefb_defio = {
. delay = HZ ,
. deferred_io = metronomefb_dpy_deferred_io ,
} ;
2012-12-22 01:07:39 +04:00
static int metronomefb_probe ( struct platform_device * dev )
2008-03-20 03:01:10 +03:00
{
struct fb_info * info ;
2008-04-28 13:15:38 +04:00
struct metronome_board * board ;
2008-03-20 03:01:10 +03:00
int retval = - ENOMEM ;
int videomemorysize ;
unsigned char * videomemory ;
struct metronomefb_par * par ;
const struct firmware * fw_entry ;
int i ;
2008-08-19 14:17:55 +04:00
int panel_type ;
int fw , fh ;
int epd_dt_index ;
2008-03-20 03:01:10 +03:00
2008-04-28 13:15:38 +04:00
/* pick up board specific routines */
board = dev - > dev . platform_data ;
if ( ! board )
return - EINVAL ;
/* try to count device specific driver, if can't, platform recalls */
if ( ! try_module_get ( board - > owner ) )
return - ENODEV ;
2008-08-19 14:17:55 +04:00
info = framebuffer_alloc ( sizeof ( struct metronomefb_par ) , & dev - > dev ) ;
if ( ! info )
goto err ;
2008-03-20 03:01:10 +03:00
/* we have two blocks of memory.
info - > screen_base which is vm , and is the fb used by apps .
par - > metromem which is physically contiguous memory and
contains the display controller commands , waveform ,
processed image data and padding . this is the data pulled
2008-08-19 14:17:55 +04:00
by the device ' s LCD controller and pushed to Metronome .
the metromem memory is allocated by the board driver and
is provided to us */
panel_type = board - > get_panel_type ( ) ;
switch ( panel_type ) {
case 6 :
epd_dt_index = 0 ;
break ;
case 8 :
epd_dt_index = 1 ;
break ;
case 97 :
epd_dt_index = 2 ;
break ;
default :
dev_err ( & dev - > dev , " Unexpected panel type. Defaulting to 6 \n " ) ;
epd_dt_index = 0 ;
break ;
}
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
fw = epd_frame_table [ epd_dt_index ] . fw ;
fh = epd_frame_table [ epd_dt_index ] . fh ;
/* we need to add a spare page because our csum caching scheme walks
* to the end of the page */
videomemorysize = PAGE_SIZE + ( fw * fh ) ;
2011-05-28 22:13:33 +04:00
videomemory = vzalloc ( videomemorysize ) ;
2008-03-20 03:01:10 +03:00
if ( ! videomemory )
2008-08-19 14:17:55 +04:00
goto err_fb_rel ;
2008-03-20 03:01:10 +03:00
2008-04-28 13:15:38 +04:00
info - > screen_base = ( char __force __iomem * ) videomemory ;
2008-03-20 03:01:10 +03:00
info - > fbops = & metronomefb_ops ;
2008-08-19 14:17:55 +04:00
metronomefb_fix . line_length = fw ;
metronomefb_var . xres = fw ;
metronomefb_var . yres = fh ;
metronomefb_var . xres_virtual = fw ;
metronomefb_var . yres_virtual = fh ;
2008-03-20 03:01:10 +03:00
info - > var = metronomefb_var ;
info - > fix = metronomefb_fix ;
info - > fix . smem_len = videomemorysize ;
par = info - > par ;
par - > info = info ;
2008-04-28 13:15:38 +04:00
par - > board = board ;
2008-08-19 14:17:55 +04:00
par - > dt = epd_dt_index ;
2008-03-20 03:01:10 +03:00
init_waitqueue_head ( & par - > waitq ) ;
/* this table caches per page csum values. */
par - > csum_table = vmalloc ( videomemorysize / PAGE_SIZE ) ;
if ( ! par - > csum_table )
2008-08-19 14:17:55 +04:00
goto err_vfree ;
/* the physical framebuffer that we use is setup by
* the platform device driver . It will provide us
* with cmd , wfm and image memory in a contiguous area . */
retval = board - > setup_fb ( par ) ;
if ( retval ) {
dev_err ( & dev - > dev , " Failed to setup fb \n " ) ;
2008-03-20 03:01:10 +03:00
goto err_csum_table ;
2008-08-19 14:17:55 +04:00
}
2008-03-20 03:01:10 +03:00
2008-08-19 14:17:55 +04:00
/* after this point we should have a framebuffer */
if ( ( ! par - > metromem_wfm ) | | ( ! par - > metromem_img ) | |
( ! par - > metromem_dma ) ) {
dev_err ( & dev - > dev , " fb access failure \n " ) ;
retval = - EINVAL ;
goto err_csum_table ;
2008-03-20 03:01:10 +03:00
}
info - > fix . smem_start = par - > metromem_dma ;
2008-04-28 13:15:38 +04:00
/* load the waveform in. assume mode 3, temp 31 for now
a ) request the waveform file from userspace
2008-03-20 03:01:10 +03:00
b ) process waveform and decode into metromem */
2008-04-28 13:15:38 +04:00
retval = request_firmware ( & fw_entry , " metronome.wbf " , & dev - > dev ) ;
2008-03-20 03:01:10 +03:00
if ( retval < 0 ) {
2008-08-05 16:52:14 +04:00
dev_err ( & dev - > dev , " Failed to get waveform \n " ) ;
2008-08-19 14:17:55 +04:00
goto err_csum_table ;
2008-03-20 03:01:10 +03:00
}
2008-08-05 16:52:14 +04:00
retval = load_waveform ( ( u8 * ) fw_entry - > data , fw_entry - > size , 3 , 31 ,
par ) ;
2008-04-28 13:15:39 +04:00
release_firmware ( fw_entry ) ;
2008-03-20 03:01:10 +03:00
if ( retval < 0 ) {
2008-08-05 16:52:14 +04:00
dev_err ( & dev - > dev , " Failed processing waveform \n " ) ;
2008-08-19 14:17:55 +04:00
goto err_csum_table ;
2008-03-20 03:01:10 +03:00
}
2008-04-28 13:15:38 +04:00
if ( board - > setup_irq ( info ) )
2008-08-19 14:17:55 +04:00
goto err_csum_table ;
2008-03-20 03:01:10 +03:00
retval = metronome_init_regs ( par ) ;
if ( retval < 0 )
goto err_free_irq ;
2009-12-03 18:31:58 +03:00
info - > flags = FBINFO_FLAG_DEFAULT | FBINFO_VIRTFB ;
2008-03-20 03:01:10 +03:00
info - > fbdefio = & metronomefb_defio ;
fb_deferred_io_init ( info ) ;
retval = fb_alloc_cmap ( & info - > cmap , 8 , 0 ) ;
if ( retval < 0 ) {
2008-08-05 16:52:14 +04:00
dev_err ( & dev - > dev , " Failed to allocate colormap \n " ) ;
2008-08-19 14:17:55 +04:00
goto err_free_irq ;
2008-03-20 03:01:10 +03:00
}
/* set cmap */
for ( i = 0 ; i < 8 ; i + + )
info - > cmap . red [ i ] = ( ( ( 2 * i ) + 1 ) * ( 0xFFFF ) ) / 16 ;
memcpy ( info - > cmap . green , info - > cmap . red , sizeof ( u16 ) * 8 ) ;
memcpy ( info - > cmap . blue , info - > cmap . red , sizeof ( u16 ) * 8 ) ;
retval = register_framebuffer ( info ) ;
if ( retval < 0 )
goto err_cmap ;
platform_set_drvdata ( dev , info ) ;
2008-08-05 16:52:14 +04:00
dev_dbg ( & dev - > dev ,
2008-03-20 03:01:10 +03:00
" fb%d: Metronome frame buffer device, using %dK of video "
" memory \n " , info - > node , videomemorysize > > 10 ) ;
return 0 ;
err_cmap :
fb_dealloc_cmap ( & info - > cmap ) ;
err_free_irq :
2008-08-19 14:17:55 +04:00
board - > cleanup ( par ) ;
2008-03-20 03:01:10 +03:00
err_csum_table :
vfree ( par - > csum_table ) ;
err_vfree :
vfree ( videomemory ) ;
2008-08-19 14:17:55 +04:00
err_fb_rel :
framebuffer_release ( info ) ;
err :
2008-04-28 13:15:38 +04:00
module_put ( board - > owner ) ;
2008-03-20 03:01:10 +03:00
return retval ;
}
2012-12-22 01:07:39 +04:00
static int metronomefb_remove ( struct platform_device * dev )
2008-03-20 03:01:10 +03:00
{
struct fb_info * info = platform_get_drvdata ( dev ) ;
if ( info ) {
struct metronomefb_par * par = info - > par ;
2008-08-19 14:17:55 +04:00
unregister_framebuffer ( info ) ;
2008-03-20 03:01:10 +03:00
fb_deferred_io_cleanup ( info ) ;
fb_dealloc_cmap ( & info - > cmap ) ;
2008-08-19 14:17:55 +04:00
par - > board - > cleanup ( par ) ;
2008-03-20 03:01:10 +03:00
vfree ( par - > csum_table ) ;
vfree ( ( void __force * ) info - > screen_base ) ;
2008-04-28 13:15:38 +04:00
module_put ( par - > board - > owner ) ;
2008-08-19 14:17:55 +04:00
dev_dbg ( & dev - > dev , " calling release \n " ) ;
2008-03-20 03:01:10 +03:00
framebuffer_release ( info ) ;
}
return 0 ;
}
static struct platform_driver metronomefb_driver = {
. probe = metronomefb_probe ,
2012-12-22 01:07:39 +04:00
. remove = metronomefb_remove ,
2008-03-20 03:01:10 +03:00
. driver = {
2008-04-28 13:15:38 +04:00
. owner = THIS_MODULE ,
2008-03-20 03:01:10 +03:00
. name = " metronomefb " ,
} ,
} ;
static int __init metronomefb_init ( void )
{
2008-04-28 13:15:38 +04:00
return platform_driver_register ( & metronomefb_driver ) ;
2008-03-20 03:01:10 +03:00
}
static void __exit metronomefb_exit ( void )
{
platform_driver_unregister ( & metronomefb_driver ) ;
}
2008-08-19 14:17:55 +04:00
module_param ( user_wfm_size , uint , 0 ) ;
MODULE_PARM_DESC ( user_wfm_size , " Set custom waveform size " ) ;
2008-03-20 03:01:10 +03:00
module_init ( metronomefb_init ) ;
module_exit ( metronomefb_exit ) ;
MODULE_DESCRIPTION ( " fbdev driver for Metronome controller " ) ;
MODULE_AUTHOR ( " Jaya Kumar " ) ;
MODULE_LICENSE ( " GPL " ) ;