1 /* 2 * dice_midi.c - a part of driver for Dice based devices 3 * 4 * Copyright (c) 2014 Takashi Sakamoto 5 * 6 * Licensed under the terms of the GNU General Public License, version 2. 7 */ 8 #include "dice.h" 9 10 static int midi_open(struct snd_rawmidi_substream *substream) 11 { 12 struct snd_dice *dice = substream->rmidi->private_data; 13 int err; 14 15 err = snd_dice_stream_lock_try(dice); 16 if (err < 0) 17 return err; 18 19 mutex_lock(&dice->mutex); 20 21 err = snd_dice_stream_reserve_duplex(dice, 0); 22 if (err >= 0) { 23 ++dice->substreams_counter; 24 err = snd_dice_stream_start_duplex(dice); 25 } 26 27 mutex_unlock(&dice->mutex); 28 29 if (err < 0) 30 snd_dice_stream_lock_release(dice); 31 32 return err; 33 } 34 35 static int midi_close(struct snd_rawmidi_substream *substream) 36 { 37 struct snd_dice *dice = substream->rmidi->private_data; 38 39 mutex_lock(&dice->mutex); 40 41 --dice->substreams_counter; 42 snd_dice_stream_stop_duplex(dice); 43 snd_dice_stream_release_duplex(dice); 44 45 mutex_unlock(&dice->mutex); 46 47 snd_dice_stream_lock_release(dice); 48 return 0; 49 } 50 51 static void midi_capture_trigger(struct snd_rawmidi_substream *substrm, int up) 52 { 53 struct snd_dice *dice = substrm->rmidi->private_data; 54 unsigned long flags; 55 56 spin_lock_irqsave(&dice->lock, flags); 57 58 if (up) 59 amdtp_am824_midi_trigger(&dice->tx_stream[0], 60 substrm->number, substrm); 61 else 62 amdtp_am824_midi_trigger(&dice->tx_stream[0], 63 substrm->number, NULL); 64 65 spin_unlock_irqrestore(&dice->lock, flags); 66 } 67 68 static void midi_playback_trigger(struct snd_rawmidi_substream *substrm, int up) 69 { 70 struct snd_dice *dice = substrm->rmidi->private_data; 71 unsigned long flags; 72 73 spin_lock_irqsave(&dice->lock, flags); 74 75 if (up) 76 amdtp_am824_midi_trigger(&dice->rx_stream[0], 77 substrm->number, substrm); 78 else 79 amdtp_am824_midi_trigger(&dice->rx_stream[0], 80 substrm->number, NULL); 81 82 spin_unlock_irqrestore(&dice->lock, flags); 83 } 84 85 static void set_midi_substream_names(struct snd_dice *dice, 86 struct snd_rawmidi_str *str) 87 { 88 struct snd_rawmidi_substream *subs; 89 90 list_for_each_entry(subs, &str->substreams, list) { 91 snprintf(subs->name, sizeof(subs->name), 92 "%s MIDI %d", dice->card->shortname, subs->number + 1); 93 } 94 } 95 96 int snd_dice_create_midi(struct snd_dice *dice) 97 { 98 static const struct snd_rawmidi_ops capture_ops = { 99 .open = midi_open, 100 .close = midi_close, 101 .trigger = midi_capture_trigger, 102 }; 103 static const struct snd_rawmidi_ops playback_ops = { 104 .open = midi_open, 105 .close = midi_close, 106 .trigger = midi_playback_trigger, 107 }; 108 struct snd_rawmidi *rmidi; 109 struct snd_rawmidi_str *str; 110 unsigned int midi_in_ports, midi_out_ports; 111 int i; 112 int err; 113 114 midi_in_ports = 0; 115 midi_out_ports = 0; 116 for (i = 0; i < MAX_STREAMS; ++i) { 117 midi_in_ports = max(midi_in_ports, dice->tx_midi_ports[i]); 118 midi_out_ports = max(midi_out_ports, dice->rx_midi_ports[i]); 119 } 120 121 if (midi_in_ports + midi_out_ports == 0) 122 return 0; 123 124 /* create midi ports */ 125 err = snd_rawmidi_new(dice->card, dice->card->driver, 0, 126 midi_out_ports, midi_in_ports, 127 &rmidi); 128 if (err < 0) 129 return err; 130 131 snprintf(rmidi->name, sizeof(rmidi->name), 132 "%s MIDI", dice->card->shortname); 133 rmidi->private_data = dice; 134 135 if (midi_in_ports > 0) { 136 rmidi->info_flags |= SNDRV_RAWMIDI_INFO_INPUT; 137 138 snd_rawmidi_set_ops(rmidi, SNDRV_RAWMIDI_STREAM_INPUT, 139 &capture_ops); 140 141 str = &rmidi->streams[SNDRV_RAWMIDI_STREAM_INPUT]; 142 143 set_midi_substream_names(dice, str); 144 } 145 146 if (midi_out_ports > 0) { 147 rmidi->info_flags |= SNDRV_RAWMIDI_INFO_OUTPUT; 148 149 snd_rawmidi_set_ops(rmidi, SNDRV_RAWMIDI_STREAM_OUTPUT, 150 &playback_ops); 151 152 str = &rmidi->streams[SNDRV_RAWMIDI_STREAM_OUTPUT]; 153 154 set_midi_substream_names(dice, str); 155 } 156 157 if ((midi_out_ports > 0) && (midi_in_ports > 0)) 158 rmidi->info_flags |= SNDRV_RAWMIDI_INFO_DUPLEX; 159 160 return 0; 161 } 162