; WEDGEF64
;
; Programma per Proteus
;
; (C) 2004 Simone Zanella Productions
;
; Riceve da un dispositivo seriale i dati inviati nel formato Datalogic® wedge protocol (utilizzato da F64/F67)
; e li inserisce in emulazione di tastiera.
; Questo programma può essere installato come script associato al servizio Proteus.
;
; I parametri di comunicazione si trovano all'inizio del programma e sono:
; - COMPORT = porta di comunicazione ("COM1", "COM2", ecc.)
; - COMSPEED = velocità di comunicazione (1200, 2400, 4800, 9600, 19200, 38400, 57600, 115200)
; - COMPARITY = "N" (nessuna), "E" (pari), "O" (dispari), "M" (marca), "S" (spazio)
; - COMDATA = 7 o 8 (bit di dati)
; - COMSTOP = 1 o 2 (bit di stop)
; - COMFLOW = "R" (RTS/CTS detto anche hardware), "X" (XON/XOFF detto anche software), "E" (entrambi), "N" (nessuno)
; - POSTFIX = terminatore (di solito: "{ENTER}")
;
; E' poi possibile specificare:
; - PACE: ritardo dopo l'invio in emulazione di tastiera di ciascuna riga (in secondi);
; - POLLINTERVAL: periodicità (in secondi) con la quale il programma verifica la presenza di dati disponibili
;   (default: 0.64);
; - TIMEOUT: timeout (in millisecondi) per la ricezione di un pacchetto dal terminale.
; 
; Modificare la funzione ProcessRecord per scegliere quale parte del record inviare.
;
; Il codice commentato più sotto permette anche in alternativa di:
; - selezionare una specifica finestra (applicativo Ultra-Edit) ed inviare il dato verbatim, seguito da Invio;
; - lanciare il blocco note (se non aperto), introdurre il dato verbatim (seguito da Invio) e tornare alla finestra
;   iniziale.

#!proteus -z -j

!include "win32.prt"

; Parametri di comunicazione
; --------------------------

; Porta seriale
COMPORT = "COM1"
; Baud rate
COMSPEED = 9600
; Parità (None, Odd, Even, Mark, Space: solo iniziale)
COMPARITY = "E"
; Bit di dati
COMDATA = 7
; Bit di stop
COMSTOP = 1
; Controllo di flusso (Rts/cts, Xon/xoff, Entrambi, Nessuno: solo iniziale)
COMFLOW = "N"

; Ritardo interlinea
PACE = 0.02

; Intervallo di polling
POLLINTERVAL = 0.64

; Timeout (in millisecondi) per la ricezione di un pacchetto dal terminale
TIMEOUT = 500

; Terminatore (default = "{ENTER}")
POSTFIX = "{ENTER}"

; ---------------------------

HCOM = OpenCom(COMPORT, COMSPEED, COMPARITY, COMDATA, COMSTOP, COMFLOW)

ACK = CHR(6)
NAK = CHR(21)

NULL = CHR(0)
ESC = CHR(0x1B)
CR = CHR(0x0D)
LF = CHR(0x0A)
ETB = CHR(0x17)
EOT = CHR(0x04)

PACKETENQ = MKPacket(NULL ESC "5$" ESC CR)

WHILE 1
  ; Attende per l'intervallo specificato
  SLEEP(POLLINTERVAL)
  
  ; Se non è un servizio e l'utente ha premuto ESC, esce
  !ifndef SERVICE
    IF KBDHIT()
      IF EQ(GETCH(), 27)
        BREAK
      FI
    FI
  !endif

  ; Invia la richiesta dati
  W32WRITEFILE(HCOM, PACKETENQ)

  NCHAR = W32WAITRXCHAR(HCOM, TIMEOUT)
  IF NCHAR
    ; Ricevuti dei caratteri
    RXWedgeProtocol()
  FI
LOOP
W32CLOSEHANDLE(HCOM)
ABORT 0   


FUNCTION RXWedgeProtocol()

