/** * The MIT License (MIT) * * Copyright (c) 2019 by Helmut Tschemernjak - www.radioshuttle.de * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation 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 * furnished 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 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. * * ThingPulse invests considerable time and money to develop these open source libraries. * Please support us by buying our products (and not the clones) from * https://thingpulse.com * */ #ifndef SSD1306I2C_h #define SSD1306I2C_h #ifdef __MBED__ #include "OLEDDisplay.h" #include #ifndef UINT8_MAX #define UINT8_MAX 0xff #endif class SSD1306I2C : public OLEDDisplay { public: SSD1306I2C(uint8_t _address, PinName _sda, PinName _scl, OLEDDISPLAY_GEOMETRY g = GEOMETRY_128_64) { setGeometry(g); this->_address = _address << 1; // convert from 7 to 8 bit for mbed. this->_sda = _sda; this->_scl = _scl; _i2c = new I2C(_sda, _scl); } bool connect() { // mbed supports 100k and 400k some device maybe 1000k #ifdef TARGET_STM32L4 _i2c->frequency(1000000); #else _i2c->frequency(400000); #endif return true; } void display(void) { const int x_offset = (128 - this->width()) / 2; #ifdef OLEDDISPLAY_DOUBLE_BUFFER uint8_t minBoundY = UINT8_MAX; uint8_t maxBoundY = 0; uint8_t minBoundX = UINT8_MAX; uint8_t maxBoundX = 0; uint8_t x, y; // Calculate the Y bounding box of changes // and copy buffer[pos] to buffer_back[pos]; for (y = 0; y < (this->height() / 8); y++) { for (x = 0; x < this->width(); x++) { uint16_t pos = x + y * this->width(); if (buffer[pos] != buffer_back[pos]) { minBoundY = std::min(minBoundY, y); maxBoundY = std::max(maxBoundY, y); minBoundX = std::min(minBoundX, x); maxBoundX = std::max(maxBoundX, x); } buffer_back[pos] = buffer[pos]; } yield(); } // If the minBoundY wasn't updated // we can savely assume that buffer_back[pos] == buffer[pos] // holdes true for all values of pos if (minBoundY == UINT8_MAX) return; sendCommand(COLUMNADDR); sendCommand(x_offset + minBoundX); // column start address (0 = reset) sendCommand(x_offset + maxBoundX); // column end address (127 = reset) sendCommand(PAGEADDR); sendCommand(minBoundY); // page start address sendCommand(maxBoundY); // page end address for (y = minBoundY; y <= maxBoundY; y++) { uint8_t *start = &buffer[(minBoundX + y * this->width())-1]; uint8_t save = *start; *start = 0x40; // control _i2c->write(_address, (char *)start, (maxBoundX-minBoundX) + 1 + 1); *start = save; } #else sendCommand(COLUMNADDR); sendCommand(x_offset); // column start address (0 = reset) sendCommand(x_offset + (this->width() - 1));// column end address (127 = reset) sendCommand(PAGEADDR); sendCommand(0x0); // page start address (0 = reset) if (geometry == GEOMETRY_128_64) { sendCommand(0x7); } else if (geometry == GEOMETRY_128_32) { sendCommand(0x3); } buffer[-1] = 0x40; // control _i2c->write(_address, (char *)&buffer[-1], displayBufferSize + 1); #endif } private: int getBufferOffset(void) { return 0; } inline void sendCommand(uint8_t command) __attribute__((always_inline)) { char _data[2]; _data[0] = 0x80; // control _data[1] = command; _i2c->write(_address, _data, sizeof(_data)); } uint8_t _address; PinName _sda; PinName _scl; I2C *_i2c; }; #endif #endif