Basically i glued Peter Drescher and Simon Ford libs in a GraphicsDisplay class, then derived TFT or LCD class (which inherits Protocols class), then the most derived ones (Inits), which are per-display and are the only part needed to be adapted to diff hw.
Display/TFT.cpp
- Committer:
- Geremia
- Date:
- 2015-02-16
- Revision:
- 5:b222a9461d6b
- Parent:
- 4:12ba0ecc2c1f
- Child:
- 7:bb0383b91104
File content as of revision 5:b222a9461d6b:
/* mbed UniGraphic library - universal TFT driver class * Copyright (c) 2015 Giuliano Dianda * Released under the MIT License: http://mbed.org/license/mit * * Derived work of: * * mbed library for 240*320 pixel display TFT based on ILI9341 LCD Controller * Copyright (c) 2013 Peter Drescher - DC2PD * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR 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 "TFT.h" //#include "mbed_debug.h" #define SWAP(a, b) { a ^= b; b ^= a; a ^= b; } TFT::TFT(proto_t displayproto, PortName port, PinName CS, PinName reset, PinName DC, PinName WR, PinName RD, const int lcdsize_x, const int lcdsize_y, const char *name) : GraphicsDisplay(name), LCDSIZE_X(lcdsize_x), LCDSIZE_Y(lcdsize_y) { if(displayproto==PAR_8) proto = new PAR8(port, CS, reset, DC, WR, RD); else if(displayproto==PAR_16) proto = new PAR16(port, CS, reset, DC, WR, RD); useNOP=false; scrollbugfix=0; mipistd=false; set_orientation(0); foreground(White); background(Black); set_auto_up(false); //we don't have framebuffer // cls(); // locate(0,0); } TFT::TFT(proto_t displayproto, int Hz, PinName mosi, PinName miso, PinName sclk, PinName CS, PinName reset, PinName DC, const int lcdsize_x, const int lcdsize_y, const char *name) : GraphicsDisplay(name), LCDSIZE_X(lcdsize_x), LCDSIZE_Y(lcdsize_y) { if(displayproto==SPI_8) { proto = new SPI8(Hz, mosi, miso, sclk, CS, reset, DC); useNOP=false; } else if(displayproto==SPI_16) { proto = new SPI16(Hz, mosi, miso, sclk, CS, reset, DC); useNOP=true; } scrollbugfix=0; mipistd=false; set_orientation(0); foreground(White); background(Black); set_auto_up(false); // locate(0,0); } void TFT::wr_cmd8(unsigned char cmd) { if(useNOP) proto->wr_cmd16(cmd); // 0x0000|cmd, 00 is NOP cmd for TFT else proto->wr_cmd8(cmd); } void TFT::wr_data8(unsigned char data) { proto->wr_data8(data); } void TFT::wr_data16(unsigned short data) { proto->wr_data16(data); } void TFT::wr_gram(unsigned short data) { proto->wr_gram(data); } void TFT::wr_gram(unsigned short data, unsigned int count) { proto->wr_gram(data, count); } void TFT::wr_grambuf(unsigned short* data, unsigned int lenght) { proto->wr_grambuf(data, lenght); } unsigned int TFT::rd_data32_wdummy() { return proto->rd_data32_wdummy(); } unsigned short TFT::rd_gram() { return (proto->rd_gram()); } //for TFT, just send data, position counters are in hw void TFT::window_pushpixel(unsigned short color) { proto->wr_gram(color); } void TFT::window_pushpixel(unsigned short color, unsigned int count) { proto->wr_gram(color, count); } void TFT::window_pushpixelbuf(unsigned short* color, unsigned int lenght) { proto->wr_grambuf(color, lenght); } void TFT::hw_reset() { proto->hw_reset(); } void TFT::BusEnable(bool enable) { proto->BusEnable(enable); } // color TFT can rotate in hw (swap raw<->columns) for landscape views void TFT::set_orientation(int o) { orientation = o; wr_cmd8(0x36); switch (orientation) { case 0:// default, portrait view 0° if(mipistd) wr_data8(0x0A); // this is in real a vertical flip enabled, seems most displays are vertical flipped else wr_data8(0x48); //for some other ILIxxxx set_width(LCDSIZE_X); set_height(LCDSIZE_Y); break; case 1:// landscape view +90° if(mipistd) wr_data8(0x28); else wr_data8(0x29);//for some other ILIxxxx set_width(LCDSIZE_Y); set_height(LCDSIZE_X); break; case 2:// portrait view +180° if(mipistd) wr_data8(0x09); else wr_data8(0x99);//for some other ILIxxxx set_width(LCDSIZE_X); set_height(LCDSIZE_Y); break; case 3:// landscape view -90° if(mipistd) wr_data8(0x2B); else wr_data8(0xF8);//for some other ILIxxxx set_width(LCDSIZE_Y); set_height(LCDSIZE_X); break; } } // TFT have both column and raw autoincrement inside a window, with internal counters void TFT::window(int x, int y, int w, int h) { //ili9486 does not like truncated 2A/2B cmds, at least in par mode //setting only start column/page would speedup, but needs a windowmax() before, maybe implement later wr_cmd8(0x2A); wr_data16(x); //start column wr_data16(x+w-1);//end column wr_cmd8(0x2B); wr_data16(y); //start page wr_data16(y+h-1);//end page wr_cmd8(0x2C); //write mem, just send pixels color next } void TFT::window4read(int x, int y, int w, int h) { wr_cmd8(0x2A); wr_data16(x); //start column wr_data16(x+w-1);//end column wr_cmd8(0x2B); wr_data16(y); //start page wr_data16(y+h-1);//end page wr_cmd8(0x2E); //read mem, just pixelread next } void TFT::pixel(int x, int y, unsigned short color) { window(x,y,1,1); // proto->wr_gram(color); // 2C expects 16bit parameters wr_gram(color); } unsigned short TFT::pixelread(int x, int y) { unsigned short color; window4read(x,y,1,1); // proto->wr_gram(color); // 2C expects 16bit parameters color = rd_gram(); if(mipistd) color = BGR2RGB(color); // in case, convert BGR to RGB (should depend on cmd36 bit3) but maybe is device specific return color; } void TFT::cls (void) { WindowMax(); // proto->wr_gram(_background,LCDSIZE_X*LCDSIZE_Y); // proto->wr_gram(0,LCDSIZE_X*LCDSIZE_Y); wr_gram(_background,LCDSIZE_X*LCDSIZE_Y); }