Subversion Repositories freemyipod

Rev

Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
273 theseven 1
//
2
//
3
//    Copyright 2010 TheSeven
4
//
5
//
6
//    This file is part of emBIOS.
7
//
8
//    emBIOS is free software: you can redistribute it and/or
9
//    modify it under the terms of the GNU General Public License as
10
//    published by the Free Software Foundation, either version 2 of the
11
//    License, or (at your option) any later version.
12
//
13
//    emBIOS is distributed in the hope that it will be useful,
14
//    but WITHOUT ANY WARRANTY; without even the implied warranty of
15
//    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
16
//    See the GNU General Public License for more details.
17
//
18
//    You should have received a copy of the GNU General Public License along
19
//    with emBIOS.  If not, see <http://www.gnu.org/licenses/>.
20
//
21
//
22
 
23
 
24
#include "global.h"
25
#include "thread.h"
26
#include "s5l8702.h"
27
#include "util.h"
28
 
29
 
278 theseven 30
static struct dma_lli lcd_lli[(LCD_WIDTH * LCD_HEIGHT - 1) / 0xfff]
31
    IDATA_ATTR __attribute__((aligned(16)));
273 theseven 32
 
33
static uint16_t lcd_color IDATA_ATTR;
34
 
35
 
36
void lcd_init()
37
{
38
}
39
 
40
int lcd_get_width()
41
{
42
    return LCD_WIDTH;
43
}
44
 
45
int lcd_get_height()
46
{
47
    return LCD_HEIGHT;
48
}
49
 
50
int lcd_get_bytes_per_pixel()
51
{
52
    return LCD_BYTESPERPIXEL;
53
}
54
 
55
static void lcd_send_cmd(uint16_t cmd)
56
{
57
    while (LCDSTATUS & 0x10);
58
    LCDWCMD = cmd;
59
}
60
 
61
static void lcd_send_data(uint16_t data)
62
{
63
    while (LCDSTATUS & 0x10);
419 theseven 64
    LCDWDATA = (data & 0xff) | ((data & 0x7f00) << 1);
273 theseven 65
}
66
 
67
static uint32_t lcd_detect()
68
{
69
    return (PDAT6 & 0x30) >> 4;
70
}
71
 
72
bool displaylcd_busy()
73
{
388 theseven 74
    return DMAC0C4CONFIG & 1;
273 theseven 75
}
76
 
77
bool displaylcd_safe()
78
{
388 theseven 79
    return !(DMAC0C4CONFIG & 1);
273 theseven 80
}
81
 
82
void displaylcd_sync()
83
{
84
    while (displaylcd_busy()) sleep(100);
85
}
86
 
87
void displaylcd(unsigned int startx, unsigned int endx,
88
                unsigned int starty, unsigned int endy, void* data, int color)
89
{
90
    displaylcd_sync();
378 theseven 91
    if (lcd_detect() & 2)
273 theseven 92
    {
93
        lcd_send_cmd(0x210);
94
        lcd_send_data(startx);
95
        lcd_send_cmd(0x211);
96
        lcd_send_data(endx);
97
        lcd_send_cmd(0x212);
98
        lcd_send_data(starty);
99
        lcd_send_cmd(0x213);
100
        lcd_send_data(endy);
101
        lcd_send_cmd(0x200);
102
        lcd_send_data(startx);
103
        lcd_send_cmd(0x201);
104
        lcd_send_data(starty);
105
        lcd_send_cmd(0x202);
106
    }
107
    else
108
    {
109
        lcd_send_cmd(0x2a);
110
        lcd_send_data(startx >> 8);
111
        lcd_send_data(startx & 0xff);
112
        lcd_send_data(endx >> 8);
113
        lcd_send_data(endx & 0xff);
114
        lcd_send_cmd(0x2b);
115
        lcd_send_data(starty >> 8);
116
        lcd_send_data(starty & 0xff);
117
        lcd_send_data(endy >> 8);
118
        lcd_send_data(endy & 0xff);
119
        lcd_send_cmd(0x2c);
120
    }
121
    int pixels = (endx - startx + 1) * (endy - starty + 1);
278 theseven 122
    if (pixels <= 0) return;
273 theseven 123
    int i;
124
    bool solid = (int)data == -1;
125
    if (solid) lcd_color = color;
278 theseven 126
    for (i = -1; i < (int)ARRAYLEN(lcd_lli) && pixels > 0; i++, pixels -= 0xfff)
273 theseven 127
    {
278 theseven 128
        bool last = i + 1 >= ARRAYLEN(lcd_lli) || pixels <= 0xfff;
388 theseven 129
        struct dma_lli* lli = i < 0 ? (struct dma_lli*)((int)&DMAC0C4LLI) : &lcd_lli[i];
278 theseven 130
        lli->srcaddr = solid ? &lcd_color : data;
131
        lli->dstaddr = (void*)((int)&LCDWDATA);
132
        lli->nextlli = last ? NULL : &lcd_lli[i + 1];
133
        lli->control = 0x70240000 | (last ? pixels : 0xfff)
134
                     | (last ? 0x80000000 : 0) | (solid ? 0 : 0x4000000);
273 theseven 135
        data = (void*)(((uint32_t)data) + 0x1ffe);
136
    }
137
    clean_dcache();
388 theseven 138
    DMAC0C4CONFIG = 0x88c1;
273 theseven 139
}
140
 
