/***************************************************************************
 *             __________               __   ___.
 *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 *                     \/            \/     \/    \/            \/
 * $Id$
 *
 * Rockbox driver for Sansa View LCDs
 *
 * Copyright (c) 2009 Robert Keevil
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 * KIND, either express or implied.
 *
 ****************************************************************************/
#include <string.h>
#include "cpu.h"
#include "system.h"
#include "backlight-target.h"
#include "lcd.h"

#include "bitmaps/rockboxlogo.h"

/* Power and display status */
static bool power_on   = false; /* Is the power turned on?   */
static bool display_on SHAREDBSS_ATTR = false; /* Is the display turned on? */
static unsigned lcd_yuv_options SHAREDBSS_ATTR = 0;

#define LCD_DATA_OUT_GPIO GPIOH_OUTPUT_VAL
#define LCD_DATA_OUT_PIN 4

#define LCD_CLOCK_GPIO GPIOH_OUTPUT_VAL
#define LCD_CLOCK_PIN 6

#define LCD_CS_GPIO GPIOH_OUTPUT_VAL
#define LCD_CS_PIN 7

#ifdef BOOTLOADER
static void lcd_init_gpio(void)
{
    // OF: 0x5CC8

    outl(inl(0x70000010) | 0xFC000000, 0x70000010);
    outl(inl(0x70000014) | 0xC300000, 0x70000014);

    GPIOE_ENABLE = 0;
    GPIOM_ENABLE &= ~0x3;
    GPIOJ_ENABLE &= ~0x1a;
    GPIOB_ENABLE &= ~0x8;
    GPIOH_OUTPUT_VAL |= 0x80;
    GPIOH_OUTPUT_EN |= 0x80;
    GPIOH_ENABLE |= 0x80;
    GPIOH_OUTPUT_VAL |= 0x40;
    GPIOH_OUTPUT_EN |= 0x40;
    GPIOH_ENABLE |= 0x40;
    GPIOH_OUTPUT_VAL |= 0x20;
    GPIOH_OUTPUT_EN |= 0x20;
    GPIOH_ENABLE |= 0x20;
    GPIOH_OUTPUT_VAL |= 0x10;
    GPIOH_OUTPUT_EN |= 0x10;
    GPIOH_ENABLE |= 0x10;
//    GPIOD_OUTOUT_VAL &= ~0x1;   //backlight on
//    GPIOD_ENABLE |= 0x1;
    GPIOB_OUTPUT_VAL |= 0x4;
    GPIOB_ENABLE |= 0x4;
    GPIOB_OUTPUT_EN |= 0x4;
    GPIOG_ENABLE |= 0x8;
    GPIOG_OUTPUT_EN &= ~0x8;

// more to add here...
}
#endif

static void lcd_send_msg(unsigned char count, unsigned int data)
{
    // OF: 0x645C
    int i;

    LCD_CLOCK_GPIO |= (1 << LCD_CLOCK_PIN);
    LCD_CS_GPIO &= ~(1 << LCD_CS_PIN);

    for (i = count - 1; i >= 0; i--)
    {
        if (data & (1 << count))
        {
            LCD_DATA_OUT_GPIO &= ~(1 << LCD_DATA_OUT_PIN);
        } else {
            LCD_DATA_OUT_GPIO |= (1 << LCD_DATA_OUT_PIN);
        }
        LCD_CLOCK_GPIO &= ~(1 << LCD_CLOCK_PIN);
        udelay(1);
        LCD_CLOCK_GPIO |= (1 << LCD_CLOCK_PIN);
        udelay(2);
    }

    LCD_CS_GPIO |= (1 << LCD_CS_PIN);
    LCD_CLOCK_GPIO |= (1 << LCD_CLOCK_PIN);
    udelay(1);
}

/*
static void lcd_write_reg(unsigned int reg, unsigned int data)
{
//  OF: 0x6278 - referenced from 0x62840

//  So far this function is always called with shift = 0x0 ?
//    If so can remove and simplify 

    unsigned int cmd;
    unsigned int shift = 0;

    cmd = shift << 0x2;
    cmd |= 0x70;
    cmd = cmd << 0x10;
    cmd |= reg;
    lcd_send_msg(0x18, cmd);

    cmd = shift << 0x2;
    cmd |= 0x72;
    cmd = cmd << 0x10;
    cmd |= data;
    lcd_send_msg(0x18, cmd);

//     lcd_send_msg(0x70, reg);
//     lcd_send_msg(0x72, data);
}
*/

