aboutsummaryrefslogtreecommitdiffstats
path: root/klippy/extras/display/st7920.py
diff options
context:
space:
mode:
Diffstat (limited to 'klippy/extras/display/st7920.py')
-rw-r--r--klippy/extras/display/st7920.py165
1 files changed, 100 insertions, 65 deletions
diff --git a/klippy/extras/display/st7920.py b/klippy/extras/display/st7920.py
index 5b2a943c..9411a064 100644
--- a/klippy/extras/display/st7920.py
+++ b/klippy/extras/display/st7920.py
@@ -7,46 +7,50 @@ import logging
from .. import bus
from . import font8x14
-BACKGROUND_PRIORITY_CLOCK = 0x7fffffff00000000
+BACKGROUND_PRIORITY_CLOCK = 0x7FFFFFFF00000000
# Spec says 72us, but faster is possible in practice
-ST7920_CMD_DELAY = .000020
-ST7920_SYNC_DELAY = .000045
+ST7920_CMD_DELAY = 0.000020
+ST7920_SYNC_DELAY = 0.000045
+
+TextGlyphs = {"right_arrow": b"\x1a"}
+CharGlyphs = {"degrees": bytearray(font8x14.VGA_FONT[0xF8])}
-TextGlyphs = { 'right_arrow': b'\x1a' }
-CharGlyphs = { 'degrees': bytearray(font8x14.VGA_FONT[0xf8]) }
class DisplayBase:
def __init__(self):
# framebuffers
- self.text_framebuffer = bytearray(b' '*64)
+ self.text_framebuffer = bytearray(b" " * 64)
self.glyph_framebuffer = bytearray(128)
self.graphics_framebuffers = [bytearray(32) for i in range(32)]
self.all_framebuffers = [
# Text framebuffer
- (self.text_framebuffer, bytearray(b'~'*64), 0x80),
+ (self.text_framebuffer, bytearray(b"~" * 64), 0x80),
# Glyph framebuffer
- (self.glyph_framebuffer, bytearray(b'~'*128), 0x40),
+ (self.glyph_framebuffer, bytearray(b"~" * 128), 0x40),
# Graphics framebuffers
- ] + [(self.graphics_framebuffers[i], bytearray(b'~'*32), i)
- for i in range(32)]
+ ] + [
+ (self.graphics_framebuffers[i], bytearray(b"~" * 32), i) for i in range(32)
+ ]
self.cached_glyphs = {}
self.icons = {}
+
def flush(self):
# Find all differences in the framebuffers and send them to the chip
for new_data, old_data, fb_id in self.all_framebuffers:
if new_data == old_data:
continue
# Find the position of all changed bytes in this framebuffer
- diffs = [[i, 1] for i, (n, o) in enumerate(zip(new_data, old_data))
- if n != o]
+ diffs = [
+ [i, 1] for i, (n, o) in enumerate(zip(new_data, old_data)) if n != o
+ ]
# Batch together changes that are close to each other
- for i in range(len(diffs)-2, -1, -1):
+ for i in range(len(diffs) - 2, -1, -1):
pos, count = diffs[i]
- nextpos, nextcount = diffs[i+1]
+ nextpos, nextcount = diffs[i + 1]
if pos + 5 >= nextpos and nextcount < 16:
diffs[i][1] = nextcount + (nextpos - pos)
- del diffs[i+1]
+ del diffs[i + 1]
# Transmit changes
for pos, count in diffs:
count += pos & 0x01
@@ -58,19 +62,23 @@ class DisplayBase:
self.send([0x80 + fb_id, 0x80 + chip_pos], is_extended=True)
else:
self.send([fb_id + chip_pos])
- self.send(new_data[pos:pos+count], is_data=True)
+ self.send(new_data[pos : pos + count], is_data=True)
old_data[:] = new_data
+
def init(self):
- cmds = [0x24, # Enter extended mode
- 0x40, # Clear vertical scroll address
- 0x02, # Enable CGRAM access
- 0x26, # Enable graphics
- 0x22, # Leave extended mode
- 0x02, # Home the display
- 0x06, # Set positive update direction
- 0x0c] # Enable display and hide cursor
+ cmds = [
+ 0x24, # Enter extended mode
+ 0x40, # Clear vertical scroll address
+ 0x02, # Enable CGRAM access
+ 0x26, # Enable graphics
+ 0x22, # Leave extended mode
+ 0x02, # Home the display
+ 0x06, # Set positive update direction
+ 0x0C,
+ ] # Enable display and hide cursor
self.send(cmds)
self.flush()
+
def cache_glyph(self, glyph_name, base_glyph_name, glyph_id):
icon = self.icons.get(glyph_name)
base_icon = self.icons.get(base_glyph_name)
@@ -79,23 +87,26 @@ class DisplayBase:
all_bits = zip(icon[0], icon[1], base_icon[0], base_icon[1])
for i, (ic1, ic2, b1, b2) in enumerate(all_bits):
x1, x2 = ic1 ^ b1, ic2 ^ b2
- pos = glyph_id*32 + i*2
- self.glyph_framebuffer[pos:pos+2] = [x1, x2]
- self.all_framebuffers[1][1][pos:pos+2] = [x1 ^ 1, x2 ^ 1]
- self.cached_glyphs[glyph_name] = (base_glyph_name, (0, glyph_id*2))
+ pos = glyph_id * 32 + i * 2
+ self.glyph_framebuffer[pos : pos + 2] = [x1, x2]
+ self.all_framebuffers[1][1][pos : pos + 2] = [x1 ^ 1, x2 ^ 1]
+ self.cached_glyphs[glyph_name] = (base_glyph_name, (0, glyph_id * 2))
+
def set_glyphs(self, glyphs):
for glyph_name, glyph_data in glyphs.items():
- icon = glyph_data.get('icon16x16')
+ icon = glyph_data.get("icon16x16")
if icon is not None:
self.icons[glyph_name] = icon
# Setup animated glyphs
- self.cache_glyph('fan2', 'fan1', 0)
- self.cache_glyph('bed_heat2', 'bed_heat1', 1)
+ self.cache_glyph("fan2", "fan1", 0)
+ self.cache_glyph("bed_heat2", "bed_heat1", 1)
+
def write_text(self, x, y, data):
if x + len(data) > 16:
- data = data[:16 - min(x, 16)]
+ data = data[: 16 - min(x, 16)]
pos = [0, 32, 16, 48][y] + x
- self.text_framebuffer[pos:pos+len(data)] = data
+ self.text_framebuffer[pos : pos + len(data)] = data
+
def write_graphics(self, x, y, data):
if x >= 16 or y >= 4 or len(data) != 16:
return
@@ -105,6 +116,7 @@ class DisplayBase:
x += 16
for i, bits in enumerate(data):
self.graphics_framebuffers[gfx_fb + i][x] = bits
+
def write_glyph(self, x, y, glyph_name):
glyph_id = self.cached_glyphs.get(glyph_name)
if glyph_id is not None and x & 1 == 0:
@@ -128,28 +140,33 @@ class DisplayBase:
self.write_graphics(x, y, font)
return 1
return 0
+
def clear(self):
- self.text_framebuffer[:] = b' '*64
+ self.text_framebuffer[:] = b" " * 64
zeros = bytearray(32)
for gfb in self.graphics_framebuffers:
gfb[:] = zeros
+
def get_dimensions(self):
return (16, 4)
+
# Display driver for stock ST7920 displays
class ST7920(DisplayBase):
def __init__(self, config):
printer = config.get_printer()
# pin config
- ppins = printer.lookup_object('pins')
- pins = [ppins.lookup_pin(config.get(name + '_pin'))
- for name in ['cs', 'sclk', 'sid']]
+ ppins = printer.lookup_object("pins")
+ pins = [
+ ppins.lookup_pin(config.get(name + "_pin"))
+ for name in ["cs", "sclk", "sid"]
+ ]
mcu = None
for pin_params in pins:
- if mcu is not None and pin_params['chip'] != mcu:
+ if mcu is not None and pin_params["chip"] != mcu:
raise ppins.error("st7920 all pins must be on same mcu")
- mcu = pin_params['chip']
- self.pins = [pin_params['pin'] for pin_params in pins]
+ mcu = pin_params["chip"]
+ self.pins = [pin_params["pin"] for pin_params in pins]
# prepare send functions
self.mcu = mcu
self.oid = self.mcu.create_oid()
@@ -158,19 +175,29 @@ class ST7920(DisplayBase):
self.is_extended = False
# init display base
DisplayBase.__init__(self)
+
def build_config(self):
# configure send functions
self.mcu.add_config_cmd(
"config_st7920 oid=%u cs_pin=%s sclk_pin=%s sid_pin=%s"
- " sync_delay_ticks=%d cmd_delay_ticks=%d" % (
- self.oid, self.pins[0], self.pins[1], self.pins[2],
+ " sync_delay_ticks=%d cmd_delay_ticks=%d"
+ % (
+ self.oid,
+ self.pins[0],
+ self.pins[1],
+ self.pins[2],
self.mcu.seconds_to_clock(ST7920_SYNC_DELAY),
- self.mcu.seconds_to_clock(ST7920_CMD_DELAY)))
+ self.mcu.seconds_to_clock(ST7920_CMD_DELAY),
+ )
+ )
cmd_queue = self.mcu.alloc_command_queue()
self.send_cmds_cmd = self.mcu.lookup_command(
- "st7920_send_cmds oid=%c cmds=%*s", cq=cmd_queue)
+ "st7920_send_cmds oid=%c cmds=%*s", cq=cmd_queue
+ )
self.send_data_cmd = self.mcu.lookup_command(
- "st7920_send_data oid=%c data=%*s", cq=cmd_queue)
+ "st7920_send_data oid=%c data=%*s", cq=cmd_queue
+ )
+
def send(self, cmds, is_data=False, is_extended=False):
cmd_type = self.send_cmds_cmd
if is_data:
@@ -182,26 +209,30 @@ class ST7920(DisplayBase):
cmds = [add_cmd] + cmds
self.is_extended = is_extended
cmd_type.send([self.oid, cmds], reqclock=BACKGROUND_PRIORITY_CLOCK)
- #logging.debug("st7920 %d %s", is_data, repr(cmds))
+ # logging.debug("st7920 %d %s", is_data, repr(cmds))
+
# Helper code for toggling the en pin on startup
class EnableHelper:
def __init__(self, pin_desc, spi):
- self.en_pin = bus.MCU_bus_digital_out(spi.get_mcu(), pin_desc,
- spi.get_command_queue())
+ self.en_pin = bus.MCU_bus_digital_out(
+ spi.get_mcu(), pin_desc, spi.get_command_queue()
+ )
+
def init(self):
mcu = self.en_pin.get_mcu()
curtime = mcu.get_printer().get_reactor().monotonic()
print_time = mcu.estimated_print_time(curtime)
# Toggle enable pin
- minclock = mcu.print_time_to_clock(print_time + .100)
+ minclock = mcu.print_time_to_clock(print_time + 0.100)
self.en_pin.update_digital_out(0, minclock=minclock)
- minclock = mcu.print_time_to_clock(print_time + .200)
+ minclock = mcu.print_time_to_clock(print_time + 0.200)
self.en_pin.update_digital_out(1, minclock=minclock)
# Force a delay to any subsequent commands on the command queue
- minclock = mcu.print_time_to_clock(print_time + .300)
+ minclock = mcu.print_time_to_clock(print_time + 0.300)
self.en_pin.update_digital_out(1, minclock=minclock)
+
# Display driver for displays that emulate the ST7920 in software.
# These displays rely on the CS pin to be toggled in order to initialize the
# SPI correctly. This display driver uses a software SPI with an unused pin
@@ -209,19 +240,22 @@ class EnableHelper:
class EmulatedST7920(DisplayBase):
def __init__(self, config):
# create software spi
- ppins = config.get_printer().lookup_object('pins')
- sw_pin_names = ['spi_software_%s_pin' % (name,)
- for name in ['miso', 'mosi', 'sclk']]
- sw_pin_params = [ppins.lookup_pin(config.get(name), share_type=name)
- for name in sw_pin_names]
+ ppins = config.get_printer().lookup_object("pins")
+ sw_pin_names = [
+ "spi_software_%s_pin" % (name,) for name in ["miso", "mosi", "sclk"]
+ ]
+ sw_pin_params = [
+ ppins.lookup_pin(config.get(name), share_type=name) for name in sw_pin_names
+ ]
mcu = None
for pin_params in sw_pin_params:
- if mcu is not None and pin_params['chip'] != mcu:
- raise ppins.error("%s: spi pins must be on same mcu" % (
- config.get_name(),))
- mcu = pin_params['chip']
- sw_pins = tuple([pin_params['pin'] for pin_params in sw_pin_params])
- speed = config.getint('spi_speed', 1000000, minval=100000)
+ if mcu is not None and pin_params["chip"] != mcu:
+ raise ppins.error(
+ "%s: spi pins must be on same mcu" % (config.get_name(),)
+ )
+ mcu = pin_params["chip"]
+ sw_pins = tuple([pin_params["pin"] for pin_params in sw_pin_params])
+ speed = config.getint("spi_speed", 1000000, minval=100000)
self.spi = bus.MCU_SPI(mcu, None, None, 0, speed, sw_pins)
# create enable helper
self.en_helper = EnableHelper(config.get("en_pin"), self.spi)
@@ -229,11 +263,12 @@ class EmulatedST7920(DisplayBase):
# init display base
self.is_extended = False
DisplayBase.__init__(self)
+
def send(self, cmds, is_data=False, is_extended=False):
# setup sync byte and check for exten mode switch
- sync_byte = 0xfa
+ sync_byte = 0xFA
if not is_data:
- sync_byte = 0xf8
+ sync_byte = 0xF8
if self.is_extended != is_extended:
add_cmd = 0x22
if is_extended:
@@ -254,4 +289,4 @@ class EmulatedST7920(DisplayBase):
self.en_set = True
# send data
self.spi.spi_send(spi_data, reqclock=BACKGROUND_PRIORITY_CLOCK)
- #logging.debug("st7920 %s", repr(spi_data))
+ # logging.debug("st7920 %s", repr(spi_data))