309 theseven 141
void lcd_shutdown()
142
{
143
    displaylcd_sync();
319 theseven 144
    uint32_t type = lcd_detect();
419 theseven 145
    if (type & 2)
319 theseven 146
    {
147
        lcd_send_cmd(0x7);
148
        lcd_send_data(0x172);
149
        lcd_send_cmd(0x30);
150
        lcd_send_data(0x3ff);
151
        sleep(90000);
152
        lcd_send_cmd(0x7);
153
        lcd_send_data(0x120);
154
        lcd_send_cmd(0x30);
155
        lcd_send_data(0x0);
156
        lcd_send_cmd(0x100);
157
        lcd_send_data(0x780);
158
        lcd_send_cmd(0x7);
159
        lcd_send_data(0x0);
160
        lcd_send_cmd(0x101);
161
        lcd_send_data(0x260);
162
        lcd_send_cmd(0x102);
163
        lcd_send_data(0xa9);
164
        sleep(30000);
165
        lcd_send_cmd(0x100);
166
        lcd_send_data(0x700);
167
        lcd_send_cmd(0x100);
168
        lcd_send_data(0x704);
169
    }
170
    else if (type == 1)
171
    {
172
        lcd_send_cmd(0x28);
173
        lcd_send_cmd(0x10);
174
        sleep(100000);
175
    }
176
    else
177
    {
178
        lcd_send_cmd(0x28);
179
        sleep(50000);
180
        lcd_send_cmd(0x10);
181
        sleep(50000);
182
    }
309 theseven 183
}
184
 
388 theseven 185
void INT_DMAC0C4()
273 theseven 186
{
388 theseven 187
    DMAC0INTTCCLR = 0x10;
273 theseven 188
    lcdconsole_callback();
189
}
190
 
191
int lcd_translate_color(uint8_t alpha, uint8_t red, uint8_t green, uint8_t blue)
192
    ICODE_ATTR __attribute__((naked, noinline));
193
int lcd_translate_color(uint8_t alpha, uint8_t red, uint8_t green, uint8_t blue)
194
{
195
    asm volatile(
196
        "cmp r0, #0xff             \n\t"
197
        "moveq r0, #-1             \n\t"
198
        "moveq pc, lr              \n\t"
199
        "cmp r0, #0                \n\t"
200
        "movne r0, #0xff000000     \n\t"
201
        "orrne r0, r0, #0xff0000   \n\t"
202
        "mov r2, r2,lsr#2          \n\t"
203
        "orr r0, r0, r3,lsr#3      \n\t"
204
        "mov r1, r1,lsr#3          \n\t"
205
        "orr r0, r0, r2,lsl#5      \n\t"
206
        "orr r0, r0, r1,lsl#11     \n\t"
207
        "mov pc, lr                \n\t"
208
    );
209
}