# Support for ST7920 (128x64 graphics) LCD displays # # Copyright (C) 2018 Kevin O'Connor <kevin@koconnor.net> # # This file may be distributed under the terms of the GNU GPLv3 license. import logging from .. import bus from . import font8x14 BACKGROUND_PRIORITY_CLOCK = 0x7fffffff00000000 # Spec says 72us, but faster is possible in practice ST7920_CMD_DELAY = .000020 ST7920_SYNC_DELAY = .000045 TextGlyphs = { 'right_arrow': '\x1a' } CharGlyphs = { 'degrees': bytearray(font8x14.VGA_FONT[0xf8]) } class DisplayBase: def __init__(self): # framebuffers self.text_framebuffer = bytearray(' '*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('~'*64), 0x80), # Glyph framebuffer (self.glyph_framebuffer, bytearray('~'*128), 0x40), # Graphics framebuffers ] + [(self.graphics_framebuffers[i], bytearray('~'*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] # Batch together changes that are close to each other for i in range(len(diffs)-2, -1, -1): pos, count = diffs[i] nextpos, nextcount = diffs[i+1] if pos + 5 >= nextpos and nextcount < 16: diffs[i][1] = nextcount + (nextpos - pos) del diffs[i+1] # Transmit changes for pos, count in diffs: count += pos & 0x01 count += count & 0x01 pos = pos & ~0x01 chip_pos = pos >> 1 if fb_id < 0x40: # Graphics framebuffer update 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) 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 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) if icon is None or base_icon is None: return 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)) def set_glyphs(self, glyphs): for glyph_name, glyph_data in glyphs.items(): 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) def write_text(self, x, y, data): if x + len(data) > 16: data = data[:16 - min(x, 16)] pos = [0, 32, 16, 48][y] + x 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 gfx_fb = y * 16 if gfx_fb >= 32: gfx_fb -= 32 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: # Render cached icon using character generator glyph_name = glyph_id[0] self.write_text(x, y, glyph_id[1]) icon = self.icons.get(glyph_name) if icon is not None: # Draw icon in graphics mode self.write_graphics(x, y, icon[0]) self.write_graphics(x + 1, y, icon[1]) return 2 char = TextGlyphs.get(glyph_name) if char is not None: # Draw character self.write_text(x, y, char) return 1 font = CharGlyphs.get(glyph_name) if font is not None: # Draw single width character self.write_graphics(x, y, font) return 1 return 0 def clear(self): self.text_framebuffer[:] = ' '*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']] mcu = None for pin_params in pins: 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] # prepare send functions self.mcu = mcu self.oid = self.mcu.create_oid() self.mcu.register_config_callback(self.build_config) self.send_data_cmd = self.send_cmds_cmd = None 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], self.mcu.seconds_to_clock(ST7920_SYNC_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) self.send_data_cmd = self.mcu.lookup_command( "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: cmd_type = self.send_data_cmd elif self.is_extended != is_extended: add_cmd = 0x22 if is_extended: add_cmd = 0x26 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)) # 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()) 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) self.en_pin.update_digital_out(0, minclock=minclock) minclock = mcu.print_time_to_clock(print_time + .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) 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 # as the MISO pin. 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] 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) 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) self.en_set = False # 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 if not is_data: sync_byte = 0xf8 if self.is_extended != is_extended: add_cmd = 0x22 if is_extended: add_cmd = 0x26 cmds = [add_cmd] + cmds self.is_extended = is_extended # copy data to ST7920 data format spi_data = [0] * (2 * len(cmds) + 1) spi_data[0] = sync_byte i = 1 for b in cmds: spi_data[i] = b & 0xF0 spi_data[i + 1] = (b & 0x0F) << 4 i = i + 2 # check if enable pin has been set if not self.en_set: self.en_helper.init() self.en_set = True # send data self.spi.spi_send(spi_data, reqclock=BACKGROUND_PRIORITY_CLOCK) #logging.debug("st7920 %s", repr(spi_data))