Skip to content

Commit 6aea570

Browse files
committed
ALSA: rawmidi: A lightweight function to discard pending bytes
For discarding the pending bytes on rawmidi, we process with a loop of snd_rawmidi_transmit() which is just a waste of CPU power. Implement a lightweight API function to discard the pending bytes and the proceed the ring buffer instantly, and use it instead of open codes. Signed-off-by: Takashi Iwai <[email protected]>
1 parent cd3b711 commit 6aea570

File tree

4 files changed

+25
-5
lines changed

4 files changed

+25
-5
lines changed

include/sound/rawmidi.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -171,6 +171,7 @@ int __snd_rawmidi_transmit_peek(struct snd_rawmidi_substream *substream,
171171
unsigned char *buffer, int count);
172172
int __snd_rawmidi_transmit_ack(struct snd_rawmidi_substream *substream,
173173
int count);
174+
int snd_rawmidi_proceed(struct snd_rawmidi_substream *substream);
174175

175176
/* main midi functions */
176177

sound/core/rawmidi.c

Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1236,6 +1236,28 @@ int snd_rawmidi_transmit(struct snd_rawmidi_substream *substream,
12361236
}
12371237
EXPORT_SYMBOL(snd_rawmidi_transmit);
12381238

1239+
/**
1240+
* snd_rawmidi_proceed - Discard the all pending bytes and proceed
1241+
* @substream: rawmidi substream
1242+
*
1243+
* Return: the number of discarded bytes
1244+
*/
1245+
int snd_rawmidi_proceed(struct snd_rawmidi_substream *substream)
1246+
{
1247+
struct snd_rawmidi_runtime *runtime = substream->runtime;
1248+
unsigned long flags;
1249+
int count = 0;
1250+
1251+
spin_lock_irqsave(&runtime->lock, flags);
1252+
if (runtime->avail < runtime->buffer_size) {
1253+
count = runtime->buffer_size - runtime->avail;
1254+
__snd_rawmidi_transmit_ack(substream, count);
1255+
}
1256+
spin_unlock_irqrestore(&runtime->lock, flags);
1257+
return count;
1258+
}
1259+
EXPORT_SYMBOL(snd_rawmidi_proceed);
1260+
12391261
static long snd_rawmidi_kernel_write1(struct snd_rawmidi_substream *substream,
12401262
const unsigned char __user *userbuf,
12411263
const unsigned char *kernelbuf,

sound/core/seq/seq_virmidi.c

Lines changed: 1 addition & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -149,9 +149,7 @@ static void snd_vmidi_output_work(struct work_struct *work)
149149
/* discard the outputs in dispatch mode unless subscribed */
150150
if (vmidi->seq_mode == SNDRV_VIRMIDI_SEQ_DISPATCH &&
151151
!(vmidi->rdev->flags & SNDRV_VIRMIDI_SUBSCRIBE)) {
152-
char buf[32];
153-
while (snd_rawmidi_transmit(substream, buf, sizeof(buf)) > 0)
154-
; /* ignored */
152+
snd_rawmidi_proceed(substream);
155153
return;
156154
}
157155

sound/usb/midi.c

Lines changed: 1 addition & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1175,8 +1175,7 @@ static void snd_usbmidi_output_trigger(struct snd_rawmidi_substream *substream,
11751175
if (port->ep->umidi->disconnected) {
11761176
/* gobble up remaining bytes to prevent wait in
11771177
* snd_rawmidi_drain_output */
1178-
while (!snd_rawmidi_transmit_empty(substream))
1179-
snd_rawmidi_transmit_ack(substream, 1);
1178+
snd_rawmidi_proceed(substream);
11801179
return;
11811180
}
11821181
tasklet_schedule(&port->ep->tasklet);

0 commit comments

Comments
 (0)