from machine import Pin
from rp2 import PIO, StateMachine, asm_pio
from time import sleep_ms

class PIO_QENCONE:
    def __init__(self, sm_id, pins, freq=10_000_000):
        if len(pins) != 2:
            raise ValueError('2 pins required')

        pinA, pinB = pins

        # Ellenőrizzük, hogy a pinA és pinB egymás mellett van-e
        if abs(pinA - pinB) != 1:
            raise ValueError('Pins must be consecutive')

        # Pin-ek inicializálása
        self.pinA = Pin(pinA, Pin.IN, Pin.PULL_UP)
        self.pinB = Pin(pinB, Pin.IN, Pin.PULL_UP)

        # Állapotgép inicializálása
        self.sm_qenc = StateMachine(sm_id, self.sm_qenc1, freq=freq, in_base=self.pinA, out_base=self.pinB)
        self.sm_qenc.active(1)
    
    @staticmethod
    @rp2.asm_pio(in_shiftdir=PIO.SHIFT_LEFT, out_shiftdir=PIO.SHIFT_RIGHT)
    def sm_qenc1():
        jmp("read")        # 000 : B from 0 to 0         => no change
        jmp("decr")        # 001 : B from 0 to 1, A = 0  => backward
        jmp("read")        # 010 : B from 0 to 0         => no change
        jmp("incr")        # 011 : B from 0 to 1, A = 1  => forward
        jmp("incr")        # 100 : B from 1 to 0, A = 0  => forward
        jmp("read")        # 101 : B from 0 to 0         => no change

        label("decr")
        jmp(x_dec, "read") # 110 : B from 1 to 0, A = 1  => backward

        label("read")      # 111 : B from 0 to 0         => no change
        mov(osr, isr)      # save last pin input in OSR
        mov(isr, x)
        push(noblock)
        out(isr, 1)        # right bit B' of OSR into ISR, all other 0
        in_(pins, 2)       # combined with current reading A B of input pins

        mov(pc, isr)       # jump into jump-table at addr 0

        label("incr")      # increment x by inverting, decrementing and inverting
        mov(x, invert(x))
        jmp(x_dec, "here")
        label("here")
        mov(x, invert(x))  # we rely on implicit .wrap with micropython
        jmp("read")

    def read(self):
        for _ in range(self.sm_qenc.rx_fifo()):
            self.sm_qenc.get()
        n = self.sm_qenc.get()
        return n if n < (1<<31) else n - (1<<32)



class PIO_QENC:
    def __init__(self, sm_id, pins, freq=10_000_000):
        if len(pins) != 2:
            raise ValueError('2 pins required')

        pinA, pinB = pins

        # Ellenőrizzük, hogy a pinA és pinB egymás mellett van-e
        if abs(pinA - pinB) != 1:
            raise ValueError('Pins must be consecutive')

        # Pin-ek inicializálása
        self.pinA = Pin(pinA, Pin.IN, Pin.PULL_UP)
        self.pinB = Pin(pinB, Pin.IN, Pin.PULL_UP)

        # Állapotgép inicializálása
        self.sm_qenc = StateMachine(sm_id, self.sm_qenc, freq=freq, in_base=self.pinA, out_base=self.pinB)
        self.sm_qenc.active(1)
    
    @staticmethod
    @rp2.asm_pio(in_shiftdir=PIO.SHIFT_LEFT, out_shiftdir=PIO.SHIFT_RIGHT)
    def sm_qenc():
        # kvadraturális encoder feldolgozó kódja
        jmp("read")        # 0000 : from 00 to 00 = no change
        jmp("decr")        # 0001 : from 00 to 01 = backward
        jmp("incr")        # 0010 : from 00 to 10 = forward
        jmp("read")        # 0011 : from 00 to 11 = error
        jmp("incr")        # 0100 : from 01 to 00 = forward
        jmp("read")        # 0101 : from 01 to 01 = no change
        jmp("read")        # 0110 : from 01 to 10 = error
        jmp("decr")        # 0111 : from 01 to 11 = backward
        jmp("decr")        # 1000 : from 10 to 00 = backward
        jmp("read")        # 1001 : from 10 to 01 = error
        jmp("read")        # 1010 : from 10 to 10 = no change
        jmp("incr")        # 1011 : from 10 to 11 = forward
        jmp("read")        # 1100 : from 11 to 00 = error
        jmp("incr")        # 1101 : from 11 to 01 = forward

        label("decr")
        jmp(x_dec, "read") # 1110 : from 11 to 10 = backward

        label("read")      # 1111 : from 11 to 11 = no change
        mov(osr, isr)      # save last pin input in OSR
        mov(isr, x)
        push(noblock)
        out(isr, 2)        # 2 right bits of OSR into ISR, all other 0
        in_(pins, 2)       # combined with current reading of input pins
        mov(pc, isr)       # jump into jump-table at addr 0

        label("incr")      # increment x by inverting, decrementing and inverting
        mov(x, invert(x))
        jmp(x_dec, "here")
        label("here")
        mov(x, invert(x))
        jmp("read")
        
        nop()
        nop()
        nop()
        nop()
        nop()
        nop()
        nop()

    def read(self):
        for _ in range(self.sm_qenc.rx_fifo()):
            self.sm_qenc.get()
        n = self.sm_qenc.get()
        return n if n < (1<<31) else n - (1<<32)

# A két encoder pin beállítása (egymás melletti pin)
pinA = 15  # Pin szám
pinB = 16  # Pin szám

pinC = 17  # Pin szám
pinD = 18  # Pin szám

# Két encoder, egyik 1x másik 4x módon működik
qenc1 = PIO_QENC(0, (pinA, pinB))
qenc2 = PIO_QENCONE(1, (pinC, pinD))

# Tesztelés: 1 másodpercenként kiírjuk az encoderek számláló értékeit
# for i in range(120):
c=0
while True:
    print(c,'.Encoder 1:', qenc1.read())
    print(c,'.Encoder 2:', qenc2.read())
    sleep_ms(10)
    c+=1
qenc1.sm_qenc.active(0)
qenc2.sm_qenc.active(0)
print('Stopped')
BOOTSELLED1239USBRaspberryPiPico©2020RP2-8020/21P64M15.00TTT