Subversion Repositories freemyipod

Rev

Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
268 theseven 1
//
2
//
3
//    Copyright 2010 TheSeven
4
//
5
//
427 farthen 6
//    This file is part of emCORE.
268 theseven 7
//
427 farthen 8
//    emCORE is free software: you can redistribute it and/or
268 theseven 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
//
427 farthen 13
//    emCORE is distributed in the hope that it will be useful,
268 theseven 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
427 farthen 19
//    with emCORE.  If not, see <http://www.gnu.org/licenses/>.
268 theseven 20
//
21
//
22
 
23
 
24
#include "global.h"
25
#include "thread.h"
26
#include "s5l8720.h"
27
#include "util.h"
560 theseven 28
#include "clockgates-target.h"
268 theseven 29
 
30
 
278 theseven 31
static struct dma_lli lcd_lli[(LCD_WIDTH * LCD_HEIGHT - 1) / 0xfff]
32
    IDATA_ATTR __attribute__((aligned(16)));
268 theseven 33
 
34
static uint16_t lcd_color IDATA_ATTR;
35
 
560 theseven 36
static struct mutex lcd_mutex IDATA_ATTR;
37
static struct wakeup lcd_wakeup IDATA_ATTR;
268 theseven 38
 
560 theseven 39
static bool lcd_dma_busy IDATA_ATTR;
40
static bool lcd_in_irq IDATA_ATTR;
492 theseven 41
 
560 theseven 42
 
268 theseven 43
void lcd_init()
44
{
560 theseven 45
    mutex_init(&lcd_mutex);
46
    wakeup_init(&lcd_wakeup);
47
    lcd_in_irq = false;
48
    lcd_dma_busy = true;
49
    clockgate_dma(0, 4, true);
50
    if (!(DMAC0C4CONFIG & 1))
51
    {
52
        lcd_dma_busy = false;
53
        clockgate_dma(0, 4, false);
54
    }
268 theseven 55
}
56
 
57
int lcd_get_width()
58
{
59
    return LCD_WIDTH;
60
}
61
 
62
int lcd_get_height()
63
{
64
    return LCD_HEIGHT;
65
}
66
 
67
int lcd_get_bytes_per_pixel()
68
{
69
    return LCD_BYTESPERPIXEL;
70
}
71
 
489 theseven 72
int lcd_get_format()
73
{
74
    return LCD_FORMAT;
75
}
76
 
492 theseven 77
static void lcd_send_cmd(uint16_t cmd) ICODE_ATTR __attribute__((noinline));
268 theseven 78
static void lcd_send_cmd(uint16_t cmd)
79
{
80
    while (LCDSTATUS & 0x10);
81
    LCDWCMD = cmd;
82
}
83
 
492 theseven 84
static void lcd_send_data(uint16_t data) ICODE_ATTR __attribute__((noinline));
268 theseven 85
static void lcd_send_data(uint16_t data)
86
{
87
    while (LCDSTATUS & 0x10);
492 theseven 88
    LCDWDATA = (data & 0xff) | ((data & 0x7f00) << 1);
268 theseven 89
}
90
 
91
void lcd_shutdown()
92
{
93
}
94
 
560 theseven 95
bool displaylcd_busy() ICODE_ATTR;
268 theseven 96
bool displaylcd_busy()
97
{
560 theseven 98
    return lcd_dma_busy;
268 theseven 99
}
100
 
101
bool displaylcd_safe()
102
{
560 theseven 103
    lcd_in_irq = true;
104
    if (!lcd_dma_busy) return true;
492 theseven 105
    return !(DMAC0C4CONFIG & 1);
268 theseven 106
}
107
 
492 theseven 108
void displaylcd_sync() ICODE_ATTR;
268 theseven 109
void displaylcd_sync()
110
{
492 theseven 111
    mutex_lock(&lcd_mutex, TIMEOUT_BLOCK);
560 theseven 112
    while (displaylcd_busy()) wakeup_wait(&lcd_wakeup, TIMEOUT_BLOCK);
492 theseven 113
    mutex_unlock(&lcd_mutex);
268 theseven 114
}
115
 
492 theseven 116
void displaylcd_setup(unsigned int startx, unsigned int endx,
117
                      unsigned int starty, unsigned int endy) ICODE_ATTR;
118
void displaylcd_setup(unsigned int startx, unsigned int endx,
119
                      unsigned int starty, unsigned int endy)
