Individuare i tasti della tastiera intercettando i dati grezzi dal suo file-device

Da Gambas-it.org - Wikipedia.

E' possibile intercettare i dati grezzi provenienti dalla tastiera, per individuare di ciascun tasto premuto il suo Codice di Scansione.

Per gestire direttamente i dati grezzi della tastiera, bisogna innanzitutto individuare il suo file-device.
Sappiamo che esso è presente nel percorso /dev/input/, ed è rappresentato da uno dei file con il nome eventN; laddove "N" è un suffisso del nome rappresentato da un numero. [nota 1]
Per individuare precisamente quale sia il file-device associato alla tastiera, fra quelli che iniziano per event, bisognerà interrogare il file:

/proc/bus/input/devices

Tale file mostra quali dispositivi e gestori di input sono attualmente attivi, ed all'interno del quale bisognerà individuare la riga che inizia per "H:", e che rappresenta i driver gestore associati ad un determinato dispositivo. Pertanto, la riga "H:" che si riferisce alla tastiera si presenta visivamente come segue:

H: Handlers=sysrq kbd eventn

laddove n è in realtà un numero. [nota 2]

Individuato, dunque, il file-device associato alla tastiera, lo gestiremo con una variabile di tipo File o di tipo Process.

Ogni volta che è premuto un tasto della tastiera, vengono scritti nel file-device della tastiera alcuni dati, che potranno essere ovviamente letti dal programma Gambas.
Il 21° dato ci darà il "Codice di Scansione" del tasto premuto. Tale codice, identificando univocamente il tasto appena premuto o rilasciato, è dunque universale per tutti i calcolatori, e quindi tranquillamente utilizzabile nei nostri programmi Gambas.


Codice esemplificativo in ambiente grafico con le sole risorse di Gambas

Mostriamo di seguito un possibile codice esemplificativo per intercettare e ricavare in ambiente grafico di Gambas, per ciascun tasto premuto, il "Codice di Scansione"
Supponiamo, nell'esempio che segue, che sia stato già individuato il file-device corrispondente alla tastiera.
Tale file-device sarà innazitutto privato della sua protezione e quindi sarà aperto in "lettura", gestito con una variabile di tipo File e posto sotto osservazione con la parola-chiave Watch.
I dati letti dal file-device saranno caricati in una variabile di tipo Stringa, dalla quale sarà estrapolato il 21° dato-valore.

Private fl As File


Public Sub Form_Open()

' Supponiamo che file-device da leggere individuato nel file "/proc/bus/input/devices" sia "/dev/input/event4".
' Eliminiamo innanzitutto la sua protezione:
 Shell "echo " & InputBox("Immettere la propria PASSWORD di sistema:") & " | sudo -S chmod 444 /dev/input/event4" Wait

' Quindi lo apriamo in "lettura" e lo poniamo sotto "osservazione":
 fl = Open "/dev/input/event4" For Read Watch
  
End


Public Sub File_Read()
 
 Dim s As String
 
 TextArea1.Text &= gb.NewLine
 
 Read #fl, s, -256
 
 TextArea1.Text &= "Codice di scansione del tasto premuto o rilasciato:  " & Asc(s, 21) & "  =  "
    
End


Public Sub Form_Close()
 
 fl.Close
 
End

Come il precedente, ma in applicazione a riga di comando:

Private fl As File


Public Sub Main()

' Con questa riga di comando viene eliminata la protezione al file-device da leggere.
' E' necessario porre la propria "password" di sistema al posto della parola "MIA_PASSWORD":
 Shell "echo MIA_PASSWORD | sudo -S chmod 444 /dev/input/event4" Wait

 fl = Open "/dev/input/event4" For Read Watch

End


Public Sub File_Read()
 
 Dim s As String

 Read #fl, s, -256
 
 Print "Codice di scansione del tasto premuto:  " & Asc(s, 21)
   
End


Public Sub Application_Read() ' Se si preme il tasto "Invio" della tastiera, viene sollevato questo Evento
 
 fl.Close

' Avendo sollevato l'Evento "Application_Read()", per chiudere l'applicazione è necessario usare l'istruzione "Quit":
 Quit
 
End


Usando la funzione Exec, il comando bash cat e la Classe Process

