winealsa.drv: Use separate port for each output device.
This fixes use of multiple MIDI outputs while keeping the flexibility of proper ALSA sequencer port subscriptions. Wine-Bug: https://bugs.winehq.org/show_bug.cgi?id=52220 Signed-off-by: Jacek Konieczny <jajcus@jajcus.net> Signed-off-by: Andrew Eikum <aeikum@codeweavers.com> Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
parent
3e73437ae8
commit
240f2f3963
|
@ -83,6 +83,7 @@ typedef struct {
|
||||||
void* lpExtra; /* according to port type (MIDI, FM...), extra data when needed */
|
void* lpExtra; /* according to port type (MIDI, FM...), extra data when needed */
|
||||||
MIDIOUTCAPSW caps;
|
MIDIOUTCAPSW caps;
|
||||||
snd_seq_addr_t addr;
|
snd_seq_addr_t addr;
|
||||||
|
int port_out;
|
||||||
} WINE_MIDIOUT;
|
} WINE_MIDIOUT;
|
||||||
|
|
||||||
static WINE_MIDIIN MidiInDev [MAX_MIDIINDRV ];
|
static WINE_MIDIIN MidiInDev [MAX_MIDIINDRV ];
|
||||||
|
@ -106,7 +107,6 @@ static int numOpenMidiSeq = 0;
|
||||||
static int numStartedMidiIn = 0;
|
static int numStartedMidiIn = 0;
|
||||||
|
|
||||||
static int port_in;
|
static int port_in;
|
||||||
static int port_out;
|
|
||||||
|
|
||||||
static CRITICAL_SECTION crit_sect; /* protects all MidiIn buffer queues */
|
static CRITICAL_SECTION crit_sect; /* protects all MidiIn buffer queues */
|
||||||
static CRITICAL_SECTION_DEBUG critsect_debug =
|
static CRITICAL_SECTION_DEBUG critsect_debug =
|
||||||
|
@ -245,14 +245,6 @@ static int midiOpenSeq(BOOL create_client)
|
||||||
/* Setting the client name is the only init to do */
|
/* Setting the client name is the only init to do */
|
||||||
snd_seq_set_client_name(midiSeq, "WINE midi driver");
|
snd_seq_set_client_name(midiSeq, "WINE midi driver");
|
||||||
|
|
||||||
port_out = snd_seq_create_simple_port(midiSeq, "WINE ALSA Output",
|
|
||||||
SND_SEQ_PORT_CAP_READ|SND_SEQ_PORT_CAP_SUBS_READ|SND_SEQ_PORT_CAP_SUBS_WRITE,
|
|
||||||
SND_SEQ_PORT_TYPE_MIDI_GENERIC|SND_SEQ_PORT_TYPE_APPLICATION);
|
|
||||||
if (port_out < 0)
|
|
||||||
TRACE("Unable to create output port\n");
|
|
||||||
else
|
|
||||||
TRACE("Outport port %d created successfully\n", port_out);
|
|
||||||
|
|
||||||
port_in = snd_seq_create_simple_port(midiSeq, "WINE ALSA Input",
|
port_in = snd_seq_create_simple_port(midiSeq, "WINE ALSA Input",
|
||||||
SND_SEQ_PORT_CAP_WRITE|SND_SEQ_PORT_CAP_SUBS_READ|SND_SEQ_PORT_CAP_SUBS_WRITE,
|
SND_SEQ_PORT_CAP_WRITE|SND_SEQ_PORT_CAP_SUBS_READ|SND_SEQ_PORT_CAP_SUBS_WRITE,
|
||||||
SND_SEQ_PORT_TYPE_MIDI_GENERIC|SND_SEQ_PORT_TYPE_APPLICATION);
|
SND_SEQ_PORT_TYPE_MIDI_GENERIC|SND_SEQ_PORT_TYPE_APPLICATION);
|
||||||
|
@ -274,7 +266,6 @@ static int midiCloseSeq(void)
|
||||||
{
|
{
|
||||||
EnterCriticalSection(&midiSeqLock);
|
EnterCriticalSection(&midiSeqLock);
|
||||||
if (--numOpenMidiSeq == 0) {
|
if (--numOpenMidiSeq == 0) {
|
||||||
snd_seq_delete_simple_port(midiSeq, port_out);
|
|
||||||
snd_seq_delete_simple_port(midiSeq, port_in);
|
snd_seq_delete_simple_port(midiSeq, port_in);
|
||||||
snd_seq_close(midiSeq);
|
snd_seq_close(midiSeq);
|
||||||
midiSeq = NULL;
|
midiSeq = NULL;
|
||||||
|
@ -736,6 +727,8 @@ static DWORD modGetDevCaps(WORD wDevID, LPMIDIOUTCAPSW lpCaps, DWORD dwSize)
|
||||||
static DWORD modOpen(WORD wDevID, LPMIDIOPENDESC lpDesc, DWORD dwFlags)
|
static DWORD modOpen(WORD wDevID, LPMIDIOPENDESC lpDesc, DWORD dwFlags)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
int port_out;
|
||||||
|
char port_out_name[32];
|
||||||
|
|
||||||
TRACE("(%04X, %p, %08X);\n", wDevID, lpDesc, dwFlags);
|
TRACE("(%04X, %p, %08X);\n", wDevID, lpDesc, dwFlags);
|
||||||
if (lpDesc == NULL) {
|
if (lpDesc == NULL) {
|
||||||
|
@ -782,12 +775,37 @@ static DWORD modOpen(WORD wDevID, LPMIDIOPENDESC lpDesc, DWORD dwFlags)
|
||||||
MidiOutDev[wDevID].bufsize = 0x3FFF;
|
MidiOutDev[wDevID].bufsize = 0x3FFF;
|
||||||
MidiOutDev[wDevID].midiDesc = *lpDesc;
|
MidiOutDev[wDevID].midiDesc = *lpDesc;
|
||||||
|
|
||||||
/* Connect our app port to the device port */
|
|
||||||
EnterCriticalSection(&midiSeqLock);
|
EnterCriticalSection(&midiSeqLock);
|
||||||
|
/* Create a port dedicated to a specific device */
|
||||||
|
/* Keep the old name without a number for the first port */
|
||||||
|
if (wDevID)
|
||||||
|
sprintf(port_out_name, "WINE ALSA Output #%d", wDevID);
|
||||||
|
|
||||||
|
port_out = snd_seq_create_simple_port(midiSeq, wDevID?port_out_name:"WINE ALSA Output",
|
||||||
|
SND_SEQ_PORT_CAP_READ|SND_SEQ_PORT_CAP_SUBS_READ|SND_SEQ_PORT_CAP_SUBS_WRITE,
|
||||||
|
SND_SEQ_PORT_TYPE_MIDI_GENERIC|SND_SEQ_PORT_TYPE_APPLICATION);
|
||||||
|
|
||||||
|
if (port_out < 0) {
|
||||||
|
TRACE("Unable to create output port\n");
|
||||||
|
MidiOutDev[wDevID].port_out = -1;
|
||||||
|
} else {
|
||||||
|
TRACE("Outport port %d created successfully\n", port_out);
|
||||||
|
MidiOutDev[wDevID].port_out = port_out;
|
||||||
|
|
||||||
|
/* Connect our app port to the device port */
|
||||||
ret = snd_seq_connect_to(midiSeq, port_out, MidiOutDev[wDevID].addr.client,
|
ret = snd_seq_connect_to(midiSeq, port_out, MidiOutDev[wDevID].addr.client,
|
||||||
MidiOutDev[wDevID].addr.port);
|
MidiOutDev[wDevID].addr.port);
|
||||||
LeaveCriticalSection(&midiSeqLock);
|
|
||||||
|
/* usually will happen when the port is already connected */
|
||||||
|
/* other errors should not be fatal either */
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
WARN("Could not connect port %d to %d:%d: %s\n",
|
||||||
|
wDevID, MidiOutDev[wDevID].addr.client,
|
||||||
|
MidiOutDev[wDevID].addr.port, snd_strerror(ret));
|
||||||
|
}
|
||||||
|
LeaveCriticalSection(&midiSeqLock);
|
||||||
|
|
||||||
|
if (port_out < 0)
|
||||||
return MMSYSERR_NOTENABLED;
|
return MMSYSERR_NOTENABLED;
|
||||||
|
|
||||||
TRACE("Output port :%d connected %d:%d\n",port_out,MidiOutDev[wDevID].addr.client,MidiOutDev[wDevID].addr.port);
|
TRACE("Output port :%d connected %d:%d\n",port_out,MidiOutDev[wDevID].addr.client,MidiOutDev[wDevID].addr.port);
|
||||||
|
@ -823,7 +841,9 @@ static DWORD modClose(WORD wDevID)
|
||||||
case MOD_MIDIPORT:
|
case MOD_MIDIPORT:
|
||||||
case MOD_SYNTH:
|
case MOD_SYNTH:
|
||||||
EnterCriticalSection(&midiSeqLock);
|
EnterCriticalSection(&midiSeqLock);
|
||||||
snd_seq_disconnect_to(midiSeq, port_out, MidiOutDev[wDevID].addr.client, MidiOutDev[wDevID].addr.port);
|
TRACE("Deleting port :%d, connected to %d:%d\n", MidiOutDev[wDevID].port_out, MidiOutDev[wDevID].addr.client, MidiOutDev[wDevID].addr.port);
|
||||||
|
snd_seq_delete_simple_port(midiSeq, MidiOutDev[wDevID].port_out);
|
||||||
|
MidiOutDev[wDevID].port_out = -1;
|
||||||
LeaveCriticalSection(&midiSeqLock);
|
LeaveCriticalSection(&midiSeqLock);
|
||||||
midiCloseSeq();
|
midiCloseSeq();
|
||||||
break;
|
break;
|
||||||
|
@ -868,7 +888,7 @@ static DWORD modData(WORD wDevID, DWORD dwParam)
|
||||||
snd_seq_event_t event;
|
snd_seq_event_t event;
|
||||||
snd_seq_ev_clear(&event);
|
snd_seq_ev_clear(&event);
|
||||||
snd_seq_ev_set_direct(&event);
|
snd_seq_ev_set_direct(&event);
|
||||||
snd_seq_ev_set_source(&event, port_out);
|
snd_seq_ev_set_source(&event, MidiOutDev[wDevID].port_out);
|
||||||
snd_seq_ev_set_subs(&event);
|
snd_seq_ev_set_subs(&event);
|
||||||
|
|
||||||
switch (evt & 0xF0) {
|
switch (evt & 0xF0) {
|
||||||
|
@ -1037,7 +1057,7 @@ static DWORD modLongData(WORD wDevID, LPMIDIHDR lpMidiHdr, DWORD dwSize)
|
||||||
}
|
}
|
||||||
snd_seq_ev_clear(&event);
|
snd_seq_ev_clear(&event);
|
||||||
snd_seq_ev_set_direct(&event);
|
snd_seq_ev_set_direct(&event);
|
||||||
snd_seq_ev_set_source(&event, port_out);
|
snd_seq_ev_set_source(&event, MidiOutDev[wDevID].port_out);
|
||||||
snd_seq_ev_set_subs(&event);
|
snd_seq_ev_set_subs(&event);
|
||||||
snd_seq_ev_set_sysex(&event, lpMidiHdr->dwBufferLength + len_add, lpNewData ? lpNewData : lpData);
|
snd_seq_ev_set_sysex(&event, lpMidiHdr->dwBufferLength + len_add, lpNewData ? lpNewData : lpData);
|
||||||
EnterCriticalSection(&midiSeqLock);
|
EnterCriticalSection(&midiSeqLock);
|
||||||
|
@ -1199,6 +1219,7 @@ static void ALSA_AddMidiPort(snd_seq_client_info_t* cinfo, snd_seq_port_info_t*
|
||||||
MidiOutDev[MODM_NumDevs].caps.wNotes = 16;
|
MidiOutDev[MODM_NumDevs].caps.wNotes = 16;
|
||||||
}
|
}
|
||||||
MidiOutDev[MODM_NumDevs].bEnabled = TRUE;
|
MidiOutDev[MODM_NumDevs].bEnabled = TRUE;
|
||||||
|
MidiOutDev[MODM_NumDevs].port_out = -1;
|
||||||
|
|
||||||
TRACE("MidiOut[%d]\tname='%s' techn=%d voices=%d notes=%d chnMsk=%04x support=%d\n"
|
TRACE("MidiOut[%d]\tname='%s' techn=%d voices=%d notes=%d chnMsk=%04x support=%d\n"
|
||||||
"\tALSA info: midi dev-type=%x, capa=0\n",
|
"\tALSA info: midi dev-type=%x, capa=0\n",
|
||||||
|
|
Loading…
Reference in New Issue