268 theseven 120
{
560 theseven 121
    if (!lcd_in_irq)
122
    {
123
        mutex_lock(&lcd_mutex, TIMEOUT_BLOCK);
124
        displaylcd_sync();
125
    }
268 theseven 126
    lcd_send_cmd(0x2a);
492 theseven 127
    lcd_send_data(startx);
128
    lcd_send_data(endx);
268 theseven 129
    lcd_send_cmd(0x2b);
492 theseven 130
    lcd_send_data(starty);
131
    lcd_send_data(endy);
268 theseven 132
    lcd_send_cmd(0x2c);
492 theseven 133
}
134
 
135
static void displaylcd_dma(void* data, int pixels, bool solid) ICODE_ATTR;
136
static void displaylcd_dma(void* data, int pixels, bool solid)
137
{
268 theseven 138
    int i;
560 theseven 139
    lcd_dma_busy = true;
140
    clockgate_dma(0, 4, true);
278 theseven 141
    for (i = -1; i < (int)ARRAYLEN(lcd_lli) && pixels > 0; i++, pixels -= 0xfff)
268 theseven 142
    {
278 theseven 143
        bool last = i + 1 >= ARRAYLEN(lcd_lli) || pixels <= 0xfff;
492 theseven 144
        struct dma_lli* lli = i < 0 ? (struct dma_lli*)((int)&DMAC0C4LLI) : &lcd_lli[i];
145
        lli->srcaddr = data;
278 theseven 146
        lli->dstaddr = (void*)((int)&LCDWDATA);
147
        lli->nextlli = last ? NULL : &lcd_lli[i + 1];
148
        lli->control = 0x70240000 | (last ? pixels : 0xfff)
149
                     | (last ? 0x80000000 : 0) | (solid ? 0 : 0x4000000);
560 theseven 150
        if (!solid) data = (void*)(((uint32_t)data) + 0x1ffe);
268 theseven 151
    }
152
    clean_dcache();
492 theseven 153
    DMAC0C4CONFIG = 0x88c1;
268 theseven 154
}
155
 
492 theseven 156
void displaylcd_native(unsigned int startx, unsigned int endx,
157
                       unsigned int starty, unsigned int endy, void* data)
268 theseven 158
{
492 theseven 159
    int pixels = (endx - startx + 1) * (endy - starty + 1);
160
    if (pixels <= 0) return;
161
    displaylcd_setup(startx, endx, starty, endy);
162
    displaylcd_dma(data, pixels, false);
560 theseven 163
    if (!lcd_in_irq) mutex_unlock(&lcd_mutex);
492 theseven 164
}
165
 
166
void filllcd_native(unsigned int startx, unsigned int endx,
167
                    unsigned int starty, unsigned int endy, int color)
168
{
169
    int pixels = (endx - startx + 1) * (endy - starty + 1);
170
    if (pixels <= 0) return;
171
    displaylcd_setup(startx, endx, starty, endy);
172
    lcd_color = color;
173
    displaylcd_dma(&lcd_color, pixels, true);
560 theseven 174
    if (!lcd_in_irq) mutex_unlock(&lcd_mutex);
492 theseven 175
}
176
 
177
void displaylcd_dither(unsigned int x, unsigned int y, unsigned int width,
178
                       unsigned int height, void* data, unsigned int datax,
560 theseven 179
                       unsigned int datay, unsigned int stride, bool solid)
180
     ICODE_ATTR __attribute__((naked,noinline));
492 theseven 181
void displaylcd_dither(unsigned int x, unsigned int y, unsigned int width,
182
                       unsigned int height, void* data, unsigned int datax,
183
                       unsigned int datay, unsigned int stride, bool solid)
