Added driver for OLED displays sh1106 and ssd1306

This commit is contained in:
Benjamin Burkhardt 2021-12-04 16:15:39 +00:00
parent f0560c41ff
commit eba3bef12a

View File

@ -0,0 +1,249 @@
#!/usr/bin/env python3
# The MIT License (MIT)
#
# Copyright (c) 2015 Richard Hull
#
# 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.
# Example usage:
#
# from oled.device import ssd1306, sh1106
# from oled.render import canvas
# from PIL import ImageFont, ImageDraw
#
# font = ImageFont.load_default()
# device = sh1106(port=1, address=0x3C)
#
# with canvas(device) as draw:
# draw.rectangle((0, 0, device.width, device.height), outline=0, fill=0)
# draw.text(30, 40, "Hello World", font=font, fill=255)
#
# As soon as the with-block scope level is complete, the graphics primitives
# will be flushed to the device.
#
# Creating a new canvas is effectively 'carte blanche': If you want to retain
# an existing canvas, then make a reference like:
#
# c = canvas(device)
# for X in ...:
# with c as draw:
# draw.rectangle(...)
#
# As before, as soon as the with block completes, the canvas buffer is flushed
# to the device
import smbus2
from PIL import Image, ImageDraw
class device(object):
"""
Base class for OLED driver classes
"""
def __init__(self, port=1, address=0x3C, cmd_mode=0x00, data_mode=0x40):
self.cmd_mode = cmd_mode
self.data_mode = data_mode
self.bus = smbus.SMBus(port)
self.addr = address
def command(self, *cmd):
"""
Sends a command or sequence of commands through to the
device - maximum allowed is 32 bytes in one go.
"""
assert(len(cmd) <= 32)
self.bus.write_i2c_block_data(self.addr, self.cmd_mode, list(cmd))
def data(self, data):
"""
Sends a data byte or sequence of data bytes through to the
device - maximum allowed in one transaction is 32 bytes, so if
data is larger than this it is sent in chunks.
"""
for i in xrange(0, len(data), 32):
self.bus.write_i2c_block_data(self.addr,
self.data_mode,
list(data[i:i+32]))
class sh1106(device):
"""
A device encapsulates the I2C connection (address/port) to the SH1106
OLED display hardware. The init method pumps commands to the display
to properly initialize it. Further control commands can then be
called to affect the brightness. Direct use of the command() and
data() methods are discouraged.
"""
def __init__(self, port=1, address=0x3C):
super(sh1106, self).__init__(port, address)
self.width = 128
self.height = 64
self.pages = self.height / 8
self.command(
const.DISPLAYOFF,
const.MEMORYMODE,
const.SETHIGHCOLUMN, 0xB0, 0xC8,
const.SETLOWCOLUMN, 0x10, 0x40,
const.SETCONTRAST, 0x7F,
const.SETSEGMENTREMAP,
const.NORMALDISPLAY,
const.SETMULTIPLEX, 0x3F,
const.DISPLAYALLON_RESUME,
const.SETDISPLAYOFFSET, 0x00,
const.SETDISPLAYCLOCKDIV, 0xF0,
const.SETPRECHARGE, 0x22,
const.SETCOMPINS, 0x12,
const.SETVCOMDETECT, 0x20,
const.CHARGEPUMP, 0x14,
const.DISPLAYON)
def display(self, image):
"""
Takes a 1-bit image and dumps it to the SH1106 OLED display.
"""
assert(image.mode == '1')
assert(image.size[0] == self.width)
assert(image.size[1] == self.height)
page = 0xB0
pix = list(image.getdata())
step = self.width * 8
for y in xrange(0, self.pages * step, step):
# move to given page, then reset the column address
self.command(page, 0x02, 0x10)
page += 1
buf = []
for x in xrange(self.width):
byte = 0
for n in xrange(0, step, self.width):
byte |= (pix[x + y + n] & 0x01) << 8
byte >>= 1
buf.append(byte)
self.data(buf)
class ssd1306(device):
"""
A device encapsulates the I2C connection (address/port) to the SSD1306
OLED display hardware. The init method pumps commands to the display
to properly initialize it. Further control commands can then be
called to affect the brightness. Direct use of the command() and
data() methods are discouraged.
"""
def __init__(self, port=1, address=0x3C):
super(ssd1306, self).__init__(port, address)
self.width = 128
self.height = 64
self.pages = self.height / 8
self.command(
const.DISPLAYOFF,
const.SETDISPLAYCLOCKDIV, 0x80,
const.SETMULTIPLEX, 0x3F,
const.SETDISPLAYOFFSET, 0x00,
const.SETSTARTLINE,
const.CHARGEPUMP, 0x14,
const.MEMORYMODE, 0x00,
const.SEGREMAP,
const.COMSCANDEC,
const.SETCOMPINS, 0x12,
const.SETCONTRAST, 0xCF,
const.SETPRECHARGE, 0xF1,
const.SETVCOMDETECT, 0x40,
const.DISPLAYALLON_RESUME,
const.NORMALDISPLAY,
const.DISPLAYON)
def display(self, image):
"""
Takes a 1-bit image and dumps it to the SSD1306 OLED display.
"""
assert(image.mode == '1')
assert(image.size[0] == self.width)
assert(image.size[1] == self.height)
self.command(
const.COLUMNADDR, 0x00, self.width-1, # Column start/end address
const.PAGEADDR, 0x00, self.pages-1) # Page start/end address
pix = list(image.getdata())
step = self.width * 8
buf = []
for y in xrange(0, self.pages * step, step):
i = y + self.width-1
while i >= y:
byte = 0
for n in xrange(0, step, self.width):
byte |= (pix[i + n] & 0x01) << 8
byte >>= 1
buf.append(byte)
i -= 1
self.data(buf)
class canvas(object):
"""
A canvas returns a properly-sized `ImageDraw` object onto which the caller
can draw upon. As soon as the with-block completes, the resultant image is
flushed onto the device.
"""
def __init__(self, device):
self.image = Image.new('1', (device.width, device.height))
self.device = device
self.draw = ImageDraw.Draw(self.image)
class const:
CHARGEPUMP = 0x8D
COLUMNADDR = 0x21
COMSCANDEC = 0xC8
COMSCANINC = 0xC0
DISPLAYALLON = 0xA5
DISPLAYALLON_RESUME = 0xA4
DISPLAYOFF = 0xAE
DISPLAYON = 0xAF
EXTERNALVCC = 0x1
INVERTDISPLAY = 0xA7
MEMORYMODE = 0x20
NORMALDISPLAY = 0xA6
PAGEADDR = 0x22
SEGREMAP = 0xA0
SETCOMPINS = 0xDA
SETCONTRAST = 0x81
SETDISPLAYCLOCKDIV = 0xD5
SETDISPLAYOFFSET = 0xD3
SETHIGHCOLUMN = 0x10
SETLOWCOLUMN = 0x00
SETMULTIPLEX = 0xA8
SETPRECHARGE = 0xD9
SETSEGMENTREMAP = 0xA1
SETSTARTLINE = 0x40
SETVCOMDETECT = 0xDB
SWITCHCAPVCC = 0x2