Inviare dati grezzi Midi attraverso il subsistema RawMidi
Indice
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