Skip to content
Snippets Groups Projects
sys_display.c 8.78 KiB
Newer Older
  • Learn to ignore specific revisions
  • #include "epicardium.h"
    
    
    Gerd's avatar
    Gerd committed
    #include "py/obj.h"
    #include "py/objint.h"
    
    #include "py/objstr.h"
    
    Gerd's avatar
    Gerd committed
    #include "py/runtime.h"
    
    #include <stdio.h>
    
    static uint16_t rgb888_to_rgb565(uint8_t *bytes)
    {
    	return ((bytes[0] & 0b11111000) << 8) | ((bytes[1] & 0b11111100) << 3) |
    	       (bytes[2] >> 3);
    }
    
    static uint16_t get_color(mp_obj_t color_in)
    {
    	if (mp_obj_get_int(mp_obj_len(color_in)) < 3) {
    		mp_raise_ValueError("color must have 3 elements");
    	}
    
    	uint8_t red = mp_obj_get_int(
    		mp_obj_subscr(color_in, mp_obj_new_int(0), MP_OBJ_SENTINEL)
    	);
    	uint8_t green = mp_obj_get_int(
    		mp_obj_subscr(color_in, mp_obj_new_int(1), MP_OBJ_SENTINEL)
    	);
    	uint8_t blue = mp_obj_get_int(
    		mp_obj_subscr(color_in, mp_obj_new_int(2), MP_OBJ_SENTINEL)
    	);
    	uint8_t colors[3] = { red, green, blue };
    	return rgb888_to_rgb565(colors);
    }
    
    /* print something on the display */
    static mp_obj_t mp_display_print(size_t n_args, const mp_obj_t *args)
    {
    	if (!mp_obj_is_str_or_bytes(args[0])) {
    		mp_raise_TypeError("input text must be a string");
    	}
    	GET_STR_DATA_LEN(args[0], print, print_len);
    
    	int32_t posx = mp_obj_get_int(args[1]);
    	int32_t posy = mp_obj_get_int(args[2]);
    	uint32_t fg  = get_color(args[3]);
    	uint32_t bg  = get_color(args[4]);
    	int res      = epic_disp_print(posx, posy, (const char *)print, fg, bg);
    
    Gerd's avatar
    Gerd committed
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_print_obj, 5, 5, mp_display_print
    );
    
    /* print something on the display */
    static mp_obj_t mp_display_print_adv(size_t n_args, const mp_obj_t *args)
    {
    	if (!mp_obj_is_str_or_bytes(args[0])) {
    		mp_raise_TypeError("input text must be a string");
    	}
    	GET_STR_DATA_LEN(args[0], print, print_len);
    
    	int32_t posx     = mp_obj_get_int(args[1]);
    	int32_t posy     = mp_obj_get_int(args[2]);
    
    	uint32_t fg      = get_color(args[3]);
    	uint32_t bg      = get_color(args[4]);
    	uint8_t fontName = mp_obj_get_int(args[5]);
    	int res          = epic_disp_print_adv(
                    fontName, posx, posy, (const char *)print, fg, bg
    	);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    	display_print_adv_obj, 6, 6, mp_display_print_adv
    );
    
    
    /* draw pixel on the display */
    static mp_obj_t mp_display_pixel(size_t n_args, const mp_obj_t *args)
    {
    
    	int16_t x    = mp_obj_get_int(args[0]);
    	int16_t y    = mp_obj_get_int(args[1]);
    
    	uint16_t col = get_color(args[2]);
    
    	int res = epic_disp_pixel(x, y, col);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_pixel_obj, 3, 3, mp_display_pixel
    );
    
    /* blit image to display */
    static mp_obj_t mp_display_blit(size_t n_args, const mp_obj_t *args)
    {
    	/* Required arguments: posx, posy (on display),
    	                       width, height (of image),
    						   buffer (rgb data of image) */
    
    	int pos_x   = mp_obj_get_int(args[0]);
    	int pos_y   = mp_obj_get_int(args[1]);
    	int width   = mp_obj_get_int(args[2]);
    	int height  = mp_obj_get_int(args[3]);
    	bool rgb565 = mp_obj_is_true(args[5]);
    
    	mp_buffer_info_t img;
    
    	int res = 0;
    
    	/* Load buffer and ensure it contains enough data */
    	if (!mp_get_buffer(args[4], &img, MP_BUFFER_READ)) {
    		mp_raise_TypeError("'img' does not support buffer protocol.");
    	}
    
    
    	int bpp = rgb565 ? 2 : 3;
    	if ((int)img.len < width * height * bpp) {
    
    		mp_raise_ValueError("'img' is too small.");
    	}
    
    
    	uint16_t *buf = NULL;
    
    	if (rgb565) {
    		buf = (uint16_t *)img.buf;
    	} else {
    		/* Will raise an exception if out of memory: */
    		buf = m_malloc(width * height * bpp);
    		for (int i = 0; i < width * height; i++) {
    			buf[i] = rgb888_to_rgb565(((uint8_t *)img.buf) + 3 * i);
    		}
    	}
    
    	if (n_args > 6) {
    
    		mp_buffer_info_t alpha;
    
    		/* Load alpha buffer and check size */
    
    		if (!mp_get_buffer(args[6], &alpha, MP_BUFFER_READ)) {
    
    			mp_raise_TypeError(
    				"'alpha' does not support buffer protocol."
    			);
    		}
    
    		if ((int)alpha.len < width * height) {
    
    			mp_raise_ValueError("'alpha' is too small.");
    		}
    
    		res = epic_disp_blit(
    
    			pos_x, pos_y, width, height, buf, (uint8_t *)alpha.buf
    
    		res = epic_disp_blit(pos_x, pos_y, width, height, buf, NULL);
    
    	if (!rgb565) {
    		/* Do not free rgb565 data. It is owned by the caller */
    
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_blit_obj, 5, 6, mp_display_blit
    
    /* set display backlight brightness */
    static mp_obj_t mp_display_backlight(size_t n_args, const mp_obj_t *args)
    {
    	uint16_t brightness = mp_obj_get_int(args[0]);
    
    	int res = epic_disp_backlight(brightness);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    	display_backlight_obj, 1, 1, mp_display_backlight
    );
    
    
    Gerd's avatar
    Gerd committed
    /* draw line on the display */
    static mp_obj_t mp_display_line(size_t n_args, const mp_obj_t *args)
    {
    
    	int16_t xs   = mp_obj_get_int(args[0]);
    	int16_t ys   = mp_obj_get_int(args[1]);
    	int16_t xe   = mp_obj_get_int(args[2]);
    	int16_t ye   = mp_obj_get_int(args[3]);
    
    Gerd's avatar
    Gerd committed
    	uint16_t col = get_color(args[4]);
    	uint16_t ls  = mp_obj_get_int(args[5]);
    	uint16_t ps  = mp_obj_get_int(args[6]);
    
    
    	if (ls > 1) {
    
    Gerd's avatar
    Gerd committed
    		mp_raise_ValueError("Line style has to be 0 or 1");
    	}
    
    	int res = epic_disp_line(xs, ys, xe, ye, col, ls, ps);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_line_obj, 7, 7, mp_display_line
    );
    
    Gerd's avatar
    Gerd committed
    
    /* draw rectangle on the display */
    static mp_obj_t mp_display_rect(size_t n_args, const mp_obj_t *args)
    {
    
    	int16_t xs   = mp_obj_get_int(args[0]);
    	int16_t ys   = mp_obj_get_int(args[1]);
    	int16_t xe   = mp_obj_get_int(args[2]);
    	int16_t ye   = mp_obj_get_int(args[3]);
    
    Gerd's avatar
    Gerd committed
    	uint16_t col = get_color(args[4]);
    	uint16_t fs  = mp_obj_get_int(args[5]);
    	uint16_t ps  = mp_obj_get_int(args[6]);
    
    
    	if (fs > 1) {
    
    Gerd's avatar
    Gerd committed
    		mp_raise_ValueError("Fill style has to be 0 or 1");
    	}
    
    	int res = epic_disp_rect(xs, ys, xe, ye, col, fs, ps);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_rect_obj, 7, 7, mp_display_rect
    );
    
    Gerd's avatar
    Gerd committed
    
    /* draw rectangle on the display */
    static mp_obj_t mp_display_circ(size_t n_args, const mp_obj_t *args)
    {
    
    	int16_t x    = mp_obj_get_int(args[0]);
    	int16_t y    = mp_obj_get_int(args[1]);
    
    Gerd's avatar
    Gerd committed
    	uint16_t rad = mp_obj_get_int(args[2]);
    	uint16_t col = get_color(args[3]);
    	uint16_t fs  = mp_obj_get_int(args[4]);
    	uint16_t ps  = mp_obj_get_int(args[5]);
    
    
    	if (fs > 1) {
    
    Gerd's avatar
    Gerd committed
    		mp_raise_ValueError("Fill style has to be 0 or 1");
    	}
    
    	int res = epic_disp_circ(x, y, rad, col, fs, ps);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(
    
    	display_circ_obj, 6, 6, mp_display_circ
    );
    
    Gerd's avatar
    Gerd committed
    
    /* clear the display */
    static mp_obj_t mp_display_clear(mp_obj_t col)
    {
    	uint16_t color = get_color(col);
    	int res        = epic_disp_clear(color);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_1(display_clear_obj, mp_display_clear);
    
    Gerd's avatar
    Gerd committed
    
    static mp_obj_t mp_display_update()
    {
    	int res = epic_disp_update();
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_0(display_update_obj, mp_display_update);
    
    Gerd's avatar
    Gerd committed
    
    static mp_obj_t mp_display_open()
    {
    	int res = epic_disp_open();
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_0(display_open_obj, mp_display_open);
    
    Gerd's avatar
    Gerd committed
    
    static mp_obj_t mp_display_close()
    {
    	int res = epic_disp_close();
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    	return mp_const_none;
    }
    
    static MP_DEFINE_CONST_FUN_OBJ_0(display_close_obj, mp_display_close);
    
    Gerd's avatar
    Gerd committed
    
    /* The globals table for this module */
    static const mp_rom_map_elem_t display_module_globals_table[] = {
    	{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_sys_display) },
    	{ MP_ROM_QSTR(MP_QSTR_open), MP_ROM_PTR(&display_open_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_close), MP_ROM_PTR(&display_close_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_print), MP_ROM_PTR(&display_print_obj) },
    
    	{ MP_ROM_QSTR(MP_QSTR_print_adv), MP_ROM_PTR(&display_print_adv_obj) },
    
    	{ MP_ROM_QSTR(MP_QSTR_pixel), MP_ROM_PTR(&display_pixel_obj) },
    
    	{ MP_ROM_QSTR(MP_QSTR_blit), MP_ROM_PTR(&display_blit_obj) },
    
    	{ MP_ROM_QSTR(MP_QSTR_backlight), MP_ROM_PTR(&display_backlight_obj) },
    
    Gerd's avatar
    Gerd committed
    	{ MP_ROM_QSTR(MP_QSTR_line), MP_ROM_PTR(&display_line_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_rect), MP_ROM_PTR(&display_rect_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_circ), MP_ROM_PTR(&display_circ_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_clear), MP_ROM_PTR(&display_clear_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_update), MP_ROM_PTR(&display_update_obj) },
    };
    static MP_DEFINE_CONST_DICT(
    	display_module_globals, display_module_globals_table
    );
    
    const mp_obj_module_t display_module = {
    	.base    = { &mp_type_module },
    	.globals = (mp_obj_dict_t *)&display_module_globals,
    };
    
    /* clang-format off */
    MP_REGISTER_MODULE(MP_QSTR_sys_display, display_module, MODULE_DISPLAY_ENABLED);