Differenze tra le versioni di "Inviare dati grezzi Midi attraverso il subsistema RawMidi"

Da Gambas-it.org - Wikipedia.
 
Riga 49: Riga 49:
 
===Esempio===
 
===Esempio===
 
Di seguito un semplice esempio di invio di due Messaggi Midi: un ''Note-On'' ed un ''Note-Off'' al device output Midi dell'interfaccia ''RawMidi'' di Alsa:
 
Di seguito un semplice esempio di invio di due Messaggi Midi: un ''Note-On'' ed un ''Note-Off'' al device output Midi dell'interfaccia ''RawMidi'' di Alsa:
  Library "libasound:2"
+
  Library "libasound:2.0.0"
 
   
 
   
 
  <FONT color=Gray>' ''int snd_rawmidi_open (snd_rawmidi_t **in_rmidi, snd_rawmidi_t **out_rmidi, const char *name, int mode)  --> Opens a New Connection To the RawMidi interface''</font>
 
  <FONT color=Gray>' ''int snd_rawmidi_open (snd_rawmidi_t **in_rmidi, snd_rawmidi_t **out_rmidi, const char *name, int mode)  --> Opens a New Connection To the RawMidi interface''</font>
  '''Private''' Extern snd_rawmidi_open(in_rmidi As Pointer, out_rmidi As Pointer, name As String, mode As Integer) As Integer
+
  Private Extern snd_rawmidi_open(in_rmidi As Pointer, out_rmidi As Pointer, name As String, mode As Integer) As Integer
 
   
 
   
 
  <FONT color=Gray>' ''ssize_t snd_rawmidi_write (snd_rawmidi_t *rmidi, const void *buffer, size_t size)  --> Write MIDI bytes To MIDI stream''</font>
 
  <FONT color=Gray>' ''ssize_t snd_rawmidi_write (snd_rawmidi_t *rmidi, const void *buffer, size_t size)  --> Write MIDI bytes To MIDI stream''</font>
  '''Private''' Extern snd_rawmidi_write(rmidi As Pointer, buffer As Stream, size As Integer) As Integer
+
  Private Extern snd_rawmidi_write(rmidi As Pointer, buffer As Stream, size As Integer) As Integer
 
   
 
   
 
  <FONT color=Gray>' ''const char * snd_strerror (int errnum)  --> Returns the message For an Error code''</font>  
 
  <FONT color=Gray>' ''const char * snd_strerror (int errnum)  --> Returns the message For an Error code''</font>  
  '''Private''' Extern snd_strerror(errnum As Integer) As String
+
  Private Extern snd_strerror(errnum As Integer) As String
 
   
 
   
 
  <FONT color=Gray>' ''int snd_rawmidi_close (snd_rawmidi_t *rmidi)  --> Close RawMidi handle''</font>
 
  <FONT color=Gray>' ''int snd_rawmidi_close (snd_rawmidi_t *rmidi)  --> Close RawMidi handle''</font>
  '''Private''' Extern snd_rawmidi_close(rmidi As Pointer) As Integer
+
  Private Extern snd_rawmidi_close(rmidi As Pointer) As Integer
 
   
 
   
 
   
 
   
  '''Public''' Sub Form_Open()
+
  Public Sub Form_Open()
 
  Dim posMid, err As Integer
 
  Dim SND_RAWMIDI_SYNC As Byte = 4
 
  Dim midiout, noteon, noteoff As Pointer
 
  Dim nome_porta As String
 
  Dim strOn, strOff As Stream
 
 
   
 
   
 +
  Dim posMid, err As Integer
 +
  Dim SND_RAWMIDI_SYNC As Byte = 4
 +
  Dim midiout, noteon, noteoff As Pointer
 +
  Dim nome_porta As String
 +
  Dim strOn, strOff As Stream
 
   
 
   
 
  <FONT color=Gray>' ''Algoritmo per individuare automaticamente il nome dell'handle dell'interfaccia "RawMidi":''</font>
 
  <FONT color=Gray>' ''Algoritmo per individuare automaticamente il nome dell'handle dell'interfaccia "RawMidi":''</font>