static void lcd_write_cmd(unsigned int cmd)
{
    lcd_send_msg(0x18, (0x700000 | cmd));
}

static void lcd_write_info(unsigned int data)
{
    lcd_send_msg(0x18, (0x720000 | data));
}

static void lcd_write_reg(unsigned int cmd, unsigned int data)
{
    lcd_write_cmd(cmd);
    lcd_write_info(data);
}

/* Run the powerup sequence for the driver IC */
static void lcd_power_on(void)
{
    /* OF: 0x5DC0 *
    * r2: cmd    *
    * r3: data   */
    lcd_write_reg(0xE5, 0x8000);
    lcd_write_reg(0x0, 0x1);
    lcd_write_reg(0x1, 0x100);
    lcd_write_reg(0x2, 0x700);
    lcd_write_reg(0x3, 0x1230);
    lcd_write_reg(0x4, 0x0);
    lcd_write_reg(0x8, 0x408);
    lcd_write_reg(0x9, 0x0);
    lcd_write_reg(0xa, 0x0);
    lcd_write_reg(0xd, 0x0);
    lcd_write_reg(0xf, 0x2);
    lcd_write_reg(0x10, 0x0);
    lcd_write_reg(0x11, 0x0);
    lcd_write_reg(0x12, 0x0);
    lcd_write_reg(0x13, 0x0);
    sleep(HZ/5);
    lcd_write_reg(0x10, 0x17B0);
    lcd_write_reg(0x11, 0x7);
    sleep(HZ/20);
    lcd_write_reg(0x12, 0x13c);
    sleep(HZ/20);

    // OF: BNE 0x5fb2

    // two different models in use?!?
    if (1)
    {
        lcd_write_reg(0x13, 0x1800);
        lcd_write_reg(0x29, 0x13);
        sleep(HZ/10);
        lcd_write_reg(0x20, 0x0);
        lcd_write_reg(0x21, 0x0);

        lcd_write_reg(0x30, 0x2);
        lcd_write_reg(0x31, 0xF07); // 0x37 option in other controller
        lcd_write_reg(0x32, 0x403); // 0x31 option in other controller
        lcd_write_reg(0x35, 0x206);
        lcd_write_reg(0x36, 0x504);
        lcd_write_reg(0x37, 0x707);
        lcd_write_reg(0x38, 0x403);
    }
    else
    {
        // OF: last func continues, 0x5EFC
        lcd_write_reg(0x13, 0x1700);
        lcd_write_reg(0x29, 0x10);
        sleep(HZ/10);
        lcd_write_reg(0x20, 0x0);
        lcd_write_reg(0x21, 0x0);

        lcd_write_reg(0x30, 0x7);
        lcd_write_reg(0x31, 0x403);
        lcd_write_reg(0x32, 0x400);
        lcd_write_reg(0x35, 0x3);
        lcd_write_reg(0x36, 0xF07);
        lcd_write_reg(0x37, 0x403);
        lcd_write_reg(0x37, 0x106);
    }

    // OF: b 0x6066
    lcd_write_reg(0x39, 0x7);
    lcd_write_reg(0x3c, 0x700);
    lcd_write_reg(0x3d, 0x700);

    lcd_write_reg(0x50, 0x0);
    lcd_write_reg(0x51, 0xef);  // 239 - LCD_WIDTH
    lcd_write_reg(0x52, 0x0);
    lcd_write_reg(0x53, 0x13f); // 319 - LCD_HEIGHT

    // OF: b 0x6114
    lcd_write_reg(0x60, 0x2700);
    lcd_write_reg(0x61, 0x1);
    lcd_write_reg(0x6a, 0x0);

    lcd_write_reg(0x80, 0x0);
    lcd_write_reg(0x81, 0x0);
    lcd_write_reg(0x82, 0x0);
    lcd_write_reg(0x83, 0x0);
    lcd_write_reg(0x84, 0x0);
    lcd_write_reg(0x85, 0x0);

    // OF: 0x61A8
    lcd_write_reg(0x90, 0x10);
    lcd_write_reg(0x92, 0x0);
    lcd_write_reg(0x93, 0x3);
    lcd_write_reg(0x95, 0x110);
    lcd_write_reg(0x97, 0x0);
    lcd_write_reg(0x98, 0x0);

    lcd_write_reg(0xc, 0x110);
    lcd_write_reg(0x7, 0x173);
    sleep(HZ/10);
    lcd_write_cmd(0x22);

    power_on = true;
}

