#include "pxt.h" #include "ev3const.h" #include #include #include #include #include #include #include #include namespace pxt { #define ROW_SIZE 23 #define FB_SIZE (60 * LCD_HEIGHT) static const uint8_t pixmap[] = {0x00, 0x03, 0x1C, 0x1F, 0xE0, 0xE3, 0xFC, 0xFF}; static void bitBufferToFrameBufferSwap(uint8_t *bitBuffer, uint8_t *fb) { int mask = 0x01; uint8_t *currLine = bitBuffer; for (int line = 0; line < LCD_HEIGHT; line++) { uint8_t *ptr = currLine; int n = 60; while (n--) { uint8_t v = 0; if (*ptr & mask) v |= 0xE0; ptr += LCD_HEIGHT / 8; if (*ptr & mask) v |= 0x1C; ptr += LCD_HEIGHT / 8; if (*ptr & mask) v |= 0x03; ptr += LCD_HEIGHT / 8; *fb++ = v; } mask <<= 1; if (mask == 0x100) { mask = 0x01; currLine++; } } } static void bitBufferToFrameBuffer(uint8_t *bitBuffer, uint8_t *fb) { uint32_t pixels; for (int line = 0; line < LCD_HEIGHT; line++) { int n = 7; while (n--) { pixels = *bitBuffer++ << 16; pixels |= *bitBuffer++ << 8; pixels |= *bitBuffer++ << 0; int m = 8; while (m--) { *fb++ = pixmap[(pixels >> 21) & 0x07]; pixels <<= 3; } } pixels = *bitBuffer++ << 8; pixels |= *bitBuffer++ << 0; int m = 4; while (m--) { *fb++ = pixmap[(pixels >> 13) & 0x07]; pixels <<= 3; } } } static uint8_t *mappedFrameBuffer; static Image_ lastImg; //% void updateScreen(Image_ img) { if (img && img != lastImg) { decrRC(lastImg); incrRC(img); lastImg = img; } if (lastImg && mappedFrameBuffer != MAP_FAILED) { if (lastImg->bpp() != 1 || lastImg->width() != LCD_WIDTH || lastImg->height() != LCD_HEIGHT) target_panic(906); bitBufferToFrameBufferSwap(lastImg->pix(), mappedFrameBuffer); } } //% void updateStats(String str) { } void screen_init() { DMESG("init screen"); if (mappedFrameBuffer) return; int fd = open("/dev/fb0", O_RDWR); DMESG("init screen %d", fd); mappedFrameBuffer = (uint8_t *)mmap(NULL, FB_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); DMESG("map %p", mappedFrameBuffer); if (mappedFrameBuffer == MAP_FAILED) { target_panic(903); } } static const uint8_t numbers[] = { 0x06, 0x09, 0x09, 0x09, 0x06, 0x04, 0x06, 0x04, 0x04, 0x0e, 0x07, 0x08, 0x06, 0x01, 0x0f, 0x0f, 0x08, 0x04, 0x09, 0x06, 0x0c, 0x0a, 0x09, 0x1f, 0x08, 0x1f, 0x01, 0x0f, 0x10, 0x0f, 0x08, 0x04, 0x0e, 0x11, 0x0e, 0x1f, 0x08, 0x04, 0x02, 0x01, 0x0e, 0x11, 0x0e, 0x11, 0x0e, 0x0e, 0x11, 0x0e, 0x04, 0x02, 0x1b, 0x1b, 0x00, 0x1f, 0x1b, }; static void drawNumber(uint8_t *dst, int idx) { const uint8_t *src = &numbers[idx * 5]; for (int i = 0; i < 5; i++) { uint8_t ch = *src++; for (int jj = 0; jj < 8; ++jj) { for (int j = 0; j < 5; j++) { if (ch & (1 << j)) *dst = 0xff; dst++; } dst += ROW_SIZE - 5; } } } extern "C" void drawPanic(int code) { int fd = open("/dev/lms_ui", O_RDWR); uint8_t cmd[] = {48 + 5, 0}; write(fd, cmd, 2); close(fd); uint8_t bitBuffer[ROW_SIZE * LCD_HEIGHT]; memset(bitBuffer, 0, sizeof(bitBuffer)); auto ptr = &bitBuffer[ROW_SIZE * 16 + 3 + 6]; drawNumber(ptr, 10); ptr = &bitBuffer[ROW_SIZE * 70 + 3]; drawNumber(ptr, (code / 100) % 10); ptr += 6; drawNumber(ptr, (code / 10) % 10); ptr += 6; drawNumber(ptr, (code / 1) % 10); ptr += 6; if (mappedFrameBuffer != MAP_FAILED) { bitBufferToFrameBuffer(bitBuffer, mappedFrameBuffer); } } } // namespace pxt