Toying around with a 4.2 in. e-Paper module from Waveshare and the Raspberry Pi Pico. Maybe it'll become a library some day.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

222 lines
5.2 KiB

/* epd.c - E-Paper Display library API */
/* Includes */
#include "epd.h"
#include "string.h"
#include "util.h"
/* Function Definitions */
/**
* \brief initialize a new epd object
*
* \details Initialize a new epd object given the configuration options.
*
* \param[in,out] epd pointer to an epd_t
* \param[in] cfg pointer to a populated epd_platform_config_t
*/
void epd_init(epd_t *epd, epd_platform_config_t *cfg)
{
memcpy(&(epd->platform), cfg, sizeof(epd_platform_config_t));
epd->dbuf = NULL;
epd->font = NULL;
epd->draw_color = 0x00;
epd->display.w = 0;
epd->display.h = 0;
}
/* Function Definitions */
/**
* \brief initialize a new display
*
* \details Initialize and configure a new display.
*
* \param[in,out] epd pointer to an epd_t
* \param[in] cfg pointer to a populated epd_display_config_t
*/
void epd_display_init(epd_t *epd, epd_display_config_t *cfg)
{
memcpy(&(epd->display), cfg, sizeof(epd_display_config_t));
}
/**
* \brief attach a display buffer to the epd_t object
*
* \details Attach a display buffer to the epd_t object to do things which
* require buffering.
*
* \param[in,out] epd pointer to an epd_t
* \param[in,out] dbuf pointer to a dbuf_t array large enough to hold the data
* for this display
*
* \return 0 on success; otherwise failure
*/
int8_t epd_dbuf_set(epd_t *epd, epd_dbuf_t *dbuf)
{
return_val_if_fail(epd, -1);
return_val_if_fail(dbuf, -1);
epd->dbuf = dbuf;
memset(epd->dbuf, 0xff, (epd->display.w * epd->display.h * sizeof(epd_dbuf_t) / 8));
return 0;
}
/**
* \brief clear a display buffer
*
* \details Clear the display buffer attached to an epd_t object.
*
* \param[in,out] epd pointer to an epd_t
*
* \return 0 on success; otherwise failure
*/
int8_t epd_dbuf_clear(epd_t *epd)
{
return_val_if_fail(epd, -1);
return_val_if_fail(epd->dbuf, -1);
memset(epd->dbuf, 0xff, (epd->display.w * epd->display.h * sizeof(epd_dbuf_t) / 8));
return 0;
}
/**
* \brief draw a line
*
* \details Draw a horizontal or vertical line on the display buffer.
*
* \param[in,out] epd pointer to an epd_t
* \param[in] x starting x coordinate
* \param[in] y starting y coordinate
* \param[in] len length of the line
* \param[in] dir direction of the line
*/
void epd_draw_line(epd_t *epd, uint16_t x, uint16_t y, uint16_t len, epd_dir_t dir)
{
uint32_t i;
epd_dbuf_t tmp;
uint16_t boundary;
return_if_fail(epd);
return_if_fail(epd->dbuf);
return_if_fail(len);
return_if_fail(x < (epd->display.w * 8));
return_if_fail(y < epd->display.h);
switch (dir)
{
case EPD_DIR_L2R:
/* calculate first byte boundary */
boundary = MIN((x % 8) + len, 8);
/* calculate byte at left of line */
tmp = 0xff;
for (i = x % 8; i < boundary; i++)
{
tmp &= ~(0x80 >> i);
}
epd->dbuf[x / 8 + (y * epd->display.w) / 8] &= tmp;
/* calculate full bytes */
for (i = (x + 7) / 8; i < (x + len) / 8; i++)
{
epd->dbuf[i + (y * epd->display.w) / 8] &= 0x00;
}
/* calculate byte at right of line */
if((x % 8) + len < 8)
{
break;
}
tmp = 0xff;
for (i = 8 - ((x + len) % 8); i < boundary; i++)
{
tmp &= ~(0x01 << i);
}
epd->dbuf[(x + len) / 8 + (y * epd->display.w) / 8] &= tmp;
# if defined(AIDS)
for (i = (x + 7) / 8; i < (x + len) / 8; i++)
{
epd->dbuf[i + (y * epd->display.w) / 8] &= 0x00;
}
/* calculate byte at left of line */
tmp = 0x00;
for (i = x % 8; i < MIN(len, 8); i++)
{
tmp <<= 1;
tmp |= 0x01;
}
epd->dbuf[x / 8 + (y * epd->display.w) / 8] &= ~tmp;
/* calculate byte at right of line */
tmp = 0x00;
for (i = 7 - ((x + len) % 8); i < MIN(len, 8); i++)
{
tmp >>= 1;
tmp |= 0x80;
}
epd->dbuf[(x + len) / 8 + (y * epd->display.w) / 8] &= ~tmp;
# endif
break;
case EPD_DIR_T2B:
for (i = y; i < (y + len); i++)
{
epd->dbuf[(x + i * epd->display.w) / 8] &= ~(0x80 >> (x % 8));
}
break;
case EPD_DIR_R2L:
break;
case EPD_DIR_B2T:
return_if_fail(y - len >= 0);
for (i = y; i > (y - len); i--)
{
epd->dbuf[(x + i * epd->display.w) / 8] &= ~(0x80 >> (x % 8));
}
break;
default:
return;
}
}
/**
* \brief send the dbuf to the display
*
* \details Display the contents of the dbuf on the screen.
*
* \param[in,out] epd pointer to an epd_t
*
* \returns 0 on success
*/
int32_t epd_update(epd_t *epd)
{
return_val_if_fail(epd, -1);
return_val_if_fail(epd->dbuf, -1);
return epd->display.write(epd);
}
/**
* \brief send the dbuf to the display
*
* \details Display the contents of the dbuf on the screen within a specified
* window. Falls back on a full update if the display does not implement
* partial updates.
*
* \param[in,out] epd pointer to an epd_t
* \param[in] x0 x-coordinate of first pixel in window
* \param[in] x1 x-coordinate of last pixel in window
* \param[in] y0 y-coordinate of first pixel in window
* \param[in] y1 y-coordinate of last pixel in window
*
* \returns 0 on success
*/
int32_t epd_update_partial(epd_t *epd, uint16_t x0, uint16_t x1, uint16_t y0, uint16_t y1)
{
return_val_if_fail(epd, -1);
return_val_if_fail(epd->dbuf, -1);
if(epd->display.write_partial)
{
return epd->display.write_partial(epd, x0, x1, y0, y1);
}
else
{
return epd->display.write(epd);
}
}