184
{
560 theseven 185
//TODO: This is ARMv5E optimized assembly, should be converted to ARMv6
186
    __asm__ volatile("    muls r12, r2, r3             \n");
187
    __asm__ volatile("    bxeq lr                      \n");
188
    __asm__ volatile("    stmfd sp!, {r2-r11,lr}       \n");
189
    __asm__ volatile("    mov r12, r2                  \n");
190
    __asm__ volatile("    add r8, r2, r2,lsl#1         \n");
191
    __asm__ volatile("    add r3, r1, r3               \n");
192
    __asm__ volatile("    sub r3, r3, #1               \n");
193
    __asm__ volatile("    mov r2, r1                   \n");
194
    __asm__ volatile("    add r1, r0, r12              \n");
195
    __asm__ volatile("    sub r1, r1, #1               \n");
196
    __asm__ volatile("    bl displaylcd_setup          \n");
197
    __asm__ volatile("    mov r0, r8                   \n");
198
    __asm__ volatile("    bl malloc                    \n");
199
    __asm__ volatile("    cmp r0, #0                   \n");
200
    __asm__ volatile("    beq displaylcd_dither_unlock \n");
201
    __asm__ volatile("    mov r2, r8                   \n");
202
    __asm__ volatile("    mov r1, #0                   \n");
203
    __asm__ volatile("    mov r8, r0                   \n");
204
    __asm__ volatile("    bl memset                    \n");
205
    __asm__ volatile("    ldr r0, [sp,#0x30]           \n");
206
    __asm__ volatile("    ldr r1, [sp,#0x34]           \n");
207
    __asm__ volatile("    ldr r11, [sp,#0x38]          \n");
208
    __asm__ volatile("    ldr r3, [sp,#0x2c]           \n");
209
    __asm__ volatile("    mla r0, r1, r11, r0          \n");
210
    __asm__ volatile("    ldr r12, [sp,#0x04]          \n");
211
    __asm__ volatile("    ldr r2, [sp,#0x3c]           \n");
212
    __asm__ volatile("    add r3, r3, r0,lsl#1         \n");
213
    __asm__ volatile("    cmp r2, #0                   \n");
214
    __asm__ volatile("    ldreq r1, [sp]               \n");
215
    __asm__ volatile("    add r3, r3, r0               \n");
216
    __asm__ volatile("    subeq r11, r11, r1           \n");
217
    __asm__ volatile("    add r11, r11, r11,lsl#1      \n");
218
    __asm__ volatile("    movne r10, #3                \n");
219
    __asm__ volatile("    moveq r10, #0                \n");
220
    __asm__ volatile("    ldr r9, =0x38300040          \n");
221
    __asm__ volatile("displaylcd_dither_y:             \n");
222
    __asm__ volatile("    ldr lr, [sp]                 \n");
223
    __asm__ volatile("    mov r4, #0                   \n");
224
    __asm__ volatile("    mov r5, #0                   \n");
225
    __asm__ volatile("    mov r6, #0                   \n");
226
    __asm__ volatile("    mov r7, r8                   \n");
227
    __asm__ volatile("displaylcd_dither_x:             \n");
228
    __asm__ volatile("    mov r2, #0                   \n");
229
    __asm__ volatile("    ldrb r1, [r3], #1            \n");
230
    __asm__ volatile("    ldrsb r0, [r7]               \n");
231
    __asm__ volatile("    add r1, r1, r4               \n");
232
    __asm__ volatile("    add r1, r1, r0               \n");
233
    __asm__ volatile("    cmp r1, #0                   \n");
234
    __asm__ volatile("    movlt r1, #0                 \n");
235
    __asm__ volatile("    cmp r1, #0xff                \n");
236
    __asm__ volatile("    movgt r1, #0xff              \n");
237
    __asm__ volatile("    mov r0, r1,lsr#3             \n");
238
    __asm__ volatile("    orr r2, r0,lsl#11            \n");
239
    __asm__ volatile("    sub r1, r1, r0,lsl#3         \n");
240
    __asm__ volatile("    sub r1, r1, r0,lsr#2         \n");
241
    __asm__ volatile("    mov r4, r4,lsr#1             \n");
242
    __asm__ volatile("    add r4, r4, r1,lsr#2         \n");
243
    __asm__ volatile("    strb r4, [r7], #1            \n");
244
    __asm__ volatile("    mov r4, r1,asr#1             \n");
245
    __asm__ volatile("    ldrb r1, [r3], #1            \n");
246
    __asm__ volatile("    ldrsb r0, [r7]               \n");
247
    __asm__ volatile("    add r1, r1, r5               \n");
248
    __asm__ volatile("    add r1, r1, r0               \n");
249
    __asm__ volatile("    cmp r1, #0                   \n");
250
    __asm__ volatile("    movlt r1, #0                 \n");
251
    __asm__ volatile("    cmp r1, #0xff                \n");
252
    __asm__ volatile("    movgt r1, #0xff              \n");
253
    __asm__ volatile("    mov r0, r1,lsr#2             \n");
254
    __asm__ volatile("    orr r2, r0,lsl#5             \n");
255
    __asm__ volatile("    sub r1, r1, r0,lsl#2         \n");
256
    __asm__ volatile("    sub r1, r1, r0,lsr#4         \n");
257
    __asm__ volatile("    mov r5, r5,lsr#1             \n");
258
    __asm__ volatile("    add r5, r5, r1,lsr#2         \n");
259
    __asm__ volatile("    strb r5, [r7], #1            \n");
260
    __asm__ volatile("    mov r5, r1,asr#1             \n");
261
    __asm__ volatile("    ldrb r1, [r3], #1            \n");
262
    __asm__ volatile("    ldrsb r0, [r7]               \n");
263
    __asm__ volatile("    add r1, r1, r6               \n");
264
    __asm__ volatile("    add r1, r1, r0               \n");
265
    __asm__ volatile("    cmp r1, #0                   \n");
266
    __asm__ volatile("    movlt r1, #0                 \n");
267
    __asm__ volatile("    cmp r1, #0xff                \n");
268
    __asm__ volatile("    movgt r1, #0xff              \n");
269
    __asm__ volatile("    mov r0, r1,lsr#3             \n");
270
    __asm__ volatile("    orr r2, r0                   \n");
271
    __asm__ volatile("    sub r1, r1, r0,lsl#3         \n");
272
    __asm__ volatile("    sub r1, r1, r0,lsr#2         \n");
273
    __asm__ volatile("    mov r6, r6,lsr#1             \n");
274
    __asm__ volatile("    add r6, r6, r1,lsr#2         \n");
275
    __asm__ volatile("    strb r6, [r7], #1            \n");
276
    __asm__ volatile("displaylcd_dither_waitlcd:       \n");
277
    __asm__ volatile("    ldr r0, [r9,#-0x24]          \n");
278
    __asm__ volatile("    mov r6, r1,asr#1             \n");
279
    __asm__ volatile("    tst r0, #0x10                \n");
280
    __asm__ volatile("    bne displaylcd_dither_waitlcd\n");
281
    __asm__ volatile("    str r2, [r9]                 \n");
282
    __asm__ volatile("    sub r3, r3, r10              \n");
283
    __asm__ volatile("    subs lr, lr, #1              \n");
284
    __asm__ volatile("    bne displaylcd_dither_x      \n");
285
    __asm__ volatile("    add r3, r3, r11              \n");
286
    __asm__ volatile("    subs r12, r12, #1            \n");
287
    __asm__ volatile("    bne displaylcd_dither_y      \n");
288
    __asm__ volatile("displaylcd_dither_free:          \n");
289
    __asm__ volatile("    mov r0, r8                   \n");
290
    __asm__ volatile("    bl free                      \n");
291
    __asm__ volatile("displaylcd_dither_unlock:        \n");
292
    __asm__ volatile("    ldr r0, =lcd_mutex           \n");
293
    __asm__ volatile("    bl mutex_unlock              \n");
294
    __asm__ volatile("    ldmfd sp!, {r2-r11,pc}       \n");
492 theseven 295
}
296
 
