uc1701: Add all_framebuffers to reduce vram indexing in the code

Add a self.all_framebuffers variable and change self.vram to only
contain the new display data.  This allows the main code to access the
self.vram member variable without the need to index for current/old
data.  (It also updates the code to match the latest st7920 and
hd44780 code.)

Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
Kevin O'Connor 2018-11-21 12:43:16 -05:00
parent e231ac7422
commit a79d87ca8e
1 changed files with 16 additions and 17 deletions

View File

@ -13,7 +13,6 @@ TextGlyphs = { 'right_arrow': '\x1a', 'degrees': '\xf8' }
class UC1701: class UC1701:
DATA_PIN_NAME = "a0_pin" DATA_PIN_NAME = "a0_pin"
CURRENT_BUF, OLD_BUF = 0, 1
EMPTY_CHAR = (0, 32, 255) EMPTY_CHAR = (0, 32, 255)
def __init__(self, config): def __init__(self, config):
self.spi = extras.bus.MCU_SPI_from_config(config, 0, self.spi = extras.bus.MCU_SPI_from_config(config, 0,
@ -31,8 +30,9 @@ class UC1701:
mcu.register_config_callback(self.build_config) mcu.register_config_callback(self.build_config)
self.update_pin_cmd = None self.update_pin_cmd = None
# framebuffers # framebuffers
self.vram = ([bytearray(128) for i in range(8)], self.vram = [bytearray(128) for i in range(8)]
[bytearray('~'*128) for i in range(8)]) self.all_framebuffers = [(self.vram[i], bytearray('~'*128), i)
for i in range(8)]
def build_config(self): def build_config(self):
self.update_pin_cmd = self.spi.get_mcu().lookup_command( self.update_pin_cmd = self.spi.get_mcu().lookup_command(
"update_digital_out oid=%c value=%c", "update_digital_out oid=%c value=%c",
@ -67,14 +67,13 @@ class UC1701:
self.flush() self.flush()
logging.info("uc1701 initialized") logging.info("uc1701 initialized")
def flush(self): def flush(self):
new_data = self.vram[self.CURRENT_BUF] # Find all differences in the framebuffers and send them to the chip
old_data = self.vram[self.OLD_BUF] for new_data, old_data, page in self.all_framebuffers:
for page in range(8): if new_data == old_data:
if new_data[page] == old_data[page]:
continue continue
# Find the position of all changed bytes in this framebuffer # Find the position of all changed bytes in this framebuffer
diffs = [[i, 1] for i, (nd, od) in enumerate(zip(new_data[page], old_data[page])) diffs = [[i, 1] for i, (n, o) in enumerate(zip(new_data, old_data))
if nd != od] if n != o]
# Batch together changes that are close to each other # 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] pos, count = diffs[i]
@ -82,7 +81,7 @@ class UC1701:
if pos + 5 >= nextpos and nextcount < 16: if pos + 5 >= nextpos and nextcount < 16:
diffs[i][1] = nextcount + (nextpos - pos) diffs[i][1] = nextcount + (nextpos - pos)
del diffs[i+1] del diffs[i+1]
# Transmit # Transmit changes
for col_pos, count in diffs: for col_pos, count in diffs:
# Set Position registers # Set Position registers
ra = 0xb0 | (page & 0x0F) ra = 0xb0 | (page & 0x0F)
@ -90,22 +89,22 @@ class UC1701:
ca_lsb = col_pos & 0x0F ca_lsb = col_pos & 0x0F
self.send([ra, ca_msb, ca_lsb]) self.send([ra, ca_msb, ca_lsb])
# Send Data # Send Data
self.send(new_data[page][col_pos:col_pos+count], is_data=True) self.send(new_data[col_pos:col_pos+count], is_data=True)
old_data[page][:] = new_data[page] old_data[:] = new_data
def set_pixel(self, pix_x, pix_y, exclusive=True): def set_pixel(self, pix_x, pix_y, exclusive=True):
page_idx = pix_y // 8 page_idx = pix_y // 8
page_byte = 0x01 << (pix_y % 8) page_byte = 0x01 << (pix_y % 8)
if exclusive and self.vram[self.CURRENT_BUF][page_idx][pix_x] & page_byte: if exclusive and self.vram[page_idx][pix_x] & page_byte:
#invert pixel if it has alread been set #invert pixel if it has alread been set
self.vram[self.CURRENT_BUF][page_idx][pix_x] &= ~page_byte self.vram[page_idx][pix_x] &= ~page_byte
else: else:
#set the correct pixel in the vram buffer to 1 #set the correct pixel in the vram buffer to 1
self.vram[self.CURRENT_BUF][page_idx][pix_x] |= page_byte self.vram[page_idx][pix_x] |= page_byte
def clear_pixel(self, pix_x, pix_y): def clear_pixel(self, pix_x, pix_y):
page_idx = pix_y // 8 page_idx = pix_y // 8
page_byte = ~(0x01 << (pix_y % 8)) page_byte = ~(0x01 << (pix_y % 8))
#set the correct pixel in the vram buffer to 0 #set the correct pixel in the vram buffer to 0
self.vram[self.CURRENT_BUF][page_idx][pix_x] &= page_byte self.vram[page_idx][pix_x] &= page_byte
def write_text(self, x, y, data): def write_text(self, x, y, data):
if x + len(data) > 16: if x + len(data) > 16:
data = data[:16 - min(x, 16)] data = data[:16 - min(x, 16)]
@ -155,7 +154,7 @@ class UC1701:
return 0 return 0
def clear(self): def clear(self):
zeros = bytearray(128) zeros = bytearray(128)
for page in self.vram[self.CURRENT_BUF]: for page in self.vram:
page[:] = zeros page[:] = zeros
def get_dimensions(self): def get_dimensions(self):
return (16, 4) return (16, 4)