| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456 | /****************************************************************************** | File      	:   epd2in13_V3.cpp* | Author      :   Waveshare team* | Function    :   2.13inch e-paper V3* | Info        :*----------------* |	This version:   V1.0* | Date        :   2021-11-01* | Info        :## Permission is hereby granted, free of charge, to any person obtaining a copy# of this software and associated documnetation files (the "Software"), to deal# in the Software without restriction, including without limitation the rights# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell# copies of the Software, and to permit persons to  whom the Software is# furished to do so, subject to the following conditions:## The above copyright notice and this permission notice shall be included in# all copies or substantial portions of the Software.## THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,# FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER# LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN# THE SOFTWARE.#******************************************************************************/#include <stdlib.h>#include "epd2in13_V3.h"const unsigned char lut_full_update[]= {	0x80,	0x4A,	0x40,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x40,	0x4A,	0x80,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x80,	0x4A,	0x40,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x40,	0x4A,	0x80,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0xF,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0xF,	0x0,	0x0,	0xF,	0x0,	0x0,	0x2,						0xF,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x1,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,						0x22,	0x22,	0x22,	0x22,	0x22,	0x22,	0x0,	0x0,	0x0,				0x22,	0x17,	0x41,	0x0,	0x32,	0x36	};const unsigned char lut_partial_update[]= { 	0x0,0x40,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x80,0x80,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x40,0x40,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x80,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x14,0x0,0x0,0x0,0x0,0x0,0x0,  	0x1,0x0,0x0,0x0,0x0,0x0,0x0,	0x1,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x0,0x0,0x0,0x0,0x0,0x0,0x0,	0x22,0x22,0x22,0x22,0x22,0x22,0x0,0x0,0x0,	0x22,0x17,0x41,0x00,0x32,0x36,};Epd::~Epd(){};/******************************************************************************function :	Pin definitionparameter:******************************************************************************/Epd::Epd(){    reset_pin = RST_PIN;    dc_pin = DC_PIN;    cs_pin = CS_PIN;    busy_pin = BUSY_PIN;    width = EPD_WIDTH;    height = EPD_HEIGHT;    bufwidth = 128/8;  //16    bufheight = 63;};/******************************************************************************function :	send commandparameter:     command : Command register******************************************************************************/void Epd::SendCommand(unsigned char command){    DigitalWrite(dc_pin, LOW);    SpiTransfer(command);}/******************************************************************************function :	send dataparameter:    Data : Write data******************************************************************************/void Epd::SendData(unsigned char data){    DigitalWrite(dc_pin, HIGH);    SpiTransfer(data);}/******************************************************************************function :	Wait until the busy_pin goes LOWparameter:******************************************************************************/void Epd::WaitUntilIdle(void){    while(1) {      //LOW: idle, HIGH: busy        if(DigitalRead(busy_pin) == 0)            break;        DelayMs(10);    }}/******************************************************************************function :	Setting the display windowparameter:	Xstart : X-axis starting position	Ystart : Y-axis starting position	Xend : End position of X-axis	Yend : End position of Y-axis******************************************************************************/void Epd::SetWindows(unsigned char Xstart, unsigned char Ystart, unsigned char Xend, unsigned char Yend){    SendCommand(0x44); // SET_RAM_X_ADDRESS_START_END_POSITION    SendData((Xstart>>3) & 0xFF);    SendData((Xend>>3) & 0xFF);	    SendCommand(0x45); // SET_RAM_Y_ADDRESS_START_END_POSITION    SendData(Ystart & 0xFF);    SendData((Ystart >> 8) & 0xFF);    SendData(Yend & 0xFF);    SendData((Yend >> 8) & 0xFF);}/******************************************************************************function :	Set Cursorparameter:	Xstart : X-axis starting position	Ystart : Y-axis starting position******************************************************************************/void Epd::SetCursor(unsigned char Xstart, unsigned char Ystart){    SendCommand(0x4E); // SET_RAM_X_ADDRESS_COUNTER    SendData(Xstart & 0xFF);    SendCommand(0x4F); // SET_RAM_Y_ADDRESS_COUNTER    SendData(Ystart & 0xFF);    SendData((Ystart >> 8) & 0xFF);}/******************************************************************************function :	Send lut data and configurationparameter:	    lut :   lut data******************************************************************************/void Epd::Lut(unsigned char *lut){	unsigned char count;	SendCommand(0x32);	for(count = 0; count < 153; count++) {		SendData(lut[count]);	}	SendCommand(0x3f);	SendData(*(lut+153));	SendCommand(0x03);	// gate voltage	SendData(*(lut+154));	SendCommand(0x04);	// source voltage	SendData(*(lut+155));	// VSH	SendData(*(lut+156));	// VSH2	SendData(*(lut+157));	// VSL	SendCommand(0x2c);		// VCOM	SendData(*(lut+158));}/******************************************************************************function :	Initialize the e-Paper registerparameter:	Mode : Mode selection******************************************************************************/int Epd::Init(char Mode){    /* this calls the peripheral hardware interface, see epdif */    if (IfInit() != 0) {        return -1;    }        Reset();        int count;    if(Mode == FULL) {        WaitUntilIdle();        SendCommand(0x12); // soft reset        WaitUntilIdle();        SendCommand(0x01); //Driver output control        SendData(0xF9);        SendData(0x00);        SendData(0x00);        SendCommand(0x11); //data entry mode        SendData(0x03);		SetWindows(0, 0, EPD_WIDTH-1, EPD_HEIGHT-1);		SetCursor(0, 0);				SendCommand(0x3C); //BorderWavefrom		SendData(0x05);			SendCommand(0x21); //  Display update control		SendData(0x00);		SendData(0x80);			SendCommand(0x18); //Read built-in temperature sensor		SendData(0x80);			WaitUntilIdle();		Lut(lut_full_update);    } else if(Mode == PART) {				DigitalWrite(reset_pin, LOW);                //module reset		DelayMs(1);		DigitalWrite(reset_pin, HIGH);		Lut(lut_partial_update);		        SendCommand(0x37);        SendData(0x00);        SendData(0x00);        SendData(0x00);        SendData(0x00);		SendData(0x00);        SendData(0x40);        SendData(0x00);		SendData(0x00);		SendData(0x00);        SendData(0x00);		SendCommand(0x3C); //BorderWavefrom		SendData(0x80);				SendCommand(0x22); //Display Update Sequence Option		SendData(0xC0);    // Enable clock and  Enable analog		SendCommand(0x20);  //Activate Display Update Sequence		WaitUntilIdle();  				SetWindows(0, 0, EPD_WIDTH-1, EPD_HEIGHT-1);		SetCursor(0, 0);    } else {        return -1;    }    return 0;}/******************************************************************************function :	Software resetparameter:******************************************************************************/void Epd::Reset(void){    DigitalWrite(reset_pin, HIGH);    DelayMs(20);    DigitalWrite(reset_pin, LOW);                //module reset    DelayMs(2);    DigitalWrite(reset_pin, HIGH);    DelayMs(20);    this->count = 0; }/******************************************************************************function :	Clear screenparameter:******************************************************************************/void Epd::Clear(void){    int w, h;    w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);    h = EPD_HEIGHT;    SendCommand(0x24);    for (int j = 0; j < h; j++) {        for (int i = 0; i < w; i++) {            SendData(0xff);        }    }    //DISPLAY REFRESH    SendCommand(0x22);    SendData(0xC7);    SendCommand(0x20);    WaitUntilIdle();}/******************************************************************************function :	Sends the image buffer in RAM to e-Paper and displaysparameter:	frame_buffer : Image data******************************************************************************/void Epd::Display(const unsigned char* frame_buffer){    int w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);    int h = EPD_HEIGHT;    if (frame_buffer != NULL) {        SendCommand(0x24);        for (int j = 0; j < h; j++) {            for (int i = 0; i < w; i++) {                SendData(pgm_read_byte(&frame_buffer[i + j * w]));            }        }    }    //DISPLAY REFRESH    SendCommand(0x22);    SendData(0xC7);    SendCommand(0x20);    WaitUntilIdle();}void Epd::Display1(const unsigned char* frame_buffer) {    if(this->count == 0){        SendCommand(0x24);        this->count++;    }else if(this->count > 0 && this->count < 4 ){        this->count++;    }    for(int i = 0; i < this->bufwidth * this->bufheight; i++){            SendData(frame_buffer[i]);    }    if(this->count == 4){        SendCommand(0x22);        SendData(0xC7);        SendCommand(0x20);        WaitUntilIdle();        this->count = 0;    }}/******************************************************************************function :	Refresh a base imageparameter:	frame_buffer : Image data	******************************************************************************/void Epd::DisplayPartBaseImage(const unsigned char* frame_buffer){    int w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);    int h = EPD_HEIGHT;    if (frame_buffer != NULL) {        SendCommand(0x24);        for (int j = 0; j < h; j++) {            for (int i = 0; i < w; i++) {                SendData(pgm_read_byte(&frame_buffer[i + j * w]));            }        }        SendCommand(0x26);        for (int j = 0; j < h; j++) {            for (int i = 0; i < w; i++) {                SendData(pgm_read_byte(&frame_buffer[i + j * w]));            }        }    }    //DISPLAY REFRESH    SendCommand(0x22);    SendData(0xC7);    SendCommand(0x20);    WaitUntilIdle();}/******************************************************************************function :	Sends the image buffer in RAM to e-Paper and partial refreshparameter:	frame_buffer : Image data******************************************************************************/void Epd::DisplayPart(const unsigned char* frame_buffer){    int w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);    int h = EPD_HEIGHT;    if (frame_buffer != NULL) {        SendCommand(0x24);        for (int j = 0; j < h; j++) {            for (int i = 0; i < w; i++) {                SendData(pgm_read_byte(&frame_buffer[i + j * w]));            }        }    }    //DISPLAY REFRESH    SendCommand(0x22);    SendData(0x0f);    SendCommand(0x20);    WaitUntilIdle();}/******************************************************************************function :	Clear screenparameter:******************************************************************************/void Epd::ClearPart(void){    int w, h;    w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);    h = EPD_HEIGHT;    SendCommand(0x24);    for (int j = 0; j < h; j++) {        for (int i = 0; i < w; i++) {            SendData(0xff);        }    }    //DISPLAY REFRESH    SendCommand(0x22);    SendData(0x0f);    SendCommand(0x20);    WaitUntilIdle();}/******************************************************************************function :	Enter sleep modeparameter:******************************************************************************/void Epd::Sleep(){    SendCommand(0x10); //enter deep sleep    SendData(0x01);    DelayMs(200);    DigitalWrite(reset_pin, LOW);}/* END OF FILE */
 |