eink/eink.hpp

213 lines
4.2 KiB
C++
Raw Normal View History

2022-05-26 15:11:00 +02:00
#pragma once
#include <stdint.h>
#include <avr/pgmspace.h>
#include "../clock.hpp"
#include "../io/io.hpp"
2022-05-27 11:37:23 +02:00
#include "../util/util.hpp"
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
namespace eink {
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
template <uint16_t Width, uint16_t Height, typename Spi, io::P RstPin, io::P DcPin, io::P BusyPin>
class Eink {
2022-05-28 16:14:12 +02:00
enum class Cmd : uint8_t {
DRIVER_OUTPUT_CONTROL = 0x01,
DEEP_SLEEP_MODE = 0x10,
DATA_ENTRY_MODE = 0x11,
SW_RESET = 0x12,
READ_TEMPERATURE_SENSOR = 0x18,
UPDATE_DISPLAY = 0x20,
DISPLAY_UPDATE_CONTROL_2 = 0x22,
WRITE_RAM_BLACK = 0x24,
WRITE_RAM_RED = 0x26,
BORDER_WAVEFORM_CONTROL = 0x3C,
SET_RAM_X_ADDR_POSITIONS = 0x44,
SET_RAM_Y_ADDR_POSITIONS = 0x45,
SET_RAM_X_ADDR = 0x4E,
SET_RAM_Y_ADDR = 0x4F,
};
2022-05-26 16:03:38 +02:00
static io::Pin<RstPin> m_rst;
static io::Pin<DcPin> m_dc;
static io::Pin<BusyPin> m_bsy;
2022-05-26 15:11:00 +02:00
2022-05-28 11:45:42 +02:00
static constexpr auto BLOCK_SIZE = 5;
enum class Color : uint8_t {
BLACK = 0b00,
WHITE = 0b01,
RED = 0b10,
ERROR = 0b11,
};
class ImageBlock {
public:
inline Color &operator[](const size_t idx)
{
return data[idx];
}
inline const Color &operator[](const size_t idx) const
{
return data[idx];
}
private:
Color data[BLOCK_SIZE];
};
2022-05-26 15:11:00 +02:00
public:
2022-05-26 16:03:38 +02:00
static void init()
2022-05-26 15:11:00 +02:00
{
m_rst.dir(io::Dir::OUT);
m_dc.dir(io::Dir::OUT);
m_bsy.dir(io::Dir::IN);
Spi::init();
2022-05-26 16:03:38 +02:00
reset();
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
waitUntilIdle();
sendCommand(Cmd::SW_RESET);
waitUntilIdle();
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::DRIVER_OUTPUT_CONTROL);
sendData(0xC7);
sendData(0x00);
sendData(0x01);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::DATA_ENTRY_MODE);
sendData(0x02);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::SET_RAM_X_ADDR_POSITIONS);
sendData(Width / 8 - 1);
2022-05-26 16:03:38 +02:00
sendData(0x00);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::SET_RAM_Y_ADDR_POSITIONS);
sendData(0x00);
sendData(0x00);
sendData(Height - 1);
2022-05-26 16:03:38 +02:00
sendData(0x00);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::BORDER_WAVEFORM_CONTROL);
sendData(0x05);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::READ_TEMPERATURE_SENSOR);
sendData(0x80);
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::SET_RAM_X_ADDR);
sendData(Width / 8 - 1);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::SET_RAM_Y_ADDR);
sendData(0x00);
2022-05-26 16:03:38 +02:00
sendData(0x00);
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
waitUntilIdle();
2022-05-26 15:11:00 +02:00
}
2022-05-28 16:14:12 +02:00
static void sendCommand(const Cmd command)
2022-05-26 15:11:00 +02:00
{
m_dc = false;
2022-05-28 16:14:12 +02:00
spiTransfer(static_cast<uint8_t>(command));
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
static void sendData(const uint8_t data)
2022-05-26 15:11:00 +02:00
{
m_dc = true;
2022-05-26 16:03:38 +02:00
spiTransfer(data);
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
static void waitUntilIdle()
2022-05-26 15:11:00 +02:00
{
while (m_bsy) {
_delay_ms(100);
}
}
2022-05-26 16:03:38 +02:00
static void reset()
2022-05-26 15:11:00 +02:00
{
m_rst = true;
_delay_ms(200);
2022-05-26 16:03:38 +02:00
m_rst = false;
2022-05-26 15:11:00 +02:00
_delay_ms(10);
m_rst = true;
_delay_ms(200);
}
2022-05-27 20:39:51 +02:00
template <typename Image>
static void draw(const Image &image)
2022-05-26 15:11:00 +02:00
{
2022-05-27 11:37:23 +02:00
constexpr auto lookup = [](uint8_t bits) {
2022-05-28 11:45:42 +02:00
auto block = ImageBlock{};
2022-05-27 11:37:23 +02:00
for_constexpr(
[&](const auto idx) {
block[idx.value] = static_cast<Color>(bits % 3);
bits /= 3;
},
util::make_index_sequence<BLOCK_SIZE>{});
return block;
};
constexpr auto sendImageChannel = [lookup](const auto command, const auto image) {
sendCommand(command);
auto buffer = uint8_t{0};
auto bufferPos = uint8_t{0};
for (auto i = uint16_t{0}; i < Width * Height / BLOCK_SIZE; i++) {
const auto block = lookup(pgm_read_byte(&image[i]));
for (auto p = uint8_t{0}; p < BLOCK_SIZE; ++p) {
const auto pixel = uint8_t{(command == Cmd::WRITE_RAM_BLACK) ? (block[p] != Color::BLACK)
: (block[p] == Color::RED)};
buffer |= pixel << (7 - bufferPos++);
if (bufferPos == 8) {
sendData(buffer);
buffer = 0;
bufferPos = 0;
}
}
}
};
2022-05-27 20:39:51 +02:00
sendImageChannel(Cmd::WRITE_RAM_BLACK, image.data());
sendImageChannel(Cmd::WRITE_RAM_RED, image.data());
2022-05-26 15:11:00 +02:00
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::DISPLAY_UPDATE_CONTROL_2);
sendData(0xF7);
sendCommand(Cmd::UPDATE_DISPLAY);
waitUntilIdle();
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
static void clear()
2022-05-26 15:11:00 +02:00
{
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::WRITE_RAM_BLACK);
for (auto i = uint16_t{0}; i < Width * Height / 8; i++) {
sendData(0xff);
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::WRITE_RAM_RED);
for (auto i = uint16_t{0}; i < Width * Height / 8; i++) {
sendData(0x00);
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::DISPLAY_UPDATE_CONTROL_2);
sendData(0xF7);
sendCommand(Cmd::UPDATE_DISPLAY);
waitUntilIdle();
2022-05-26 15:11:00 +02:00
}
2022-05-26 16:03:38 +02:00
static void sleep()
2022-05-26 15:11:00 +02:00
{
2022-05-26 16:03:38 +02:00
sendCommand(Cmd::DEEP_SLEEP_MODE);
sendData(0x01);
2022-05-26 15:11:00 +02:00
_delay_ms(100);
}
2022-05-26 16:03:38 +02:00
static void spiTransfer(const uint8_t data)
2022-05-26 15:11:00 +02:00
{
Spi::select(true);
Spi::transfer(data);
Spi::select(false);
}
};
2022-05-26 16:03:38 +02:00
} // namespace eink