lastblock = 0
rxabort = 0
nakcount = 0
_Q = QUEUENEW()
lastpacketsent = _PACKETENQ
status = 0
WHILE 1
  s = W32READCOM(_HCOM, 0)
  WHILE NOT(STRSTR(s, _LF))
    n = W32WAITRXCHAR(_HCOM, _TIMEOUT)
    IF n
      s = s W32READCOM(_HCOM, 0)
      nakcount = 0
    ELSE
      ; In caso di timeout, invia NAK e riprova 2 volte
      CONSOLELN "s: " s
      CONSOLELN "TIMEOUT"
      INC(@nakcount)
      IF LT(nakcount, 2)
        s = ""
        W32READCOM(_HCOM, 0)
        W32WRITEFILE(_HCOM, lastpacketsent)
        CONSOLELN "RIPROVO"
        CONTINUE
      ELSE     
        ; Troppi NAK: ricezione interrotta (scarta tutti i record ricevuti) 
        rxabort = 1
      FI
      BREAK
    FI
  LOOP
  IF rxabort
    BREAK
  FI  
  p = STRSTR(s, _LF)
  s = LEFT(s, p)
  record = ""
  result = CheckFrame(s, @record, @lastblock)
  consoleln "record: " record
  W32READCOM(_HCOM, 0)      
  IF result
    SWITCH status
    ON 0
      ; Verifica se ha ricevuto [xx]<RUN>      
      IF STREQ(RIGHT(record, 5), "<RUN>")
        W32WRITEFILE(_HCOM, _ACK)
        ; Resetta il puntatore all'inizio della lista record
        lastpacketsent = MKPacket(_NULL _ESC "0#" _ESC _CR)
        W32WRITEFILE(_HCOM, lastpacketsent)
        INC(@status)
        consoleln "RUN OK, INVIATO 0#"
      ELSE
        ; Record valido: lo accodo
        lastpacketsent = _NAK
        W32WRITEFILE(_HCOM, _ACK)
        status = 2
        IF NOT(lastblock)
          ENQUEUE(_Q, record)
        FI 
        ; W32WRITEFILE(_HCOM, _NAK)
      FI

    ON 1
      IF STREQ(LEFT(record, 5), "<ACK>")
        W32WRITEFILE(_HCOM, _ACK)
        ; Chiede i primi 100 record
        lastpacketsent = MKPacket(_NULL _ESC "0#100" _ESC _CR)
        W32WRITEFILE(_HCOM, lastpacketsent)
        INC(@status)
        consoleln "ACK OK, INVIATO 0#100"
      ELSE
        ; Record valido: lo accodo
        lastpacketsent = _NAK
        W32WRITEFILE(_HCOM, _ACK)
        INC(@status)
        IF NOT(lastblock)
          ENQUEUE(_Q, record)
        FI 
        ;W32WRITEFILE(_HCOM, _NAK)
      FI
    
    ON 2
      ; Record valido: lo accodo
      lastpacketsent = _NAK
      W32WRITEFILE(_HCOM, _ACK)
      IF NOT(lastblock)
        ENQUEUE(_Q, record)
      FI
    OFF
  ELSE
    ; Record non valido
    W32WRITEFILE(_HCOM, _NAK)
  FI
  IF lastblock
    IF EQ(lastblock, 1)
      ; Ricevuto ETB, attendo EOT
      lastblock = 0
      ; Chiedo altri 100 record (se ci sono)
      lastpacketsent = MKPacket(_NULL _ESC "0#100" _ESC _CR)
      W32WRITEFILE(_HCOM, lastpacketsent)
      status = 2
    ELSE
      BREAK
    FI
  FI
LOOP
IF NOT(rxabort)
  EmulateKeyboard(_Q)
FI
QUEUEFREE(_Q)
RETURN


FUNCTION MKPacket(s)

RETURN s CalcLRC(s) _LF


FUNCTION CheckFrame(s, record, lastblock)