Riga 79: Riga 78:
 
   nome_porta = "hw:" & posMid & ",0,0"
 
   nome_porta = "hw:" & posMid & ",0,0"
 
   Print "Nome del dispositivo Midi di Alsa: "; nome_porta
 
   Print "Nome del dispositivo Midi di Alsa: "; nome_porta
 
 
 
 
 
   
 
   
    err = snd_rawmidi_open(0, VarPtr(midiout), nome_porta, SND_RAWMIDI_SYNC)
+
  err = snd_rawmidi_open(0, VarPtr(midiout), nome_porta, SND_RAWMIDI_SYNC)
    If err < 0 Then Error.Raise("Impossibile aprire la porta d'uscita del subsistema RawMidi: " & snd_strerror())
+
  If err < 0 Then Error.Raise("Impossibile aprire la porta d'uscita del subsistema RawMidi: " & snd_strerror())
 
      
 
      
 
  <FONT color=Gray>' ''Come mero esempio invieremo tramite il Midi output solo due Messaggi Midi: un NoteOn ed un NoteOff.''
 
  <FONT color=Gray>' ''Come mero esempio invieremo tramite il Midi output solo due Messaggi Midi: un NoteOn ed un NoteOff.''
 
  ' ''Bisognerà inviare tre byte di dati per volta alla funzione "snd_rawmidi_write ()".''
 
  ' ''Bisognerà inviare tre byte di dati per volta alla funzione "snd_rawmidi_write ()".''
  ' ''Poiché è previsto un Pointer, per riempirlo con i tre byte, dovremo creare un'area allocata di memoria,''
+
  ' ''Poiché è previsto un Pointer, per riempirlo con i tre byte, dovremo creare un'area allocata di memoria, e riempirla mediante la trasformazione dei Pointer in variabili di tipo "Stream", che passeremo alla predetta funzione:''</font>
' ''e riempirla mediante la trasformazione dei Pointer in variabili di tipo "Stream", che passeremo alla predetta funzione:''</font>
+
  noteon = Alloc(3)
    noteon = Alloc(3)
+
  noteoff = Alloc(3)
    noteoff = Alloc(3)
 
 
      
 
      
    strOn = Memory noteon For Write
+
  strOn = Memory noteon For Write
    strOff = Memory noteoff For Write
+
  strOff = Memory noteoff For Write
 
      
 
      
 
  <FONT color=Gray>' ''Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-On":''</font>
 
  <FONT color=Gray>' ''Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-On":''</font>
      Write #strOn, 144 As Byte
+
  Write #strOn, 144 As Byte
      Write #strOn, 60 As Byte
+
  Write #strOn, 60 As Byte
      Write #strOn, 100 As Byte
+
  Write #strOn, 100 As Byte
 
      
 
      
  <FONT color=Gray>' ''Il valore 3 sta a significare che saranno scritti nel dispositivo Midi output 3 byte fra quelli presenti nella variabile "stream",''
+
  <FONT color=Gray>' ''Il valore 3 sta a significare che saranno scritti nel dispositivo Midi output 3 byte fra quelli presenti nella variabile "stream", ossia i tre valori costitutivi del Messaggio Note On:''</font>
' ''ossia i tre valori costitutivi del Messaggio Note On:''</font>
+
  err = snd_rawmidi_write(midiout, strOn, <FONT color=#B22222>3</font>)
    err = snd_rawmidi_write(midiout, strOn, <FONT color=#B22222>3</font>)
+
  If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))
    If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))
 
 
   
 
   
 
  <FONT color=Gray>' ''Attendiamo che la nota suoni per 1 secondo:''</font>
 
  <FONT color=Gray>' ''Attendiamo che la nota suoni per 1 secondo:''</font>
    Wait 1
+
  Wait 1
 
   
 
   
 
  <FONT color=Gray>' ''Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-Off":''</font>
 
  <FONT color=Gray>' ''Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-Off":''</font>
      Write #strOff, 128 As Byte
+
  Write #strOff, 128 As Byte
      Write #strOff, 60 As Byte