297
void displaylcd(unsigned int x, unsigned int y, unsigned int width, unsigned int height,
298
                void* data, unsigned int datax, unsigned int datay, unsigned int stride)
299
{
300
    displaylcd_dither(x, y, width, height, data, datax, datay, stride, false);
301
}
302
 
303
void filllcd(unsigned int x, unsigned int y, unsigned int width, unsigned int height, int color)
304
{
305
    if (width * height <= 0) return;
306
    mutex_lock(&lcd_mutex, TIMEOUT_BLOCK);
560 theseven 307
    displaylcd_sync();
492 theseven 308
    lcd_color = color;
309
    displaylcd_dither(x, y, width, height, &lcd_color, 0, 0, 0, true);
310
    mutex_unlock(&lcd_mutex);
311
}
312
 
313
void INT_DMAC0C4()
314
{
315
    DMAC0INTTCCLR = 0x10;
560 theseven 316
    lcd_in_irq = true;
317
    lcd_dma_busy = false;
318
    clockgate_dma(0, 4, false);
268 theseven 319
    lcdconsole_callback();
560 theseven 320
    wakeup_signal(&lcd_wakeup);
321
    lcd_in_irq = false;
268 theseven 322
}
323
 
324
int lcd_translate_color(uint8_t alpha, uint8_t red, uint8_t green, uint8_t blue)
325
    ICODE_ATTR __attribute__((naked, noinline));
326
int lcd_translate_color(uint8_t alpha, uint8_t red, uint8_t green, uint8_t blue)
327
{
328
    asm volatile(
329
        "cmp r0, #0xff             \n\t"
330
        "moveq r0, #-1             \n\t"
331
        "moveq pc, lr              \n\t"
332
        "cmp r0, #0                \n\t"
333
        "movne r0, #0xff000000     \n\t"
334
        "orrne r0, r0, #0xff0000   \n\t"
335
        "mov r2, r2,lsr#2          \n\t"
336
        "orr r0, r0, r3,lsr#3      \n\t"
337
        "mov r1, r1,lsr#3          \n\t"
338
        "orr r0, r0, r2,lsl#5      \n\t"
339
        "orr r0, r0, r1,lsl#11     \n\t"
340
        "mov pc, lr                \n\t"
341
    );
342
}