; L'inizio del frame è invalido?
IF STRNEQ(LEFT(s, 2), _NULL _NULL)
  IF STRNEQ(LEFT(s, 1), _NULL)
    RETURN 0
  ELSE
    s = _NULL s
  FI
FI

; Verifica la fine del blocco
IF STRNEQ(RIGHT(s, 1), _LF)
  RETURN 0
FI

; Elimina la fine del blocco
s = LEFT(s, -1)

; Isola l'LRC
lrc = RIGHT(s, 1)

; Elimina l'LRC
s = LEFT(s, -1)

IF STRNEQ(RIGHT(s, 1), _CR)
  RETURN 0
FI

; Verifica l'LRC
IF NOT(VerifyLRC(lrc, s))
  RETURN 0
FI

; Elimina i NULL iniziali
s = RESTFROM(s, 3)
; Elimina il CR finale
s = LEFT(s, -1)

; Verifica se è l'ultimo blocco e aggiorna i parametri
SWITCH LEFT(s, 1) STREQ
ON _EOT
  lastblock = 1
ON _ETB
  lastblock = 2
OTHER
  lastblock = 0
OFF

; Memorizza il record
record = s
RETURN 1


FUNCTION CalcLRC(s)

acc = 0
l = STRLEN(s)
; XOR di tutti i caratteri in s
FOR x = 1 to l
  NXOR(@acc, ASC(SUBSTR(s, x, 1)))
NEXT
RETURN CHR(acc)


FUNCTION VerifyLRC(lrc, s)

RETURN STREQ(lrc, CalcLRC(s))


FUNCTION EmulateKeyboard(q)

; Variante con salvataggio/ripristino finestra corrente e selezione blocco notes
;   ; Preleva la finestra corrente
;   hOld = W32GETFOCUS()
;     
;   ; Cerca il blocco notes
;   hNotepad = W32FINDWINDOW("*Blocco note*")
;   IF EQ(hNotepad, 0)
;     ; Non trovato - lo lancia
;     W32SHELL("NOTEPAD.EXE")
;     ; Attende un secondo che si renda disponibile
;     SLEEP(1)
;     ; Cerca la sua finestra nuovamente
;     hNotepad = W32FINDWINDOW("*Notepad*")
;   FI
;   ; Se trovato, seleziona la sua finestra ed invia i dati
;   IF EQ(hNotepad, 0)
;     W32SETFOCUS(hOld)
;     RETURN
;   FI
;   W32SETFOCUS(hNotepad)

WHILE QUEUELEN(q)
  W32SENDKEYS(KTrans(DEQUEUE(q)) _POSTFIX)
  SLEEP(_PACE)
LOOP
; Variante - ripristina la vecchia finestra
;   W32SETFOCUS(hOld)
RETURN


FUNCTION KTrans(s)

l = STRLEN(s)
r = ""
FOR x = 1 TO l
  c = SUBSTR(s, x, 1)
  ; Mappatura dei caratteri speciali: caratteri che non si trovano
  ; sulla tastiera potrebbero richiedere l'introduzione della sequenza
  ; con ALT + numero
  SWITCH c STREQ
  ON "~"
    r = r "{ALT DOWN}{NUMPAD1}{NUMPAD2}{NUMPAD6}{ALT UP}"
  ON "{"
    r = r "{ALT DOWN}{NUMPAD1}{NUMPAD2}{NUMPAD3}{ALT UP}"    
  ON "}"
    r = r "{ALT DOWN}{NUMPAD1}{NUMPAD2}{NUMPAD5}{ALT UP}"
  ON "+", "^", "%", "(", ")", "[", "]"
    r = r "{" c "}"
  OTHER
    r = r c
  OFF
NEXT
RETURN r


FUNCTION OpenCom(comport, comspeed, comparity, comdata, comstop, comflow)

; Apre la porta seriale con i parametri specificati
hcom = W32CREATEFILE(comport, NOR(_W32_GENERIC_WRITE, _W32_GENERIC_READ), 0, \
                     _W32_OPEN_EXISTING, 0)

