| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576 | /****************************************************************************** | File      	:   epd1in54_V2.cpp* | Author      :   Waveshare team* | Function    :   1.54inch e-paper V2* | Info        :*----------------* |	This version:   V1.0* | Date        :   2019-06-24* | 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 "epd1in54_V2.h"// waveform full refreshunsigned char WF_Full_1IN54[159] ={											0x80,	0x48,	0x40,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,0x40,	0x48,	0x80,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,0x80,	0x48,	0x40,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,0x40,	0x48,	0x80,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,0xA,	0x0,	0x0,	0x0,	0x0,	0x0,	0x0,					0x8,	0x1,	0x0,	0x8,	0x1,	0x0,	0x2,					0xA,	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,	0x0,	0x32,	0x20};// waveform partial refresh(fast)unsigned char WF_PARTIAL_1IN54_0[159] ={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,0xF,0x0,0x0,0x0,0x0,0x0,0x0,0x1,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,0x0,0x0,0x0,0x0,0x0,0x0,0x22,0x22,0x22,0x22,0x22,0x22,0x0,0x0,0x0,0x02,0x17,0x41,0xB0,0x32,0x28,};Epd::~Epd(){};Epd::Epd(){	reset_pin = RST_PIN;	dc_pin = DC_PIN;	cs_pin = CS_PIN;	busy_pin = BUSY_PIN;	width = EPD_WIDTH;	height = EPD_HEIGHT;};/** *  @brief: basic function for sending commands */void Epd::SendCommand(unsigned char command){	DigitalWrite(dc_pin, LOW);	SpiTransfer(command);}/** *  @brief: basic function for sending data */void Epd::SendData(unsigned char data){	DigitalWrite(dc_pin, HIGH);	SpiTransfer(data);}/** *  @brief: Wait until the busy_pin goes HIGH */void Epd::WaitUntilIdle(void){	while(DigitalRead(busy_pin) == 1) {      //LOW: idle, HIGH: busy		DelayMs(100);	}	DelayMs(200);}void Epd::Lut(unsigned char* lut){	SendCommand(0x32);	for(unsigned char i=0; i<153; i++)		SendData(lut[i]);	WaitUntilIdle();}void Epd::SetLut(unsigned char* lut){	Lut(lut);	    SendCommand(0x3f);    SendData(lut[153]);	    SendCommand(0x03);    SendData(lut[154]);	    SendCommand(0x04);    SendData(lut[155]);	SendData(lut[156]);	SendData(lut[157]);		SendCommand(0x2c);    SendData(lut[158]);}// High Directionint Epd::HDirInit(void){	/* this calls the peripheral hardware interface, see epdif */	if (IfInit() != 0) {		return -1;	}	/* EPD hardware init start */	Reset();	WaitUntilIdle();	SendCommand(0x12);  //SWRESET	WaitUntilIdle();	SendCommand(0x01); //Driver output control	SendData(0xC7);	SendData(0x00);	SendData(0x01);	SendCommand(0x11); //data entry mode	SendData(0x01);	SendCommand(0x44); //set Ram-X address start/end position	SendData(0x00);	SendData(0x18);    //0x0C-->(18+1)*8=200	SendCommand(0x45); //set Ram-Y address start/end position	SendData(0xC7);   //0xC7-->(199+1)=200	SendData(0x00);	SendData(0x00);	SendData(0x00);	SendCommand(0x3C); //BorderWavefrom	SendData(0x01);	SendCommand(0x18);	SendData(0x80);	SendCommand(0x22); // //Load Temperature and waveform setting.	SendData(0XB1);	SendCommand(0x20);	SendCommand(0x4E);   // set RAM x address count to 0;	SendData(0x00);	SendCommand(0x4F);   // set RAM y address count to 0X199;	SendData(0xC7);	SendData(0x00);	WaitUntilIdle();	SetLut(WF_Full_1IN54);	/* EPD hardware init end */	return 0;}// Low Directionint Epd::LDirInit(void){	/* this calls the peripheral hardware interface, see epdif */	if (IfInit() != 0) {		return -1;	}	/* EPD hardware init start */	Reset();	WaitUntilIdle();	SendCommand(0x12);  //SWRESET	WaitUntilIdle();	SendCommand(0x01); //Driver output control	SendData(0xC7);	SendData(0x00);	SendData(0x00);	SendCommand(0x11); //data entry mode	SendData(0x03);  SendCommand(0x44);  /* x point must be the multiple of 8 or the last 3 bits will be ignored */  SendData((0 >> 3) & 0xFF);  SendData((199 >> 3) & 0xFF);  SendCommand(0x45);  SendData(0 & 0xFF);  SendData((0 >> 8) & 0xFF);  SendData(199 & 0xFF);  SendData((199 >> 8) & 0xFF);	SendCommand(0x3C); //BorderWavefrom	SendData(0x01);	SendCommand(0x18);	SendData(0x80);	SendCommand(0x22); // //Load Temperature and waveform setting.	SendData(0XB1);	SendCommand(0x20);	SendCommand(0x4E);   // set RAM x address count to 0;	SendData(0x00);	SendCommand(0x4F);   // set RAM y address count to 0X199;	SendData(0xC7);	SendData(0x00);	WaitUntilIdle();	SetLut(WF_Full_1IN54);	/* EPD hardware init end */	return 0;}/** *  @brief: module reset. *          often used to awaken the module in deep sleep, *          see Epd::Sleep(); */void Epd::Reset(void){	DigitalWrite(reset_pin, HIGH);	DelayMs(20);	DigitalWrite(reset_pin, LOW);                //module reset	DelayMs(5);	DigitalWrite(reset_pin, HIGH);	DelayMs(20);}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);		}	}	SendCommand(0x26);	for (int j = 0; j < h; j++) {		for (int i = 0; i < w; i++) {			SendData(0xff);		}	}	//DISPLAY REFRESH	DisplayFrame();}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	DisplayFrame();}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	DisplayFrame();}void Epd::DisplayPartBaseWhiteImage(void){	int w = (EPD_WIDTH % 8 == 0)? (EPD_WIDTH / 8 ): (EPD_WIDTH / 8 + 1);	int h = EPD_HEIGHT;	SendCommand(0x24);	for (int j = 0; j < h; j++) {		for (int i = 0; i < w; i++) {			SendData(0xff);		}	}	SendCommand(0x26);	for (int j = 0; j < h; j++) {		for (int i = 0; i < w; i++) {			SendData(0xff);		}	}	//DISPLAY REFRESH	DisplayFrame();}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	DisplayPartFrame();}/** *  @brief: private function to specify the memory area for data R/W */void Epd::SetMemoryArea(int x_start, int y_start, int x_end, int y_end){	SendCommand(0x44);	/* x point must be the multiple of 8 or the last 3 bits will be ignored */	SendData((x_start >> 3) & 0xFF);	SendData((x_end >> 3) & 0xFF);	SendCommand(0x45);	SendData(y_start & 0xFF);	SendData((y_start >> 8) & 0xFF);	SendData(y_end & 0xFF);	SendData((y_end >> 8) & 0xFF);}/** *  @brief: private function to specify the start point for data R/W */void Epd::SetMemoryPointer(int x, int y){	SendCommand(0x4e);	/* x point must be the multiple of 8 or the last 3 bits will be ignored */	SendData((x >> 3) & 0xFF);	SendCommand(0x4F);	SendData(y & 0xFF);	SendData((y >> 8) & 0xFF);	WaitUntilIdle();}/** *  @brief: update the display *          there are 2 memory areas embedded in the e-paper display *          but once this function is called, *          the the next action of SetFrameMemory or ClearFrame will *          set the other memory area. */void Epd::DisplayFrame(void){	//DISPLAY REFRESH	SendCommand(0x22);	SendData(0xc7);	SendCommand(0x20);	WaitUntilIdle();}void Epd::DisplayPartFrame(void){	SendCommand(0x22);	SendData(0xcF);	SendCommand(0x20);	WaitUntilIdle();}void Epd::SetFrameMemory(        const unsigned char* image_buffer,        int x,        int y,        int image_width,        int image_height){	int x_end;	int y_end;		DigitalWrite(reset_pin, LOW);                //module reset	DelayMs(2);	DigitalWrite(reset_pin, HIGH);	DelayMs(2);	SendCommand(0x3c);	SendData(0x80);	if (	        image_buffer == NULL ||	        x < 0 || image_width < 0 ||	        y < 0 || image_height < 0	) {		return;	}	/* x point must be the multiple of 8 or the last 3 bits will be ignored */	x &= 0xF8;	image_width &= 0xF8;	if (x + image_width >= this->width) {		x_end = this->width - 1;	} else {		x_end = x + image_width - 1;	}	if (y + image_height >= this->height) {		y_end = this->height - 1;	} else {		y_end = y + image_height - 1;	}	SetMemoryArea(x, y, x_end, y_end);	SetMemoryPointer(x, y);	SendCommand(0x24);	/* send the image data */	for (int j = 0; j < y_end - y + 1; j++) {		for (int i = 0; i < (x_end - x + 1) / 8; i++) {			SendData(image_buffer[i + j * (image_width / 8)]);		}	}}void Epd::SetFrameMemoryPartial(        const unsigned char* image_buffer,        int x,        int y,        int image_width,        int image_height){	int x_end;	int y_end;		DigitalWrite(reset_pin, LOW);                //module reset	DelayMs(2);	DigitalWrite(reset_pin, HIGH);	DelayMs(2);	SetLut(WF_PARTIAL_1IN54_0);    SendCommand(0x37);     SendData(0x00);      SendData(0x00);      SendData(0x00);      SendData(0x00);     SendData(0x00);  	    SendData(0x40);      SendData(0x00);      SendData(0x00);       SendData(0x00);      SendData(0x00);	SendCommand(0x3c);	SendData(0x80);	SendCommand(0x22); 	SendData(0xc0); 	SendCommand(0x20); 	WaitUntilIdle();		if (	        image_buffer == NULL ||	        x < 0 || image_width < 0 ||	        y < 0 || image_height < 0	) {		return;	}	/* x point must be the multiple of 8 or the last 3 bits will be ignored */	x &= 0xF8;	image_width &= 0xF8;	if (x + image_width >= this->width) {		x_end = this->width - 1;	} else {		x_end = x + image_width - 1;	}	if (y + image_height >= this->height) {		y_end = this->height - 1;	} else {		y_end = y + image_height - 1;	}	SetMemoryArea(x, y, x_end, y_end);	SetMemoryPointer(x, y);	SendCommand(0x24);	/* send the image data */	for (int j = 0; j < y_end - y + 1; j++) {		for (int i = 0; i < (x_end - x + 1) / 8; i++) {			SendData(image_buffer[i + j * (image_width / 8)]);		}	}}/** *  @brief: After this command is transmitted, the chip would enter the *          deep-sleep mode to save power. *          The deep sleep mode would return to standby by hardware reset. *          The only one parameter is a check code, the command would be *          executed if check code = 0xA5. *          You can use Epd::Init() to awaken */void Epd::Sleep(){	SendCommand(0x10); //enter deep sleep	SendData(0x01);	DelayMs(200);	DigitalWrite(reset_pin, LOW);}/* END OF FILE */
 |