+
  Write #strOff, 60 As Byte
      Write #strOff, 0 As Byte
+
  Write #strOff, 0 As Byte
 
   
 
   
    err = snd_rawmidi_write(midiout, strOn, 3)
+
  err = snd_rawmidi_write(midiout, strOn, 3)
    If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))
+
  If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))
 
      
 
      
 
 
  <FONT color=Gray>' ''Alla fine chiudiamo l'output Midi:''</font>
 
  <FONT color=Gray>' ''Alla fine chiudiamo l'output Midi:''</font>
    snd_rawmidi_close(midiout)
+
  snd_rawmidi_close(midiout)
 
      
 
      
  '''End'''
+
  End

Versione attuale delle 14:56, 19 giu 2024

Premessa

Il sistema Alsa consente di inviare dati Midi grezzi ad un dispositivo Midi esterno mediante il suo subsistema RawMidi.


Scrittura dei dati in uscita

Dopo la dichiarazione della Libreria di Alsa e l'apertura del subsistema RawMidi mediante la funzione esterna che in routine potrà essere richiamata in lettura come segue:

err =  snd_rawmidi_open(0, h_midiOut, nome_handle, 0) As Integer

l'altra funzione essenziale è quella che ci permetterà di scrivere i dati nel device del Midi output, ossia di inviare dati al dispositivo esterno:

ssize_t snd_rawmidi_write(snd_rawmidi_t * rawmidi, const void * buffer, size_t size)

che in Gambas sarà dichiarata così:

Private Extern snd_rawmidi_write(out_rmidi As Pointer, buffer As Pointer, size As Integer) As Integer

A questa funzione esterna si passa l'handle, ricevuto dalla funzione snd_rawmidi_open (), l'indirizzo di un buffer contenente alcuni dati MIDI, e la quantità di dati da inviare ogni volta.


Potrebbe sussitere un limite alla quantità di dati che questa funzione è capace di inviare. È possibile determinare quale sia la dimensione dell'output del buffer del driver chiamando la funzione esterna

size_t snd_rawmidi_params_get_buffer_size (const snd_rawmidi_params_t * params)

la quale ritorna apputno la dimensione in byte disponibile del buffer di entrata e di uscita del subsistema RawMidi. Volendo, sarà possibile cambiare quella quantità di byte, chiamando la funzione

int snd_rawmidi_params_set_buffer_size (snd_rawmidi_t * rawmidi, snd_rawmidi_params_t * params, size_t val)


Modalità Blocking e Non-Blocking

Di norma, un uscita Midi viene aperta in modalità bloccante (Blocking). Ciò significa che, quando vengono passati dei dati Midi alla funzione esterna snd_rawmidi_write(), la stessa non ritornerà alcun valore sino a che quei dati non saranno stati inviati attraverso la porta output.


In alternativa è possibile impostare la modalità di apertura non bloccante (Non-Blocking) attraverso la funzione:

 int snd_rawmidi_nonblock (snd_rawmidi_t *rmidi, int nonblock)

che in Gambas dichiareremo così:

Private Extern snd_rawmidi_nonblock(midiOut As Pointer, nonblock As Integer) as Integer

e che in codice richiameremo così:

err = snd_rawmidi_nonblock(handleOut, 1)

Nella modalità non bloccante la funzione snd_rawmidi_write() scrive nel driver output i dati presenti nel buffer imediatamente, consentendo così al programma di svolgere altre funzioni.

Una cosa da tener presente, quando si imposta la modalità Non-Blocking, è che se viene chiamata la funzione snd_rawmidi_close () per chiudere una porta output del subsitema RawMidi, immediatamente dopo aver utilizzato la funzione snd_rawmidi_write (), potrebbero perdersi alcuni dati ancora presenti nel driver.
Per ovviare a questo problema è opportuno chiamare la funzione esterna:

int snd_rawmidi_drain (snd_rawmidi_t *rmidi)


Chiudere il dispositivo Output

Per chiudere la porta di uscita dell'interfaccia RawMidi, si utilizza semplicemente la funzione esterna