In quest'altro esempio è necessario attivare il Componente gb.desktop: si provvederà da codice ad individuare e successivamente a eliminare la protezione del file-device corrispondente alla tastiera.
Il file-device relativo alla tastiera sara gestito con le risorse della Classe Process.
Per ogni tasto che viene premuto, vengono intercettati dal file-device della tastiera ben 144 dati; il 21° ci darà il codice di scansione del tasto premuto della tastiera.

Private pr As Process
Private lett As New String[]


Public Sub Form_Open()

 Dim s As String
 Dim i As Integer
  
 s = File.Load("/proc/bus/input/devices")
  
' Individuiamo la riga contenente il testo "Handlers=sysrq"; ed individuiamo il nome del file-device della tastiera:
 s = Trim(Scan(s, "*Handlers=sysrq *\n*")[1])
 i = InStr(s, "event")
 s = Mid(s, i, 6) ' ...oppure: s[i - 1, 6]
  
' Per leggere il file-device individuato, bisogna avere i necessari permessi.
' Pertanto, se non sono stati modificati i permessi del file-device, allora vengono modificati (bisognerà inserire la propria password):
 If Stat("/dev/input" &/ s).Auth <> "r--r--r--" Then
   Desktop.RunAsRoot("sudo -S chmod 444 /dev/input" &/ s)
 Endif
  
' Poniamo un'attesa mediante un ciclo, fino a che i permessi del file-device non saranno mutati:
 Do
   Wait 0.01
 Loop Until Stat("/dev/input" &/ s).Auth = "r--r--r--"

' Finalmente il file-device viene gestito mediante un "Processo":
 pr = Exec ["cat", "/dev/input" &/ s] For Read As "Processo"

End


Public Sub Processo_Read()

 Dim b As Byte
 
' Legge i dati provenienti dal file-device della tastiera:
 Read #pr, b
  
 lett.Push(CStr(b))
  
 If lett.Count = 144 Then
' Legge l'elemento di numero d'indice 20 della variabile array:
   Print "Codice scansione tasto:   "; lett[20]
   lett.Clear
 Endif
 
End

Come il precedente, ma in applicazione a riga di comando

Mostriamo di seguito un possibile codice esemplificativo per intercettare e ricavare nell'ambito di un'applicazione Gambas a riga di comando, per ciascun tasto premuto, il codice di scansione.

Private pr As Process
Private lett As New String[]


Public Sub Main()

 Dim s, pass As String
 Dim i As Integer
     
 s = File.Load("/proc/bus/input/devices")
   
' Individuiamo la riga contenente il testo "Handlers=sysrq"; ed individuiamo il nome del file-device della tastiera:
 s = Trim(Scan(s, "*Handlers=sysrq *\n*")[1])
 i = InStr(s, "event")
 s = Mid(s, i, 6)
    
' Per leggere il file-device individuato, bisogna avere i necessari permessi. Essi vengono quindi modificati.
' E' necessario inserire nell'apposito spazio sottostante la Console (o nel Terminale) la password personale dell'utente:
 Print "Inserire nello spazio sottostante la propria 'password'."
 Input pass
 Shell "echo " & pass & " | sudo -S chmod 444 /dev/input" &/ s Wait
 Print  
 
' Finalmente il file-device viene gestito mediante un "Processo":
 pr = Exec ["cat", "/dev/input" &/ s] For Read As "Processo"
  
End


Public Sub Processo_Read()

 Dim b As Byte

 While Not Eof(pr)
' Legge i dati provenienti dal file-device della tastiera:
   Read #pr, b
   lett.Push(CStr(b))
   If lett.Count = 144 Then
' Legge l'elemento di numero d'indice 20 della variabile array:
     Print "Codice scansione tasto:   "; lett[20];
     Print
     lett.Clear
   Endif
 Wend
  
End


Usando la funzione esterna "ioctl()"

In questo esempio useremo la funzione esterna "ioctl()":

Library "libc:6"

Public Struct timeval
  tv_sec As Long
  tv_usec As Long
End Struct
 
Public Struct input_event
  time_ As Struct Timeval
  type As Short
  code As Short
  value As Integer
End Struct

