| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 | /*  * (c) Philippe G. 2019, philippe_44@outlook.com * * This software is released under the MIT License. * https://opensource.org/licenses/MIT *  */ #ifndef _GDS_PRIVATE_H_#define _GDS_PRIVATE_H_#include <stdint.h>#include <stdbool.h>#include "esp_attr.h"#include "gds.h"#include "gds_err.h"#define GDS_ALLOC_NONE		0x80#define GDS_ALLOC_IRAM		0x01#define GDS_ALLOC_IRAM_SPI	0x02#define GDS_CLIPDEBUG_NONE 0#define GDS_CLIPDEBUG_WARNING 1#define GDS_CLIPDEBUG_ERROR 2#if CONFIG_GDS_CLIPDEBUG == GDS_CLIPDEBUG_NONE    /*     * Clip silently with no console output.     */    #define ClipDebug( x, y )#elif CONFIG_GDS_CLIPDEBUG == GDS_CLIPDEBUG_WARNING    /*     * Log clipping to the console as a warning.     */    #define ClipDebug( x, y ) { \        ESP_LOGW( __FUNCTION__, "Line %d: Pixel at %d, %d CLIPPED", __LINE__, x, y ); \    }#elif CONFIG_GDS_CLIPDEBUG == GDS_CLIPDEBUG_ERROR    /*     * Log clipping as an error to the console.     * Also invokes an abort with stack trace.     */    #define ClipDebug( x, y ) { \        ESP_LOGE( __FUNCTION__, "Line %d: Pixel at %d, %d CLIPPED, ABORT", __LINE__, x, y ); \        abort( ); \    }#endif#define GDS_ALWAYS_INLINE __attribute__( ( always_inline ) )#define MAX_LINES	8#if ! defined BIT#define BIT( n ) ( 1 << ( n ) )#endifstruct GDS_Device;struct GDS_FontDef;/* * These can optionally return a succeed/fail but are as of yet unused in the driver. */typedef bool ( *WriteCommandProc ) ( struct GDS_Device* Device, uint8_t Command );typedef bool ( *WriteDataProc ) ( struct GDS_Device* Device, const uint8_t* Data, size_t DataLength );struct spi_device_t;typedef struct spi_device_t* spi_device_handle_t;#define GDS_IF_SPI	0#define GDS_IF_I2C	1struct GDS_Device {	uint8_t IF;	int8_t RSTPin;	struct {		int8_t Pin, Channel;		int PWM;		} Backlight;	union {		// I2C Specific		struct {			uint8_t Address;		};		// SPI specific		struct {			spi_device_handle_t SPIHandle;			int8_t CSPin;		};	};		    // cooked text mode	struct {		int16_t Y, Space;		const struct GDS_FontDef* Font;	} Lines[MAX_LINES];		uint16_t Width;    uint16_t Height;	uint8_t Depth, Mode;		uint8_t	Alloc;		uint8_t* Framebuffer;    uint32_t FramebufferSize;	bool Dirty;	// default fonts when using direct draw		const struct GDS_FontDef* Font;    bool FontForceProportional;    bool FontForceMonospace;	// various driver-specific method	// must always provide 	bool (*Init)( struct GDS_Device* Device);	void (*Update)( struct GDS_Device* Device );	// may provide if supported	void (*SetContrast)( struct GDS_Device* Device, uint8_t Contrast );	void (*DisplayOn)( struct GDS_Device* Device );	void (*DisplayOff)( struct GDS_Device* Device );	void (*SetLayout)( struct GDS_Device* Device, bool HFlip, bool VFlip, bool Rotate );	// must provide for depth other than 1 (vertical) and 4 (may provide for optimization)	void (*DrawPixelFast)( struct GDS_Device* Device, int X, int Y, int Color );	void (*DrawBitmapCBR)(struct GDS_Device* Device, uint8_t *Data, int Width, int Height, int Color );	// may provide for optimization	void (*DrawRGB)( struct GDS_Device* Device, uint8_t *Image,int x, int y, int Width, int Height, int RGB_Mode );	void (*ClearWindow)( struct GDS_Device* Device, int x1, int y1, int x2, int y2, int Color );		    	// interface-specific methods	    WriteCommandProc WriteCommand;    WriteDataProc WriteData;	// 32 bytes for whatever the driver wants (should be aligned as it's 32 bits)		uint32_t Private[8];};bool GDS_Reset( struct GDS_Device* Device );bool GDS_Init( struct GDS_Device* Device );inline bool IsPixelVisible( struct GDS_Device* Device, int x, int y )  {    bool Result = (        ( x >= 0 ) &&        ( x < Device->Width ) &&        ( y >= 0 ) &&        ( y < Device->Height )    ) ? true : false;#if CONFIG_GDS_CLIPDEBUG > 0    if ( Result == false ) {        ClipDebug( x, y );    }#endif    return Result;}inline void IRAM_ATTR GDS_DrawPixel1Fast( struct GDS_Device* Device, int X, int Y, int Color ) {    uint32_t YBit = ( Y & 0x07 );    uint8_t* FBOffset;    /*      * We only need to modify the Y coordinate since the pitch     * of the screen is the same as the width.     * Dividing Y by 8 gives us which row the pixel is in but not     * the bit position.     */    Y>>= 3;    FBOffset = Device->Framebuffer + ( ( Y * Device->Width ) + X );    if ( Color == GDS_COLOR_XOR ) {        *FBOffset ^= BIT( YBit );    } else {        *FBOffset = ( Color == GDS_COLOR_BLACK ) ? *FBOffset & ~BIT( YBit ) : *FBOffset | BIT( YBit );    }}inline void IRAM_ATTR GDS_DrawPixel4Fast( struct GDS_Device* Device, int X, int Y, int Color ) {	uint8_t* FBOffset = Device->Framebuffer + ( (Y * Device->Width >> 1) + (X >> 1));	*FBOffset = X & 0x01 ? (*FBOffset & 0x0f) | ((Color & 0x0f) << 4) : ((*FBOffset & 0xf0) | (Color & 0x0f));}inline void IRAM_ATTR GDS_DrawPixel8Fast( struct GDS_Device* Device, int X, int Y, int Color ) {	Device->Framebuffer[Y * Device->Width + X] = Color;}// assumes that Color is 16 bits R..RG..GB..B from MSB to LSB and FB wants 1st serialized byte to start with Rinline void IRAM_ATTR GDS_DrawPixel16Fast( struct GDS_Device* Device, int X, int Y, int Color ) {	uint16_t* FBOffset = (uint16_t*) Device->Framebuffer + Y * Device->Width + X;	*FBOffset = __builtin_bswap16(Color);}// assumes that Color is 18 bits RGB from MSB to LSB RRRRRRGGGGGGBBBBBB, so byte[0] is B // FB is 3-bytes packets and starts with R for serialization so 0,1,2 ... = xxRRRRRR xxGGGGGG xxBBBBBB inline void IRAM_ATTR GDS_DrawPixel18Fast( struct GDS_Device* Device, int X, int Y, int Color ) {	uint8_t* FBOffset = Device->Framebuffer + (Y * Device->Width + X) * 3;	*FBOffset++ = Color >> 12; *FBOffset++ = (Color >> 6) & 0x3f; *FBOffset = Color & 0x3f;}// assumes that Color is 24 bits RGB from MSB to LSB RRRRRRRRGGGGGGGGBBBBBBBB, so byte[0] is B // FB is 3-bytes packets and starts with R for serialization so 0,1,2 ... = RRRRRRRR GGGGGGGG BBBBBBBB inline void IRAM_ATTR GDS_DrawPixel24Fast( struct GDS_Device* Device, int X, int Y, int Color ) {	uint8_t* FBOffset = Device->Framebuffer + (Y * Device->Width + X) * 3;	*FBOffset++ = Color >> 16; *FBOffset++ = Color >> 8; *FBOffset = Color;}inline void IRAM_ATTR GDS_DrawPixelFast( struct GDS_Device* Device, int X, int Y, int Color ) {    if (Device->DrawPixelFast) Device->DrawPixelFast( Device, X, Y, Color );	else if (Device->Depth == 4) GDS_DrawPixel4Fast( Device, X, Y, Color );	else if (Device->Depth == 1) GDS_DrawPixel1Fast( Device, X, Y, Color );	else if (Device->Depth == 16) GDS_DrawPixel16Fast( Device, X, Y, Color );		else if (Device->Depth == 24 && Device->Mode == GDS_RGB666) GDS_DrawPixel18Fast( Device, X, Y, Color );		else if (Device->Depth == 24 && Device->Mode == GDS_RGB888) GDS_DrawPixel24Fast( Device, X, Y, Color );		else if (Device->Depth == 8) GDS_DrawPixel8Fast( Device, X, Y, Color );	}	inline void IRAM_ATTR GDS_DrawPixel( struct GDS_Device* Device, int x, int y, int Color ) {    if ( IsPixelVisible( Device, x, y ) == true ) {        GDS_DrawPixelFast( Device, x, y, Color );    }}#endif
 |