mirror of
https://github.com/alsa-project/alsa-utils
synced 2025-01-03 15:19:51 +01:00
b1e3d945b7
Signed-off-by: Jaroslav Kysela <perex@perex.cz>
432 lines
11 KiB
C
432 lines
11 KiB
C
/*
|
|
iecset - change IEC958 status bits on ALSA
|
|
Copyright (C) 2003 by Takashi Iwai <tiwai@suse.de>
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <ctype.h>
|
|
#include <alsa/asoundlib.h>
|
|
|
|
void dump_iec958(snd_aes_iec958_t *iec);
|
|
|
|
static int get_bool(const char *str)
|
|
{
|
|
if (strncmp(str, "yes", 3) == 0 ||
|
|
strncmp(str, "YES", 3) == 0 ||
|
|
strncmp(str, "on", 2) == 0 ||
|
|
strncmp(str, "ON", 2) == 0 ||
|
|
strncmp(str, "true", 4) == 0 ||
|
|
strncmp(str, "TRUE", 4) == 0 ||
|
|
*str == '1')
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
enum {
|
|
CMD_BOOL, CMD_BOOL_INV, CMD_INT
|
|
};
|
|
|
|
enum {
|
|
IDX_PRO, IDX_NOAUDIO, IDX_RATE, IDX_UNLOCK, IDX_SBITS, IDX_WORD, IDX_EMP, IDX_CAT, IDX_NOCOPY, IDX_ORIG,
|
|
IDX_LAST
|
|
};
|
|
|
|
struct cmdtbl {
|
|
const char *name;
|
|
int idx;
|
|
int type;
|
|
const char *desc;
|
|
};
|
|
|
|
static const struct cmdtbl cmds[] = {
|
|
{ "pro", IDX_PRO, CMD_BOOL,
|
|
"professional (common)\n\toff = consumer mode, on = professional mode" },
|
|
{ "aud", IDX_NOAUDIO, CMD_BOOL_INV,
|
|
"audio (common)\n\ton = audio mode, off = non-audio mode" },
|
|
{ "rat", IDX_RATE, CMD_INT,
|
|
"rate (common)\n\tsample rate in Hz (0 = not indicated)" },
|
|
{ "emp", IDX_EMP, CMD_INT,
|
|
"emphasis (common)\n\t0 = none, 1 = 50/15us, 2 = CCITT" },
|
|
{ "loc", IDX_UNLOCK, CMD_BOOL_INV,
|
|
"lock (prof.)\n\toff = rate unlocked, on = rate locked" },
|
|
{ "sbi", IDX_SBITS, CMD_INT,
|
|
"sbits (prof.)\n\tsample bits 2 = 20bit, 4 = 24bit, 6 = undef" },
|
|
{ "wor", IDX_WORD, CMD_INT,
|
|
"wordlength (prof.)\n\t0=no, 2=22-18bit, 4=23-19bit, 5=24-20bit, 6=20-16bit" },
|
|
{ "cat", IDX_CAT, CMD_INT,
|
|
"category (consumer)\n\t0-0x7f" },
|
|
{ "cop", IDX_NOCOPY, CMD_BOOL_INV,
|
|
"copyright (consumer)\n\toff = non-copyright, on = copyright" },
|
|
{ "ori", IDX_ORIG, CMD_BOOL,
|
|
"original (consumer)\n\toff = 1st-gen, on = original" },
|
|
};
|
|
|
|
|
|
static void error(const char *s, int err)
|
|
{
|
|
fprintf(stderr, "%s: %s\n", s, snd_strerror(err));
|
|
}
|
|
|
|
|
|
static void usage(void)
|
|
{
|
|
int i;
|
|
|
|
printf("Usage: iecset [options] [cmd arg...]\n");
|
|
printf("Options:\n");
|
|
printf(" -D device specifies the control device to use\n");
|
|
printf(" -c card specifies the card number to use (equiv. with -Dhw:#)\n");
|
|
printf(" -n number specifies the control index number (default = 0)\n");
|
|
printf(" -x dump the dump the AESx hex code for IEC958 PCM parameters\n");
|
|
printf(" -i read commands from stdin\n");
|
|
printf("Commands:\n");
|
|
for (i = 0; i < (int)(sizeof(cmds)/sizeof(cmds[0])); i++) {
|
|
printf(" %s\n", cmds[i].desc);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* parse iecset commands
|
|
*/
|
|
static void parse_command(int *parms, const char *c, const char *arg)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < (int)(sizeof(cmds)/sizeof(cmds[0])); i++) {
|
|
if (strncmp(c, cmds[i].name, strlen(cmds[i].name)) == 0) {
|
|
int val;
|
|
switch (cmds[i].type) {
|
|
case CMD_BOOL:
|
|
val = get_bool(arg);
|
|
break;
|
|
case CMD_BOOL_INV:
|
|
val = !get_bool(arg);
|
|
break;
|
|
case CMD_INT:
|
|
default:
|
|
val = (int)strtol(arg, NULL, 0);
|
|
break;
|
|
}
|
|
parms[cmds[i].idx] = val;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static char *skipspace(char *line)
|
|
{
|
|
char *p;
|
|
for (p = line; *p && isspace(*p); p++)
|
|
;
|
|
return p;
|
|
}
|
|
|
|
/*
|
|
* parse iecset commands from the file
|
|
*/
|
|
static void parse_file(int *parms, FILE *fp)
|
|
{
|
|
char line[1024], *cmd, *arg;
|
|
while (fgets(line, sizeof(line), fp) != NULL) {
|
|
cmd = skipspace(line);
|
|
if (*cmd == '#' || ! *cmd)
|
|
continue;
|
|
for (arg = cmd; *arg && !isspace(*arg); arg++)
|
|
;
|
|
if (! *arg)
|
|
continue;
|
|
*arg++ = 0;
|
|
arg = skipspace(arg);
|
|
if (! *arg)
|
|
continue;
|
|
parse_command(parms, cmd, arg);
|
|
}
|
|
}
|
|
|
|
/* update iec958 status values
|
|
* return non-zero if the values are modified
|
|
*/
|
|
static int update_iec958_status(snd_aes_iec958_t *iec958, int *parms)
|
|
{
|
|
int changed = 0;
|
|
if (parms[IDX_PRO] >= 0) {
|
|
if (parms[IDX_PRO])
|
|
iec958->status[0] |= IEC958_AES0_PROFESSIONAL;
|
|
else
|
|
iec958->status[0] &= ~IEC958_AES0_PROFESSIONAL;
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_NOAUDIO] >= 0) {
|
|
if (parms[IDX_NOAUDIO])
|
|
iec958->status[0] |= IEC958_AES0_NONAUDIO;
|
|
else
|
|
iec958->status[0] &= ~IEC958_AES0_NONAUDIO;
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_RATE] >= 0) {
|
|
if (iec958->status[0] & IEC958_AES0_PROFESSIONAL) {
|
|
iec958->status[0] &= ~IEC958_AES0_PRO_FS;
|
|
switch (parms[IDX_RATE]) {
|
|
case 44100:
|
|
iec958->status[0] |= IEC958_AES0_PRO_FS_44100;
|
|
break;
|
|
case 48000:
|
|
iec958->status[0] |= IEC958_AES0_PRO_FS_48000;
|
|
break;
|
|
case 32000:
|
|
iec958->status[0] |= IEC958_AES0_PRO_FS_32000;
|
|
break;
|
|
}
|
|
} else {
|
|
iec958->status[3] &= ~IEC958_AES3_CON_FS;
|
|
switch (parms[IDX_RATE]) {
|
|
case 22050:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_22050;
|
|
break;
|
|
case 24000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_24000;
|
|
break;
|
|
case 32000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_32000;
|
|
break;
|
|
case 44100:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_44100;
|
|
break;
|
|
case 48000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_48000;
|
|
break;
|
|
case 88200:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_88200;;
|
|
break;
|
|
case 96000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_96000;
|
|
break;
|
|
case 176400:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_176400;
|
|
break;
|
|
case 192000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_192000;
|
|
break;
|
|
case 768000:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_768000;
|
|
break;
|
|
default:
|
|
iec958->status[3] |= IEC958_AES3_CON_FS_NOTID;
|
|
break;
|
|
}
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_NOCOPY] >= 0) {
|
|
if (! (iec958->status[0] & IEC958_AES0_PROFESSIONAL)) {
|
|
if (parms[IDX_NOCOPY])
|
|
iec958->status[0] |= IEC958_AES0_CON_NOT_COPYRIGHT;
|
|
else
|
|
iec958->status[0] &= ~IEC958_AES0_CON_NOT_COPYRIGHT;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_ORIG] >= 0) {
|
|
if (! (iec958->status[0] & IEC958_AES0_PROFESSIONAL)) {
|
|
if (parms[IDX_ORIG])
|
|
iec958->status[1] |= IEC958_AES1_CON_ORIGINAL;
|
|
else
|
|
iec958->status[1] &= ~IEC958_AES1_CON_ORIGINAL;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_EMP] >= 0) {
|
|
if (iec958->status[0] & IEC958_AES0_PROFESSIONAL) {
|
|
iec958->status[0] &= ~IEC958_AES0_PRO_EMPHASIS;
|
|
switch (parms[IDX_EMP]) {
|
|
case 0:
|
|
iec958->status[0] |= IEC958_AES0_PRO_EMPHASIS_NONE;
|
|
break;
|
|
case 1:
|
|
iec958->status[0] |= IEC958_AES0_PRO_EMPHASIS_5015;
|
|
break;
|
|
case 2:
|
|
iec958->status[0] |= IEC958_AES0_PRO_EMPHASIS_CCITT;
|
|
break;
|
|
}
|
|
} else {
|
|
if (parms[IDX_EMP])
|
|
iec958->status[0] |= IEC958_AES0_CON_EMPHASIS_5015;
|
|
else
|
|
iec958->status[0] &= ~IEC958_AES0_CON_EMPHASIS_5015;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_UNLOCK] >= 0) {
|
|
if (iec958->status[0] & IEC958_AES0_PROFESSIONAL) {
|
|
if (parms[IDX_UNLOCK])
|
|
iec958->status[0] |= IEC958_AES0_PRO_FREQ_UNLOCKED;
|
|
else
|
|
iec958->status[0] &= ~IEC958_AES0_PRO_FREQ_UNLOCKED;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_SBITS] >= 0) {
|
|
if (iec958->status[0] & IEC958_AES0_PROFESSIONAL) {
|
|
iec958->status[2] &= ~IEC958_AES2_PRO_SBITS;
|
|
iec958->status[2] |= parms[IDX_SBITS] & 7;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_WORD] >= 0) {
|
|
if (iec958->status[0] & IEC958_AES0_PROFESSIONAL) {
|
|
iec958->status[2] &= ~IEC958_AES2_PRO_WORDLEN;
|
|
iec958->status[2] |= (parms[IDX_WORD] & 7) << 3;
|
|
}
|
|
changed = 1;
|
|
}
|
|
if (parms[IDX_CAT] >= 0) {
|
|
if (! (iec958->status[0] & IEC958_AES0_PROFESSIONAL)) {
|
|
iec958->status[1] &= ~IEC958_AES1_CON_CATEGORY;
|
|
iec958->status[1] |= parms[IDX_CAT] & 0x7f;
|
|
}
|
|
changed = 1;
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
const char *dev = "default";
|
|
const char *spdif_str = SND_CTL_NAME_IEC958("", PLAYBACK, DEFAULT);
|
|
int spdif_index = -1;
|
|
snd_ctl_t *ctl;
|
|
snd_ctl_elem_list_t *clist;
|
|
snd_ctl_elem_id_t *cid;
|
|
snd_ctl_elem_value_t *cval;
|
|
snd_aes_iec958_t iec958;
|
|
int from_stdin = 0;
|
|
int dumphex = 0;
|
|
int i, c, err;
|
|
unsigned int controls, cidx;
|
|
char tmpname[32];
|
|
int parms[IDX_LAST];
|
|
|
|
for (i = 0; i < IDX_LAST; i++)
|
|
parms[i] = -1; /* not set */
|
|
|
|
while ((c = getopt(argc, argv, "D:c:n:xhi")) != -1) {
|
|
switch (c) {
|
|
case 'D':
|
|
dev = optarg;
|
|
break;
|
|
case 'c':
|
|
i = atoi(optarg);
|
|
if (i < 0 || i >= 32) {
|
|
fprintf(stderr, "invalid card index %d\n", i);
|
|
return 1;
|
|
}
|
|
sprintf(tmpname, "hw:%d", i);
|
|
dev = tmpname;
|
|
break;
|
|
case 'n':
|
|
spdif_index = atoi(optarg);
|
|
break;
|
|
case 'x':
|
|
dumphex = 1;
|
|
break;
|
|
case 'i':
|
|
from_stdin = 1;
|
|
break;
|
|
default:
|
|
usage();
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if ((err = snd_ctl_open(&ctl, dev, 0)) < 0) {
|
|
error("snd_ctl_open", err);
|
|
return 1;
|
|
}
|
|
|
|
snd_ctl_elem_list_alloca(&clist);
|
|
if ((err = snd_ctl_elem_list(ctl, clist)) < 0) {
|
|
error("snd_ctl_elem_list", err);
|
|
return 1;
|
|
}
|
|
if ((err = snd_ctl_elem_list_alloc_space(clist, snd_ctl_elem_list_get_count(clist))) < 0) {
|
|
error("snd_ctl_elem_list_alloc_space", err);
|
|
return 1;
|
|
}
|
|
if ((err = snd_ctl_elem_list(ctl, clist)) < 0) {
|
|
error("snd_ctl_elem_list", err);
|
|
return 1;
|
|
}
|
|
|
|
controls = snd_ctl_elem_list_get_used(clist);
|
|
for (cidx = 0; cidx < controls; cidx++) {
|
|
if (!strcmp(snd_ctl_elem_list_get_name(clist, cidx), spdif_str))
|
|
if (spdif_index < 0 ||
|
|
snd_ctl_elem_list_get_index(clist, cidx) == spdif_index)
|
|
break;
|
|
}
|
|
if (cidx >= controls) {
|
|
fprintf(stderr, "control \"%s\" (index %d) not found\n",
|
|
spdif_str, spdif_index);
|
|
return 1;
|
|
}
|
|
|
|
snd_ctl_elem_id_alloca(&cid);
|
|
snd_ctl_elem_list_get_id(clist, cidx, cid);
|
|
snd_ctl_elem_value_alloca(&cval);
|
|
snd_ctl_elem_value_set_id(cval, cid);
|
|
if ((err = snd_ctl_elem_read(ctl, cval)) < 0) {
|
|
error("snd_ctl_elem_read", err);
|
|
return 1;
|
|
}
|
|
|
|
snd_ctl_elem_value_get_iec958(cval, &iec958);
|
|
|
|
/* parse from stdin */
|
|
if (from_stdin)
|
|
parse_file(parms, stdin);
|
|
|
|
/* parse commands */
|
|
for (c = optind; c < argc - 1; c += 2)
|
|
parse_command(parms, argv[c], argv[c + 1]);
|
|
|
|
if (update_iec958_status(&iec958, parms)) {
|
|
/* store the values */
|
|
snd_ctl_elem_value_set_iec958(cval, &iec958);
|
|
if ((err = snd_ctl_elem_write(ctl, cval)) < 0) {
|
|
error("snd_ctl_elem_write", err);
|
|
return 1;
|
|
}
|
|
if ((err = snd_ctl_elem_read(ctl, cval)) < 0) {
|
|
error("snd_ctl_elem_write", err);
|
|
return 1;
|
|
}
|
|
snd_ctl_elem_value_get_iec958(cval, &iec958);
|
|
}
|
|
|
|
if (dumphex)
|
|
printf("AES0=0x%02x,AES1=0x%02x,AES2=0x%02x,AES3=0x%02x\n",
|
|
iec958.status[0], iec958.status[1], iec958.status[2], iec958.status[3]);
|
|
else
|
|
dump_iec958(&iec958);
|
|
|
|
snd_ctl_close(ctl);
|
|
return 0;
|
|
}
|