Private Const _IOC_NRSHIFT As Integer = 0
Private Const _IOC_NRBITS As Integer = 8
Private _IOC_TYPESHIFT As Long = _IOC_NRSHIFT + _IOC_NRBITS
Private Const _IOC_TYPEBITS As Long = 8
Private _IOC_SIZESHIFT As Long = _IOC_TYPESHIFT + _IOC_TYPEBITS
Private Const _IOC_SIZEBITS As Long = 14
Private _IOC_DIRSHIFT As Long = _IOC_SIZESHIFT + _IOC_SIZEBITS
Private Const _IOC_WRITE As Long = 1
Private Const _IOC_READ As Long = 2

' int ioctl(int __fd, unsigned long int __request, ...)
' Perform the I/O control operation specified by REQUEST on FD.
Private Extern ioctl_name(__fd As Integer, __request As Long, arg As Byte[]) As Integer Exec "ioctl"

Private Extern ioctl_grab(__fd As Integer, __request As Long, arg As Integer) As Integer Exec "ioctl"

' ssize_t read (int __fd, void *__buf, size_t __nbytes)
' Read NBYTES into BUF from FD.
Private Extern read_C(__fd As Integer, __buf As Input_event, __nbytes As Long) As Long Exec "read"

Private ke As New Input_event
Private kfl As File


Public Sub Main()

 Dim keyboard_name As New Byte[256]
 Dim rit As Long

' Supponendo che il file-device attinente alla tastiera sia "event4".
' Con questa riga di comando viene eliminata la protezione al file-device da leggere.
' E' necessario porre la propria "password" di sistema al posto della parola "MIA_PASSWORD":
 Shell "echo MIA_PASSWORD | sudo -S chmod 444 /dev/input/event4" Wait

' Quindi lo si apre in "Lettura" e lo si pone in "osservazione":
 kfl = Open "/dev/input/event4" For Read Watch
 If kfl.Handle == -1 Then
   kfl.Close
   Error.Raise("Error !")
 Endif

 ioctl_name(kfl.Handle, Eviocname(Asc("E"), &h06, keyboard_name.Count), keyboard_name)
 Print "Reading From: \e[34m"; keyboard_name.ToString(0, keyboard_name.Find(0))

 rit = ioctl_grab(kfl.Handle, Eviocgrab(Asc("E"), &h90, SizeOf(gb.Integer)), 1)
 Print "\e[0mGetting exclusive access: "; IIf(rit == 0, "\e[32mSUCCESS\e[0m", "\e[31mFAILURE\e[0m")

 Print

End


Public Sub File_Read()
 
 Dim rit As Long
 
 rit = read_C(kfl.Handle, ke, Object.SizeOf(ke))
  
 If rit > -1 Then
   Print "\e[1mtime\e[0m: "; ke.time_.tv_sec; " , \e[1mtype\e[0m: "; ke.type &
   " , \e[1mcode\e[0m: "; ke.code; " , \e[1mvalue\e[0m: "; ke.value
' Se viene premuto il tasto "Esc", il programma viene chiuso:
   If (ke.type == 4) And (ke.code == 4) And (ke.value == 1) Then kfl.Close
 Endif
  
End


Private Function Eviocname(type As Long, nr As Long, size As Long) As Long
  
 Return Shl(_IOC_READ, _IOC_DIRSHIFT) Or Shl(type, _IOC_TYPESHIFT) Or Shl(nr, _IOC_NRSHIFT) Or Shl(size, _IOC_SIZESHIFT)
  
End


Private Function Eviocgrab(type As Long, nr As Long, size As Long) As Long
  
 Return Shl(_IOC_WRITE, _IOC_DIRSHIFT) Or Shl(type, _IOC_TYPESHIFT) Or Shl(nr, _IOC_NRSHIFT) Or Shl(size, _IOC_SIZESHIFT)
  
End


Note

[1] Il nome event è dato dalla circostanza che il rapporto tra i vari elementi del sottosistema di input - ossia gli ingressi principali, i driver e gestori di eventi - avviene attraverso eventi.

[2] Nelle versioni di Mint superiori alla 17 dopo la parola "eventn" è presente anche la parola "leds". Pertanto, se ne dovrà tenere conto ai fini dell'individuazione della parola "eventn".


Riferimenti