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
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); |
|
} |
|
}
|
|
|