1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
/* mix.c - A very basic mixer.
*
* Copyright 2014 Brad Conroy, dedicated to the Public Domain.
*
USE_MIX(NEWTOY(mix, "c:d:l#r#", TOYFLAG_USR|TOYFLAG_BIN))
config MIX
bool "mix"
default y
help
usage: mix [-d DEV] [-c CHANNEL] [-l VOL] [-r RIGHT]
List OSS sound channels (module snd-mixer-oss), or set volume(s).
-c CHANNEL Set/show volume of CHANNEL (default first channel found)
-d DEV Device node (default /dev/mixer)
-l VOL Volume level
-r RIGHT Volume of right stereo channel (with -r, -l sets left volume)
*/
#define FOR_mix
#include "toys.h"
#include <linux/soundcard.h>
GLOBALS(
long right;
long level;
char *dev;
char *chan;
)
void mix_main(void)
{
const char *channels[SOUND_MIXER_NRDEVICES] = SOUND_DEVICE_NAMES;
int mask, channel = -1, level, fd;
if (!TT.dev) TT.dev = "/dev/mixer";
fd = xopen(TT.dev, O_RDWR|O_NONBLOCK);
xioctl(fd, SOUND_MIXER_READ_DEVMASK, &mask);
for (channel = 0; channel < SOUND_MIXER_NRDEVICES; channel++) {
if ((1<<channel) & mask) {
if (TT.chan) {
if (!strcmp(channels[channel], TT.chan)) break;
} else if (toys.optflags & FLAG_l) break;
else printf("%s\n", channels[channel]);
}
}
if (!(toys.optflags & (FLAG_c|FLAG_l))) return;
else if (channel == SOUND_MIXER_NRDEVICES) error_exit("bad -c '%s'", TT.chan);
if (!(toys.optflags & FLAG_l)) {
xioctl(fd, MIXER_READ(channel), &level);
if (level > 0xFF)
xprintf("%s:%s = left:%d\t right:%d\n",
TT.dev, channels[channel], level>>8, level & 0xFF);
else xprintf("%s:%s = %d\n", TT.dev, channels[channel], level);
} else {
level = TT.level;
if (!(toys.optflags & FLAG_r)) level = TT.right | (level<<8);
xioctl(fd, MIXER_WRITE(channel), &level);
}
if (CFG_TOYBOX_FREE) close(fd);
}
|