#include <string.h>
#include <stdlib.h>
#include <stdio.h>
+#include <err.h>
#include <yajl/yajl_gen.h>
#include <yajl/yajl_version.h>
#ifdef __OpenBSD__
#include <fcntl.h>
#include <unistd.h>
-#include <soundcard.h>
+#include <sys/audioio.h>
+#include <sys/ioctl.h>
#endif
#include "i3status.h"
#include "queue.h"
+static char *apply_volume_format(const char *fmt, char *outwalk, int ivolume) {
+ const char *walk = fmt;
+
+ for (; *walk != '\0'; walk++) {
+ if (*walk != '%') {
+ *(outwalk++) = *walk;
+ continue;
+ }
+ if (BEGINS_WITH(walk + 1, "%")) {
+ outwalk += sprintf(outwalk, "%s", pct_mark);
+ walk += strlen("%");
+ }
+ if (BEGINS_WITH(walk + 1, "volume")) {
+ outwalk += sprintf(outwalk, "%d%s", ivolume, pct_mark);
+ walk += strlen("volume");
+ }
+ }
+ return outwalk;
+}
+
void print_volume(yajl_gen json_gen, char *buffer, const char *fmt, const char *fmt_muted, const char *device, const char *mixer, int mixer_idx) {
char *outwalk = buffer;
int pbval = 1;
- /* Printing volume only works with ALSA at the moment */
+ /* Printing volume works with ALSA and PulseAudio at the moment */
if (output_format == O_I3BAR) {
char *instance;
asprintf(&instance, "%s.%s.%d", device, mixer, mixer_idx);
INSTANCE(instance);
free(instance);
}
+
+#ifndef __OpenBSD__
+ /* Try PulseAudio first */
+
+ /* If the device name has the format "pulse[:N]" where N is the
+ * index of the PulseAudio sink then force PulseAudio, optionally
+ * overriding the default sink */
+ if (!strncasecmp(device, "pulse", strlen("pulse"))) {
+ uint32_t sink_idx = device[5] == ':' ? (uint32_t)atoi(device + 6)
+ : DEFAULT_SINK_INDEX;
+ int cvolume = pulse_initialize() ? volume_pulseaudio(sink_idx) : 0;
+ int ivolume = DECOMPOSE_VOLUME(cvolume);
+ bool muted = DECOMPOSE_MUTED(cvolume);
+ if (muted) {
+ START_COLOR("color_degraded");
+ pbval = 0;
+ }
+ /* negative result means error, stick to 0 */
+ if (ivolume < 0)
+ ivolume = 0;
+ outwalk = apply_volume_format(muted ? fmt_muted : fmt,
+ outwalk,
+ ivolume);
+ goto out;
+ } else if (!strcasecmp(device, "default") && pulse_initialize()) {
+ /* no device specified or "default" set */
+ int cvolume = volume_pulseaudio(DEFAULT_SINK_INDEX);
+ int ivolume = DECOMPOSE_VOLUME(cvolume);
+ bool muted = DECOMPOSE_MUTED(cvolume);
+ if (ivolume >= 0) {
+ if (muted) {
+ START_COLOR("color_degraded");
+ pbval = 0;
+ }
+ outwalk = apply_volume_format(muted ? fmt_muted : fmt,
+ outwalk,
+ ivolume);
+ goto out;
+ }
+ /* negative result means error, fail PulseAudio attempt */
+ }
+/* If some other device was specified or PulseAudio is not detected,
+ * proceed to ALSA / OSS */
+#endif
+
#ifdef LINUX
int err;
snd_mixer_t *m;
snd_mixer_close(m);
snd_mixer_selem_id_free(sid);
- const char *walk = fmt;
- for (; *walk != '\0'; walk++) {
- if (*walk != '%') {
- *(outwalk++) = *walk;
- continue;
- }
- if (BEGINS_WITH(walk + 1, "%")) {
- outwalk += sprintf(outwalk, "%%");
- walk += strlen("%");
- }
- if (BEGINS_WITH(walk + 1, "volume")) {
- outwalk += sprintf(outwalk, "%d%%", avg);
- walk += strlen("volume");
- }
- }
+ outwalk = apply_volume_format(fmt, outwalk, avg);
+
#endif
#if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__DragonFly__)
char *mixerpath;
else
mixerpath = defaultmixer;
- if ((mixfd = open(mixerpath, O_RDWR)) < 0)
- return;
+ if ((mixfd = open(mixerpath, O_RDWR)) < 0) {
+#if defined(__OpenBSD__)
+ warn("audioio: Cannot open mixer");
+#else
+ warn("OSS: Cannot open mixer");
+#endif
+ goto out;
+ }
if (mixer_idx > 0)
free(mixerpath);
- if (ioctl(mixfd, SOUND_MIXER_READ_DEVMASK, &devmask) == -1)
- return;
- if (ioctl(mixfd, MIXER_READ(0), &vol) == -1)
- return;
+#if defined(__OpenBSD__)
+ int oclass_idx = -1, master_idx = -1, master_mute_idx = -1;
+ mixer_devinfo_t devinfo, devinfo2;
+ mixer_ctrl_t vinfo;
- if (((vol & 0x7f) == 0) && (((vol >> 8) & 0x7f) == 0)) {
+ devinfo.index = 0;
+ while (ioctl(mixfd, AUDIO_MIXER_DEVINFO, &devinfo) >= 0) {
+ if (devinfo.type != AUDIO_MIXER_CLASS) {
+ devinfo.index++;
+ continue;
+ }
+ if (strncmp(devinfo.label.name, AudioCoutputs, MAX_AUDIO_DEV_LEN) == 0)
+ oclass_idx = devinfo.index;
+
+ devinfo.index++;
+ }
+
+ devinfo2.index = 0;
+ while (ioctl(mixfd, AUDIO_MIXER_DEVINFO, &devinfo2) >= 0) {
+ if ((devinfo2.type == AUDIO_MIXER_VALUE) && (devinfo2.mixer_class == oclass_idx) && (strncmp(devinfo2.label.name, AudioNmaster, MAX_AUDIO_DEV_LEN) == 0))
+ master_idx = devinfo2.index;
+
+ if ((devinfo2.type == AUDIO_MIXER_ENUM) && (devinfo2.mixer_class == oclass_idx) && (strncmp(devinfo2.label.name, AudioNmute, MAX_AUDIO_DEV_LEN) == 0))
+ master_mute_idx = devinfo2.index;
+
+ devinfo2.index++;
+ }
+
+ if (master_idx == -1)
+ goto out;
+
+ devinfo.index = master_idx;
+ if (ioctl(mixfd, AUDIO_MIXER_DEVINFO, &devinfo) == -1)
+ goto out;
+
+ vinfo.dev = master_idx;
+ vinfo.type = AUDIO_MIXER_VALUE;
+ if (ioctl(mixfd, AUDIO_MIXER_READ, &vinfo) == -1)
+ goto out;
+
+ if (AUDIO_MAX_GAIN != 100) {
+ float avgf = ((float)vinfo.un.value.level[AUDIO_MIXER_LEVEL_MONO] / AUDIO_MAX_GAIN) * 100;
+ vol = (int)avgf;
+ vol = (avgf - vol < 0.5 ? vol : (vol + 1));
+ } else {
+ vol = (int)vinfo.un.value.level[AUDIO_MIXER_LEVEL_MONO];
+ }
+
+ vinfo.dev = master_mute_idx;
+ vinfo.type = AUDIO_MIXER_ENUM;
+ if (ioctl(mixfd, AUDIO_MIXER_READ, &vinfo) == -1)
+ goto out;
+
+ if (master_mute_idx != -1 && vinfo.un.ord) {
START_COLOR("color_degraded");
+ fmt = fmt_muted;
pbval = 0;
}
- const char *walk = fmt;
- for (; *walk != '\0'; walk++) {
- if (*walk != '%') {
- *(outwalk++) = *walk;
- continue;
- }
- if (BEGINS_WITH(walk + 1, "%")) {
- outwalk += sprintf(outwalk, "%%");
- walk += strlen("%");
- }
- if (BEGINS_WITH(walk + 1, "volume")) {
- outwalk += sprintf(outwalk, "%d%%", vol & 0x7f);
- walk += strlen("volume");
- }
+#else
+ if (ioctl(mixfd, SOUND_MIXER_READ_DEVMASK, &devmask) == -1) {
+ warn("OSS: Cannot read mixer information");
+ goto out;
+ }
+ if (ioctl(mixfd, MIXER_READ(0), &vol) == -1) {
+ warn("OSS: Cannot read mixer information");
+ goto out;
+ }
+
+ if (((vol & 0x7f) == 0) && (((vol >> 8) & 0x7f) == 0)) {
+ START_COLOR("color_degraded");
+ pbval = 0;
}
+
+#endif
+ outwalk = apply_volume_format(fmt, outwalk, vol & 0x7f);
close(mixfd);
#endif