int snd_rawmidi_close (snd_rawmidi_t *rmidi)

che in Gambas dichiareremo così:

Private Extern snd_rawmidi_close(midiOut As Pointer) as Integer

e che in codice richiameremo così:

err = snd_rawmidi_close(handleOut)


Esempio

Di seguito un semplice esempio di invio di due Messaggi Midi: un Note-On ed un Note-Off al device output Midi dell'interfaccia RawMidi di Alsa:

Library "libasound:2.0.0"

' int snd_rawmidi_open (snd_rawmidi_t **in_rmidi, snd_rawmidi_t **out_rmidi, const char *name, int mode)  --> Opens a New Connection To the RawMidi interface
Private Extern snd_rawmidi_open(in_rmidi As Pointer, out_rmidi As Pointer, name As String, mode As Integer) As Integer

' ssize_t snd_rawmidi_write (snd_rawmidi_t *rmidi, const void *buffer, size_t size)  --> Write MIDI bytes To MIDI stream
Private Extern snd_rawmidi_write(rmidi As Pointer, buffer As Stream, size As Integer) As Integer

' const char * snd_strerror (int errnum)  --> Returns the message For an Error code 
Private Extern snd_strerror(errnum As Integer) As String

' int snd_rawmidi_close (snd_rawmidi_t *rmidi)  --> Close RawMidi handle
Private Extern snd_rawmidi_close(rmidi As Pointer) As Integer


Public Sub Form_Open()

  Dim posMid, err As Integer
  Dim SND_RAWMIDI_SYNC As Byte = 4
  Dim midiout, noteon, noteoff As Pointer
  Dim nome_porta As String
  Dim strOn, strOff As Stream

' Algoritmo per individuare automaticamente il nome dell'handle dell'interfaccia "RawMidi":
  Print "File-device = "; Dir("/dev/snd/", "midiC*", gb.Device)[0]
  posMid = Mid(Dir("/dev/snd/", "midiC*", gb.Device)[0], 6, 1)

  nome_porta = "hw:" & posMid & ",0,0"
  Print "Nome del dispositivo Midi di Alsa: "; nome_porta

  err = snd_rawmidi_open(0, VarPtr(midiout), nome_porta, SND_RAWMIDI_SYNC)
  If err < 0 Then Error.Raise("Impossibile aprire la porta d'uscita del subsistema RawMidi: " & snd_strerror())
   
' Come mero esempio invieremo tramite il Midi output solo due Messaggi Midi: un NoteOn ed un NoteOff.
' Bisognerà inviare tre byte di dati per volta alla funzione "snd_rawmidi_write ()".
' Poiché è previsto un Pointer, per riempirlo con i tre byte, dovremo creare un'area allocata di memoria, e riempirla mediante la trasformazione dei Pointer in variabili di tipo "Stream", che passeremo alla predetta funzione:
  noteon = Alloc(3)
  noteoff = Alloc(3)
   
  strOn = Memory noteon For Write
  strOff = Memory noteoff For Write
   
' Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-On":
  Write #strOn, 144 As Byte
  Write #strOn, 60 As Byte
  Write #strOn, 100 As Byte
   
' Il valore 3 sta a significare che saranno scritti nel dispositivo Midi output 3 byte fra quelli presenti nella variabile "stream", ossia i tre valori costitutivi del Messaggio Note On:
  err = snd_rawmidi_write(midiout, strOn, 3)
  If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))

' Attendiamo che la nota suoni per 1 secondo:
  Wait 1

' Scriviamo nella memoria allocata, puntata ora da una variabile di tipo "Stream", un Messaggio "Note-Off":
  Write #strOff, 128 As Byte
  Write #strOff, 60 As Byte
  Write #strOff, 0 As Byte

  err = snd_rawmidi_write(midiout, strOn, 3)
  If err < 0 Then Error.Raise("Impossibile la scrittura al Midi output: " & snd_strerror(err))
   
' Alla fine chiudiamo l'output Midi:
  snd_rawmidi_close(midiout)
   
End