2019-05-27 08:55:05 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
2007-10-15 09:50:19 +02:00
* Copyright ( c ) by Jaroslav Kysela < perex @ perex . cz >
2005-04-16 15:20:36 -07:00
* Routines for control of SoundBlaster cards - MIDI interface
*
* - -
*
* Sun May 9 22 : 54 : 38 BST 1999 George David Morrison < gdm @ gedamo . demon . co . uk >
* Fixed typo in snd_sb8dsp_midi_new_device which prevented midi from
* working .
*
* Sun May 11 12 : 34 : 56 UTC 2003 Clemens Ladisch < clemens @ ladisch . de >
* Added full duplex UART mode for DSP version 2.0 and later .
*/
2015-01-28 16:49:33 +01:00
# include <linux/io.h>
2005-04-16 15:20:36 -07:00
# include <linux/time.h>
# include <sound/core.h>
# include <sound/sb.h>
2006-06-13 11:58:12 +02:00
irqreturn_t snd_sb8dsp_midi_interrupt ( struct snd_sb * chip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 14:34:36 +01:00
struct snd_rawmidi * rmidi ;
2005-04-16 15:20:36 -07:00
int max = 64 ;
char byte ;
2006-06-13 11:58:12 +02:00
if ( ! chip )
return IRQ_NONE ;
rmidi = chip - > rmidi ;
if ( ! rmidi ) {
2005-04-16 15:20:36 -07:00
inb ( SBP ( chip , DATA_AVAIL ) ) ; /* ack interrupt */
return IRQ_NONE ;
}
2006-06-13 11:58:12 +02:00
2005-04-16 15:20:36 -07:00
spin_lock ( & chip - > midi_input_lock ) ;
while ( max - - > 0 ) {
if ( inb ( SBP ( chip , DATA_AVAIL ) ) & 0x80 ) {
byte = inb ( SBP ( chip , READ ) ) ;
if ( chip - > open & SB_OPEN_MIDI_INPUT_TRIGGER ) {
snd_rawmidi_receive ( chip - > midi_substream_input , & byte , 1 ) ;
}
}
}
spin_unlock ( & chip - > midi_input_lock ) ;
return IRQ_HANDLED ;
}
2005-11-17 14:34:36 +01:00
static int snd_sb8dsp_midi_input_open ( struct snd_rawmidi_substream * substream )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
unsigned int valid_open_flags ;
chip = substream - > rmidi - > private_data ;
valid_open_flags = chip - > hardware > = SB_HW_20
? SB_OPEN_MIDI_OUTPUT | SB_OPEN_MIDI_OUTPUT_TRIGGER : 0 ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
if ( chip - > open & ~ valid_open_flags ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
return - EAGAIN ;
}
chip - > open | = SB_OPEN_MIDI_INPUT ;
chip - > midi_substream_input = substream ;
if ( ! ( chip - > open & SB_OPEN_MIDI_OUTPUT ) ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
snd_sbdsp_reset ( chip ) ; /* reset DSP */
if ( chip - > hardware > = SB_HW_20 )
snd_sbdsp_command ( chip , SB_DSP_MIDI_UART_IRQ ) ;
} else {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
return 0 ;
}
2005-11-17 14:34:36 +01:00
static int snd_sb8dsp_midi_output_open ( struct snd_rawmidi_substream * substream )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
unsigned int valid_open_flags ;
chip = substream - > rmidi - > private_data ;
valid_open_flags = chip - > hardware > = SB_HW_20
? SB_OPEN_MIDI_INPUT | SB_OPEN_MIDI_INPUT_TRIGGER : 0 ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
if ( chip - > open & ~ valid_open_flags ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
return - EAGAIN ;
}
chip - > open | = SB_OPEN_MIDI_OUTPUT ;
chip - > midi_substream_output = substream ;
if ( ! ( chip - > open & SB_OPEN_MIDI_INPUT ) ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
snd_sbdsp_reset ( chip ) ; /* reset DSP */
if ( chip - > hardware > = SB_HW_20 )
snd_sbdsp_command ( chip , SB_DSP_MIDI_UART_IRQ ) ;
} else {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
return 0 ;
}
2005-11-17 14:34:36 +01:00
static int snd_sb8dsp_midi_input_close ( struct snd_rawmidi_substream * substream )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
chip = substream - > rmidi - > private_data ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
chip - > open & = ~ ( SB_OPEN_MIDI_INPUT | SB_OPEN_MIDI_INPUT_TRIGGER ) ;
chip - > midi_substream_input = NULL ;
if ( ! ( chip - > open & SB_OPEN_MIDI_OUTPUT ) ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
snd_sbdsp_reset ( chip ) ; /* reset DSP */
} else {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
return 0 ;
}
2005-11-17 14:34:36 +01:00
static int snd_sb8dsp_midi_output_close ( struct snd_rawmidi_substream * substream )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
chip = substream - > rmidi - > private_data ;
2017-10-25 10:00:43 +02:00
del_timer_sync ( & chip - > midi_timer ) ;
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
chip - > open & = ~ ( SB_OPEN_MIDI_OUTPUT | SB_OPEN_MIDI_OUTPUT_TRIGGER ) ;
chip - > midi_substream_output = NULL ;
if ( ! ( chip - > open & SB_OPEN_MIDI_INPUT ) ) {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
snd_sbdsp_reset ( chip ) ; /* reset DSP */
} else {
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
return 0 ;
}
2005-11-17 14:34:36 +01:00
static void snd_sb8dsp_midi_input_trigger ( struct snd_rawmidi_substream * substream , int up )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
chip = substream - > rmidi - > private_data ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
if ( up ) {
if ( ! ( chip - > open & SB_OPEN_MIDI_INPUT_TRIGGER ) ) {
if ( chip - > hardware < SB_HW_20 )
snd_sbdsp_command ( chip , SB_DSP_MIDI_INPUT_IRQ ) ;
chip - > open | = SB_OPEN_MIDI_INPUT_TRIGGER ;
}
} else {
if ( chip - > open & SB_OPEN_MIDI_INPUT_TRIGGER ) {
if ( chip - > hardware < SB_HW_20 )
snd_sbdsp_command ( chip , SB_DSP_MIDI_INPUT_IRQ ) ;
chip - > open & = ~ SB_OPEN_MIDI_INPUT_TRIGGER ;
}
}
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
2005-11-17 14:34:36 +01:00
static void snd_sb8dsp_midi_output_write ( struct snd_rawmidi_substream * substream )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
char byte ;
int max = 32 ;
/* how big is Tx FIFO? */
chip = substream - > rmidi - > private_data ;
while ( max - - > 0 ) {
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
if ( snd_rawmidi_transmit_peek ( substream , & byte , 1 ) ! = 1 ) {
chip - > open & = ~ SB_OPEN_MIDI_OUTPUT_TRIGGER ;
del_timer ( & chip - > midi_timer ) ;
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
break ;
}
if ( chip - > hardware > = SB_HW_20 ) {
int timeout = 8 ;
while ( ( inb ( SBP ( chip , STATUS ) ) & 0x80 ) ! = 0 & & - - timeout > 0 )
;
if ( timeout = = 0 ) {
/* Tx FIFO full - try again later */
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
break ;
}
outb ( byte , SBP ( chip , WRITE ) ) ;
} else {
snd_sbdsp_command ( chip , SB_DSP_MIDI_OUTPUT ) ;
snd_sbdsp_command ( chip , byte ) ;
}
snd_rawmidi_transmit_ack ( substream , 1 ) ;
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
}
}
2017-10-24 08:34:58 -07:00
static void snd_sb8dsp_midi_output_timer ( struct timer_list * t )
2005-04-16 15:20:36 -07:00
{
2017-10-24 08:34:58 -07:00
struct snd_sb * chip = from_timer ( chip , t , midi_timer ) ;
struct snd_rawmidi_substream * substream = chip - > midi_substream_output ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
2015-01-19 11:29:42 +01:00
mod_timer ( & chip - > midi_timer , 1 + jiffies ) ;
2005-04-16 15:20:36 -07:00
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
snd_sb8dsp_midi_output_write ( substream ) ;
}
2005-11-17 14:34:36 +01:00
static void snd_sb8dsp_midi_output_trigger ( struct snd_rawmidi_substream * substream , int up )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2005-11-17 14:34:36 +01:00
struct snd_sb * chip ;
2005-04-16 15:20:36 -07:00
chip = substream - > rmidi - > private_data ;
spin_lock_irqsave ( & chip - > open_lock , flags ) ;
if ( up ) {
if ( ! ( chip - > open & SB_OPEN_MIDI_OUTPUT_TRIGGER ) ) {
2015-01-19 11:29:42 +01:00
mod_timer ( & chip - > midi_timer , 1 + jiffies ) ;
2005-04-16 15:20:36 -07:00
chip - > open | = SB_OPEN_MIDI_OUTPUT_TRIGGER ;
}
} else {
if ( chip - > open & SB_OPEN_MIDI_OUTPUT_TRIGGER ) {
chip - > open & = ~ SB_OPEN_MIDI_OUTPUT_TRIGGER ;
}
}
spin_unlock_irqrestore ( & chip - > open_lock , flags ) ;
if ( up )
snd_sb8dsp_midi_output_write ( substream ) ;
}
2017-01-05 17:29:07 +01:00
static const struct snd_rawmidi_ops snd_sb8dsp_midi_output =
2005-04-16 15:20:36 -07:00
{
. open = snd_sb8dsp_midi_output_open ,
. close = snd_sb8dsp_midi_output_close ,
. trigger = snd_sb8dsp_midi_output_trigger ,
} ;
2017-01-05 17:29:07 +01:00
static const struct snd_rawmidi_ops snd_sb8dsp_midi_input =
2005-04-16 15:20:36 -07:00
{
. open = snd_sb8dsp_midi_input_open ,
. close = snd_sb8dsp_midi_input_close ,
. trigger = snd_sb8dsp_midi_input_trigger ,
} ;
2015-01-02 12:24:41 +01:00
int snd_sb8dsp_midi ( struct snd_sb * chip , int device )
2005-04-16 15:20:36 -07:00
{
2005-11-17 14:34:36 +01:00
struct snd_rawmidi * rmidi ;
2005-04-16 15:20:36 -07:00
int err ;
2021-06-08 16:04:35 +02:00
err = snd_rawmidi_new ( chip - > card , " SB8 MIDI " , device , 1 , 1 , & rmidi ) ;
if ( err < 0 )
2005-04-16 15:20:36 -07:00
return err ;
strcpy ( rmidi - > name , " SB8 MIDI " ) ;
snd_rawmidi_set_ops ( rmidi , SNDRV_RAWMIDI_STREAM_OUTPUT , & snd_sb8dsp_midi_output ) ;
snd_rawmidi_set_ops ( rmidi , SNDRV_RAWMIDI_STREAM_INPUT , & snd_sb8dsp_midi_input ) ;
rmidi - > info_flags | = SNDRV_RAWMIDI_INFO_OUTPUT | SNDRV_RAWMIDI_INFO_INPUT ;
if ( chip - > hardware > = SB_HW_20 )
rmidi - > info_flags | = SNDRV_RAWMIDI_INFO_DUPLEX ;
rmidi - > private_data = chip ;
2017-10-25 10:00:43 +02:00
timer_setup ( & chip - > midi_timer , snd_sb8dsp_midi_output_timer , 0 ) ;
2005-04-16 15:20:36 -07:00
chip - > rmidi = rmidi ;
return 0 ;
}