Skip to content

Commit 3090292

Browse files
committed
start of ssd1681 driver
1 parent 432dacf commit 3090292

File tree

1 file changed

+177
-0
lines changed

1 file changed

+177
-0
lines changed

adafruit_epd/ssd1681.py

Lines changed: 177 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,177 @@
1+
# The MIT License (MIT)
2+
#
3+
# Copyright (c) 2018 Dean Miller for Adafruit Industries
4+
#
5+
# Permission is hereby granted, free of charge, to any person obtaining a copy
6+
# of this software and associated documentation files (the "Software"), to deal
7+
# in the Software without restriction, including without limitation the rights
8+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
# copies of the Software, and to permit persons to whom the Software is
10+
# furnished to do so, subject to the following conditions:
11+
#
12+
# The above copyright notice and this permission notice shall be included in
13+
# all copies or substantial portions of the Software.
14+
#
15+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21+
# THE SOFTWARE.
22+
"""
23+
`adafruit_epd.ssd1681` - Adafruit SSD1681 - ePaper display driver
24+
====================================================================================
25+
CircuitPython driver for Adafruit SSD1681 display breakouts
26+
* Author(s): Dean Miller, Ladyada
27+
"""
28+
29+
import time
30+
from micropython import const
31+
import adafruit_framebuf
32+
from adafruit_epd.epd import Adafruit_EPD
33+
34+
__version__ = "0.0.0-auto.0"
35+
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_EPD.git"
36+
37+
_SSD1681_DRIVER_CONTROL = const(0x01)
38+
_SSD1681_GATE_VOLTAGE = const(0x03)
39+
_SSD1681_SOURCE_VOLTAGE = const(0x04)
40+
_SSD1681_INIT_SETTING = const(0x08)
41+
_SSD1681_INIT_WRITE_REG = const(0x09)
42+
_SSD1681_INIT_READ_REG = const(0x0A)
43+
_SSD1681_BOOSTER_SOFT_START = const(0x0C)
44+
_SSD1681_DEEP_SLEEP = const(0x10)
45+
_SSD1681_DATA_MODE = const(0x11)
46+
_SSD1681_SW_RESET = const(0x12)
47+
_SSD1681_HV_DETECT = const(0x14)
48+
_SSD1681_VCI_DETECT = const(0x15)
49+
_SSD1681_TEMP_CONTROL = const(0x18)
50+
_SSD1681_TEMP_WRITE = const(0x1A)
51+
_SSD1681_TEMP_READ = const(0x1B)
52+
_SSD1681_EXTTEMP_WRITE = const(0x1C)
53+
_SSD1681_MASTER_ACTIVATE = const(0x20)
54+
_SSD1681_DISP_CTRL1 = const(0x21)
55+
_SSD1681_DISP_CTRL2 = const(0x22)
56+
_SSD1681_WRITE_BWRAM = const(0x24)
57+
_SSD1681_WRITE_REDRAM = const(0x26)
58+
_SSD1681_READ_RAM = const(0x27)
59+
_SSD1681_VCOM_SENSE = const(0x28)
60+
_SSD1681_VCOM_DURATION = const(0x29)
61+
_SSD1681_WRITE_VCOM_OTP = const(0x2A)
62+
_SSD1681_WRITE_VCOM_CTRL = const(0x2B)
63+
_SSD1681_WRITE_VCOM_REG = const(0x2C)
64+
_SSD1681_READ_OTP = const(0x2D)
65+
_SSD1681_READ_USERID = const(0x2E)
66+
_SSD1681_READ_STATUS = const(0x2F)
67+
_SSD1681_WRITE_WS_OTP = const(0x30)
68+
_SSD1681_LOAD_WS_OTP = const(0x31)
69+
_SSD1681_WRITE_LUT = const(0x32)
70+
_SSD1681_CRC_CALC = const(0x34)
71+
_SSD1681_CRC_READ = const(0x35)
72+
_SSD1681_PROG_OTP = const(0x36)
73+
_SSD1681_WRITE_DISPLAY_OPT = const(0x37)
74+
_SSD1681_WRITE_USERID = const(0x38)
75+
_SSD1681_OTP_PROGMODE = const(0x39)
76+
_SSD1681_WRITE_BORDER = const(0x3C)
77+
_SSD1681_END_OPTION = const(0x3F)
78+
_SSD1681_READ_RAM = const(0x41)
79+
_SSD1681_SET_RAMXPOS = const(0x44)
80+
_SSD1681_SET_RAMYPOS = const(0x45)
81+
_SSD1681_AUTOWRITE_RED = const(0x46)
82+
_SSD1681_AUTOWRITE_BW = const(0x47)
83+
_SSD1681_SET_RAMXCOUNT = const(0x4E)
84+
_SSD1681_SET_RAMYCOUNT = const(0x4F)
85+
_SSD1681_NOP = const(0xFF)
86+
87+
_LUT_DATA = b'\x02\x02\x01\x11\x12\x12""fiiYX\x99\x99\x88\x00\x00\x00\x00\xf8\xb4\x13Q5QQ\x19\x01\x00' # pylint: disable=line-too-long
88+
89+
class Adafruit_SSD1681(Adafruit_EPD):
90+
"""driver class for Adafruit SSD1681 ePaper display breakouts"""
91+
# pylint: disable=too-many-arguments
92+
def __init__(self, width, height, spi, *, cs_pin, dc_pin, sramcs_pin, rst_pin, busy_pin):
93+
super(Adafruit_SSD1681, self).__init__(width, height, spi, cs_pin, dc_pin,
94+
sramcs_pin, rst_pin, busy_pin)
95+
96+
if height % 8 != 0:
97+
height += (8 - height % 8)
98+
self._height = height
99+
100+
self._buffer1_size = int(width * height / 8)
101+
102+
if sramcs_pin:
103+
self._buffer1 = self.sram.get_view(0)
104+
else:
105+
self._buffer1 = bytearray((width * height) // 8)
106+
self._framebuf1 = adafruit_framebuf.FrameBuffer(self._buffer1, width, height,
107+
buf_format=adafruit_framebuf.MHMSB)
108+
self.set_black_buffer(0, True)
109+
self.set_color_buffer(0, True)
110+
# pylint: enable=too-many-arguments
111+
112+
def begin(self, reset=True):
113+
"""Begin communication with the display and set basic settings"""
114+
if reset:
115+
self.hardware_reset()
116+
self.power_down()
117+
118+
def busy_wait(self):
119+
"""Wait for display to be done with current task, either by polling the
120+
busy pin, or pausing"""
121+
if self._busy:
122+
while self._busy.value:
123+
time.sleep(0.01)
124+
else:
125+
time.sleep(0.5)
126+
127+
def power_up(self):
128+
"""Power up the display in preparation for writing RAM and updating"""
129+
self.hardware_reset()
130+
self.busy_wait()
131+
self.command(_SSD1681_SW_RESET)
132+
self.busy_wait()
133+
# driver output control
134+
self.command(_SSD1681_DRIVER_CONTROL,
135+
bytearray([self._width-1, (self._width-1) >> 8, 0x00]))
136+
# data entry mode
137+
self.command(_SSD1681_DATA_MODE, bytearray([0x01]))
138+
# Set ram X start/end postion
139+
self.command(_SSD1681_SET_RAMXPOS, bytearray([0x00, self._height//8 - 1]))
140+
# Set ram Y start/end postion
141+
self.command(_SSD1681_SET_RAMYPOS,
142+
bytearray([0, 0, self._height - 1, (self._height - 1) >> 8]))
143+
# Set border waveform
144+
self.command(_SSD1681_WRITE_BORDER, bytearray([0x05]))
145+
# Set temperature control
146+
self.command(_SSD1681_TEMP_CONTROL, bytearray([0x80]))
147+
148+
self.busy_wait()
149+
150+
def power_down(self):
151+
"""Power down the display - required when not actively displaying!"""
152+
self.command(_SSD1681_DEEP_SLEEP, bytearray([0x01]))
153+
time.sleep(0.1)
154+
155+
def update(self):
156+
"""Update the display from internal memory"""
157+
self.command(_SSD1681_DISP_CTRL2, bytearray([0xF7]))
158+
self.command(_SSD1681_MASTER_ACTIVATE)
159+
self.busy_wait()
160+
if not self._busy:
161+
time.sleep(3) # wait 3 seconds
162+
163+
def write_ram(self, index):
164+
"""Send the one byte command for starting the RAM write process. Returns
165+
the byte read at the same time over SPI. index is the RAM buffer, can be
166+
0 or 1 for tri-color displays."""
167+
if index == 0:
168+
return self.command(_SSD1681_WRITE_BWRAM, end=False)
169+
raise RuntimeError("RAM index must be 0")
170+
171+
def set_ram_address(self, x, y): # pylint: disable=unused-argument, no-self-use
172+
"""Set the RAM address location, not used on this chipset but required by
173+
the superclass"""
174+
# Set RAM X address counter
175+
self.command(_SSD1681_SET_RAMXCOUNT, bytearray([x]))
176+
# Set RAM Y address counter
177+
self.command(_SSD1681_SET_RAMYCOUNT, bytearray([y>>8, y]))

0 commit comments

Comments
 (0)