You've already forked TEF6686_Driver
89 lines
2.9 KiB
Python
89 lines
2.9 KiB
Python
import serial
|
|
import struct
|
|
import time
|
|
|
|
def crc8(data: bytes) -> int:
|
|
crc = 0x00
|
|
for byte in data:
|
|
crc ^= byte
|
|
for _ in range(8):
|
|
if crc & 0x80: crc = ((crc << 1) & 0xFF) ^ 0x07
|
|
else: crc = (crc << 1) & 0xFF
|
|
return crc
|
|
|
|
class I2CPCClient:
|
|
def __init__(self, port, baudrate=115200, timeout=1):
|
|
self.ser = serial.Serial(port=port, baudrate=baudrate, timeout=timeout)
|
|
|
|
# Gurantees resync
|
|
self.ser.write(b"~/")
|
|
self.ser.write(b"\x00"*64)
|
|
self.ser.flush()
|
|
self.ser.write(b"~/")
|
|
self.ser.flush()
|
|
while not ((d := self.ser.read_all()) and b"\xff" in d): time.sleep(0.01)
|
|
|
|
def _send_packet(self, payload: bytes, crc: bool = True):
|
|
if len(payload) > 127: raise ValueError("Payload too large")
|
|
|
|
length_byte = len(payload) | (0x80 if crc else 0x00)
|
|
data = bytes([length_byte]) + payload
|
|
if crc: data += bytes([crc8(data)])
|
|
self.ser.write(data)
|
|
|
|
# Read response length
|
|
resp_len_raw = self.ser.read(1)
|
|
if not resp_len_raw: raise TimeoutError("No response")
|
|
|
|
resp_len = resp_len_raw[0]
|
|
resp_has_crc = bool(resp_len & 0x80)
|
|
resp_len &= 0x7F
|
|
|
|
total_to_read = resp_len + (1 if resp_has_crc else 0)
|
|
|
|
response = self.ser.read(total_to_read)
|
|
if len(response) != total_to_read: raise TimeoutError("Incomplete response")
|
|
|
|
if resp_has_crc:
|
|
received_crc = response[-1]
|
|
body = resp_len_raw + response[:-1]
|
|
if crc8(body) != received_crc: raise ValueError("CRC mismatch")
|
|
response = response[:-1]
|
|
|
|
return response
|
|
|
|
|
|
def set_clock(self, clock_hz: int):
|
|
payload = bytes([0]) + struct.pack(">I", clock_hz)
|
|
return self._send_packet(payload)
|
|
|
|
def write_i2c(self, addr: int, data: bytes):
|
|
payload = bytes([1, addr]) + data
|
|
return self._send_packet(payload)
|
|
|
|
def write_read_i2c(self, addr: int, write_data: bytes, read_len: int):
|
|
payload = bytes([2, addr, len(write_data)]) + write_data + bytes([read_len])
|
|
return self._send_packet(payload)
|
|
|
|
def write_eeprom(self, addr: int, data: bytes):
|
|
payload = bytes([7, (addr >> 8) & 0xff, addr & 0xff]) + data
|
|
return self._send_packet(payload)
|
|
def read_eeprom(self, addr: int, len: int):
|
|
payload = bytes([8, (addr >> 8) & 0xff, addr & 0xff, len])
|
|
return self._send_packet(payload)
|
|
|
|
def version(self): return self._send_packet(bytes([4]))
|
|
|
|
def quit(self): return self._send_packet(bytes([3]))
|
|
|
|
def reboot(self): return self._send_packet(bytes([5]))
|
|
|
|
def get_persistence_address(self): return self._send_packet(bytes([0xfe]))
|
|
|
|
def set_baudrate(self, baud: int):
|
|
payload = bytes([6]) + struct.pack(">I", baud)
|
|
out = self._send_packet(payload)
|
|
self.ser.baudrate = baud
|
|
return out
|
|
|
|
def close(self): self.ser.close() |