194 lines
5.1 KiB
C
194 lines
5.1 KiB
C
|
/*
|
||
|
Copyright (C) 2020 Ein Terakawa
|
||
|
|
||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
you may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
*/
|
||
|
|
||
|
#include "hal.h"
|
||
|
#include "ssd1306.h"
|
||
|
#include "string.h"
|
||
|
|
||
|
/* timeout value must be increased for i2cclk less than its default, 100kHz. */
|
||
|
#define CMD_TRANSMIT_TIMEOUT TIME_MS2I(10)
|
||
|
#define DATA_TRANSMIT_TIMEOUT TIME_MS2I(100)
|
||
|
|
||
|
#define CTRL_CMD_STREAM 0x00
|
||
|
#define CTRL_DATA_STREAM 0x40
|
||
|
#define CTRL_CMD_SINGLE 0x80
|
||
|
|
||
|
#define I2CD ((ssd1306)->i2cd)
|
||
|
#define ADDR ((ssd1306)->i2caddr)
|
||
|
|
||
|
#define send_cmd(ssd1306, ...) \
|
||
|
do { \
|
||
|
uint8_t buf[] = { CTRL_CMD_STREAM, __VA_ARGS__ }; \
|
||
|
bool _success = _send_cmd(I2CD, ADDR, buf, sizeof(buf)); \
|
||
|
if (!_success) goto done; \
|
||
|
} while(0)
|
||
|
|
||
|
#define send_cmd_static(ssd1306, ...) \
|
||
|
do{ \
|
||
|
static const uint8_t buf[] = { CTRL_CMD_STREAM, __VA_ARGS__ }; \
|
||
|
bool _success = _send_cmd(I2CD, ADDR, buf, sizeof(buf)); \
|
||
|
if (!_success) goto done; \
|
||
|
} while(0)
|
||
|
|
||
|
#define CMD1(c) c
|
||
|
#define CMD2(c, d) c, d
|
||
|
#define CMD3(c, d, e) c, d, e
|
||
|
|
||
|
static bool _send_cmd(I2CDriver *i2cd, uint8_t addr, const uint8_t *buf, int len) {
|
||
|
|
||
|
msg_t status = i2cMasterTransmitTimeout(i2cd, addr,
|
||
|
buf, len, NULL, 0, CMD_TRANSMIT_TIMEOUT);
|
||
|
|
||
|
if (MSG_OK != status) {
|
||
|
/* i2cflags_t error_code = i2cGetErrors(i2cd); */
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#define send_data(ssd1306, buf, len) \
|
||
|
do { \
|
||
|
bool _success = _send_data(I2CD, ADDR, buf, len); \
|
||
|
if (!_success) goto done; \
|
||
|
} while(0)
|
||
|
|
||
|
static bool _send_data(I2CDriver *i2cd, uint8_t addr, const uint8_t *buf, int len) {
|
||
|
bool success = false;
|
||
|
|
||
|
msg_t status = i2cMasterTransmitTimeout(i2cd, addr,
|
||
|
buf, len, NULL, 0, DATA_TRANSMIT_TIMEOUT);
|
||
|
|
||
|
if (MSG_OK != status) {
|
||
|
/* i2cflags_t error_code = i2cGetErrors(I2CD); */
|
||
|
goto done;
|
||
|
}
|
||
|
|
||
|
success = true;
|
||
|
|
||
|
done:
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
bool ssd1306_init(SSD1306_DRIVER *ssd1306) {
|
||
|
bool success = false;
|
||
|
uint8_t width = ssd1306->width;
|
||
|
uint8_t height = ssd1306->height;
|
||
|
uint8_t num_pages = height / 8;
|
||
|
|
||
|
i2cStart(I2CD, ssd1306->i2ccfg);
|
||
|
|
||
|
send_cmd_static(ssd1306,
|
||
|
CMD1(SetDisplayOff),
|
||
|
CMD1(DeactivateScroll),
|
||
|
CMD1(DisableEntireDisplayOn),
|
||
|
CMD2(SetOscFreqAndClkDiv, 0x80),
|
||
|
CMD2(SetDisplayOffset, 0x00),
|
||
|
CMD1(SetDisplayStartLine | 0x00),
|
||
|
CMD2(SetMemoryAddressMode, 0x00)); /* Horizontal Addressing Mode */
|
||
|
|
||
|
send_cmd(ssd1306, CMD2(SetMultiplexRatio, height - 1));
|
||
|
|
||
|
if (ssd1306->rotate180) {
|
||
|
/* rotate 180 degrees == upside down */
|
||
|
send_cmd_static(ssd1306,
|
||
|
CMD1(SetSegmentRemapReverse),
|
||
|
CMD1(SetComScanOrderReverse));
|
||
|
} else {
|
||
|
/* no rotation */
|
||
|
send_cmd_static(ssd1306,
|
||
|
CMD1(SetSegmentRemapNormal),
|
||
|
CMD1(SetComScanOrderNormal));
|
||
|
}
|
||
|
|
||
|
if (height == 32) {
|
||
|
/* 128x32 module uses SequentialComMode */
|
||
|
send_cmd_static(ssd1306, CMD2(SetComPins, 0x02));
|
||
|
} else {
|
||
|
/* 128x64 module uses AlternativeComMode */
|
||
|
send_cmd_static(ssd1306, CMD2(SetComPins, 0x12));
|
||
|
}
|
||
|
|
||
|
/* Clear Graphic Display Data RAM */
|
||
|
send_cmd(ssd1306,
|
||
|
CMD3(SetPageAddress, 0, num_pages - 1),
|
||
|
CMD3(SetColumnAddress, 0, width - 1));
|
||
|
|
||
|
uint8_t *buf = ssd1306->buf;
|
||
|
size_t len = SSD1306_PREAMBLE_LENGTH + num_pages * width;
|
||
|
memset(buf, 0, len);
|
||
|
buf[0] = CTRL_DATA_STREAM; /* need this byte proceeding the actual data */
|
||
|
send_data(ssd1306, buf, len);
|
||
|
|
||
|
send_cmd_static(ssd1306,
|
||
|
CMD2(SetPreChargePeriod, 0xC4),
|
||
|
CMD2(SetVcomhLevel, 0x20),
|
||
|
CMD1(SetNormalDisplay),
|
||
|
CMD2(SetContrastControl, 0x3F),
|
||
|
CMD2(ChargePumpSetting, 0x14),
|
||
|
CMD1(SetDisplayOn));
|
||
|
|
||
|
success = true;
|
||
|
|
||
|
done:
|
||
|
i2cStop(I2CD);
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
bool ssd1306_data(SSD1306_DRIVER *ssd1306) {
|
||
|
bool success = false;
|
||
|
uint8_t width = ssd1306->width;
|
||
|
uint8_t height = ssd1306->height;
|
||
|
uint8_t num_pages = height / 8;
|
||
|
|
||
|
i2cStart(I2CD, ssd1306->i2ccfg);
|
||
|
|
||
|
/* Transfer to Graphic Display Data RAM */
|
||
|
send_cmd(ssd1306,
|
||
|
CMD3(SetPageAddress, 0, num_pages - 1),
|
||
|
CMD3(SetColumnAddress, 0, width - 1));
|
||
|
|
||
|
uint8_t *buf = ssd1306->buf;
|
||
|
size_t len = SSD1306_PREAMBLE_LENGTH + num_pages * width;
|
||
|
send_data(ssd1306, buf, len);
|
||
|
|
||
|
success = true;
|
||
|
|
||
|
done:
|
||
|
i2cStop(I2CD);
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
bool ssd1306_display_on(SSD1306_DRIVER *ssd1306) {
|
||
|
bool success = false;
|
||
|
i2cStart(I2CD, ssd1306->i2ccfg);
|
||
|
send_cmd_static(ssd1306, CMD1(SetDisplayOn));
|
||
|
success = true;
|
||
|
done:
|
||
|
i2cStop(I2CD);
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
bool ssd1306_display_off(SSD1306_DRIVER *ssd1306) {
|
||
|
bool success = false;
|
||
|
i2cStart(I2CD, ssd1306->i2ccfg);
|
||
|
send_cmd_static(ssd1306, CMD1(SetDisplayOff));
|
||
|
success = true;
|
||
|
done:
|
||
|
i2cStop(I2CD);
|
||
|
return success;
|
||
|
}
|