IF EQ(hcom, -1)
  !ifndef SERVICE
    CONSOLELN "Impossibile aprire " comport "."
  !endif
  ABORT 2
FI

compar = VECNEW(13)
v = W32GETCOMSTATE(hcom, compar)

VECSET(compar, 2, comspeed)
v = NOR(_W32_COM_BINARY, _W32_COM_PARITY_ON)
SWITCH LEFT(comflow, 1) STRIEQ
ON "R"
  NOR(@v, _W32_COM_RTS_HANDSHAKE, _W32_COM_CTSFLOW_ON)
ON "X"
  NOR(@v, _W32_COM_XONXOFF_OUT, _W32_COM_XONXOFF_IN, _W32_COM_XOFF_CONTINUE)
ON "E"
  NOR(@v, _W32_COM_RTS_HANDSHAKE, _W32_COM_CTSFLOW_ON, \
          _W32_COM_XONXOFF_OUT, _W32_COM_XONXOFF_IN, _W32_COM_XOFF_CONTINUE)
ON "N"
OFF

VECSET(compar, 3, v)
VECSET(compar, 7, comdata)

SWITCH LEFT(comparity, 1) STRIEQ
ON "N"
  v = _W32_COM_PARITY_NONE
ON "E"
  v = _W32_COM_PARITY_EVEN
ON "O"
  v = _W32_COM_PARITY_ODD
ON "M"
  v = _W32_COM_PARITY_MARK
ON "S"
  v = _W32_COM_PARITY_SPACE
OFF

VECSET(compar, 8, v)
SWITCH comstop
ON 1
  VECSET(compar, 9, 0)
ON 2
  VECSET(compar, 9, 2)
OFF

v = W32SETCOMSTATE(hcom, compar)
VECFREE(compar)

IF v
  !ifndef SERVICE
    CONSOLELN "Errore nell'impostazione della porta (" W32GETLASTERROR() ")."
  !endif
  W32CLOSEHANDLE(hcom)  
  ABORT 3
FI

tout = VECNEW(5)
VECSET(tout, 1, 0)
VECSET(tout, 2, 0)
VECSET(tout, 3, 0)
VECSET(tout, 4, 0)
VECSET(tout, 5, 0)
W32SETCOMTIMEOUTS(hcom, tout)
VECFREE(tout)

RETURN hcom
Midnight Lake iPhone Case Black Women Shoes Black Flat Shoes Leather Flats Black Patent Ballerinas Black Ballet Shoes Casual Shoes Black Shoes Women Balle Record Player Cufflinks Best iPhone XR Clear Cases iPhone XS/XS Max Leather Cases Sale Best iPhone 8/8 Plus Silicone Cases iPhone 7/7 Plus Cases & Screen Protector New Cases For iPhone 6/6 Plus iPhone 8 Case Sale iPhone Xr Case Online iPhone 7 Case UK Online iPhone X Case UK Sale iPhone X Case Deals iPhone Xs Case New Case For iPhone Xr UK Online Case For iPhone 8 UK Outlet Fashion Silver Cufflinks For Men Best Mens Cufflinks Outlet Online The Gold Cufflinks Shop Online Cheap Shirt Cufflinks On Sale Nice Wedding Cufflinks UK Online Top Black Cufflinks UK Online Mens Cufflinks Online Silver Cufflinks For Men Men Cufflinks UK Sale Gold Cufflinks UK Online Gold Cufflinks UK Silver Cufflinks UK Shirt Cufflinks Discount Online Mens Cufflinks Deals & Sales Girls Shoes For Dance Fashion Ballet Dance Shoes Best Ballet Flats Shoes UK Online Cheap Ballet Pointe Shoes UK Online Best Ballet Shoes Outlet Best Dance Shoes Sale Cheap Ballet Flats Sale UK Best Pointe Shoes Online UK Ballet Dance Shoes UK Shoes For Dance UK Best Ballet Slippers Shop Best Yoga Shoes Hotsell