void unknown01(void)
{
    // OF: 0x62C4

    lcd_write_reg(0x10, 0x17B0);
    udelay(100);
    lcd_write_reg(0x7, 0x173);
}

void unknown02(void)
{
    // OF: 0x6308

    lcd_write_reg(0x7, 0x160);
    lcd_write_reg(0x10, 0x17B1);
}

void unknown03(void)
{
    // OF: 0x6410
    GPIOJ_ENABLE |= 0x2;
    GPIOJ_OUTPUT_EN |= 0x2;
    GPIOJ_OUTPUT_VAL &= ~0x02;
}

void unknown04(void)
{
    // OF: 0x623C

    GPIOB_OUTPUT_VAL |= 0x4;
    udelay(1000);
    GPIOB_OUTPUT_VAL &= ~0x4;
    sleep(HZ/10);
    GPIOB_OUTPUT_VAL |= 0x4;
    udelay(1000);
}

/* Run the display on sequence for the driver IC */
static void lcd_display_on(void)
{
    display_on = true;
}


#if defined(HAVE_LCD_ENABLE) || defined(HAVE_LCD_SLEEP)
bool lcd_active(void)
{
    return display_on;
}

/* Turn off visible display operations */
static void lcd_display_off(void)
{
    display_on = false;
}
#endif

void lcd_init_device(void)
{

#ifdef BOOTLOADER /* Bother at all to do this again? */
//#if 0
/* Init GPIO ports */
    lcd_init_gpio();
    lcd_power_on();
    lcd_display_on();
#else

    power_on = true;
    display_on = true;

    lcd_set_invert_display(false);
    lcd_set_flip(false);
#endif
}

#if defined(HAVE_LCD_ENABLE)
void lcd_enable(bool on)
{
    (void)on;
}
#endif

#if defined(HAVE_LCD_SLEEP)
void lcd_sleep(void)
{

}
#endif

void lcd_update(void)
{
    const fb_data *addr;

    addr = &lcd_framebuffer[LCD_HEIGHT][LCD_WIDTH];

    int i,j;
    for(i=0; i < LCD_HEIGHT; i++)
    {
        lcd_write_reg(0x20, i);
        for(j=0; j < LCD_WIDTH; j++)
        {
            lcd_write_reg(0x21, j);
            lcd_write_reg(0x22, *addr);
            addr++;
        }
    }
    lcd_write_reg(0x20, 0x0);
    lcd_write_reg(0x21, 0x0);
}

/* Update a fraction of the display. */
void lcd_update_rect(int x, int y, int width, int height)
{
    (void)x;
    (void)y;
    (void)width;
    (void)height;
    lcd_update();
}


/*** hardware configuration ***/

void lcd_set_contrast(int val)
{
    (void)val;
}

void lcd_set_invert_display(bool yesno)
{
    (void)yesno;
}

/* turn the display upside down (call lcd_update() afterwards) */
void lcd_set_flip(bool yesno)
{
    (void)yesno;
}

/* Blitting functions */

void lcd_yuv_set_options(unsigned options)
{
    lcd_yuv_options = options;
}

/* Line write helper function for lcd_yuv_blit. Write two lines of yuv420. */
extern void lcd_write_yuv420_lines(fb_data *dst,
                                   unsigned char const * const src[3],
                                   int width,
                                   int stride);
extern void lcd_write_yuv420_lines_odither(fb_data *dst,
                                           unsigned char const * const src[3],
                                           int width,
                                           int stride,
                                           int x_screen, /* To align dither pattern */
                                           int y_screen);

void lcd_blit_yuv(unsigned char * const src[3],
                  int src_x, int src_y, int stride,
                  int x, int y, int width, int height)
{
    (void)src;
    (void)src_x;
    (void)src_y;
    (void)stride;
    (void)x;
    (void)y;
    (void)width;
    (void)height;
}

