control设备相关代码解析

tinymix主要有三个功能:

  1. 列出所有控件信息

  2. 列出某一控件的具体信息

  3. 设置某一控件的值

现在根据以上功能来分成三部分介绍

在此之前先介绍三者的共同部分。先看看相关结构体:

struct mixer_ctl {
    struct mixer_ctl_group *grp;
    struct snd_ctl_elem_info *info;
    char **ename;
    bool info_retrieved;
};

struct mixer_ctl_group {
    struct snd_ctl_elem_info *elem_info;
    struct mixer_ctl *ctl;
    unsigned int count;
    int event_cnt;

    struct mixer_ops *ops;
    void *data;
};

struct mixer {
    int fd;
    struct snd_ctl_card_info card_info;

    /* hardware/physical mixer control group */
    struct mixer_ctl_group *hw_grp;

    /*
     * Virutal mixer control group.
     * Currently supports one virtual mixer (.so)
     * per card. Could be extended to multiple
     */
    struct mixer_ctl_group *virt_grp;

    unsigned int total_ctl_count;
};

struct snd_ctl_elem_info {
        struct snd_ctl_elem_id id;      /* W: element ID */
        snd_ctl_elem_type_t type;       /* R: value type - SNDRV_CTL_ELEM_TYPE_* */
        unsigned int access;            /* R: value access (bitmask) - SNDRV_CTL_ELEM_ACCESS_* */
        unsigned int count;             /* count of values */
        __kernel_pid_t owner;           /* owner's PID of this control */
        union {
                struct {
                        long min;               /* R: minimum value */
                        long max;               /* R: maximum value */
                        long step;              /* R: step (0 variable) */
                } integer;
                struct {
                        long long min;          /* R: minimum value */
                        long long max;          /* R: maximum value */
                        long long step;         /* R: step (0 variable) */
                } integer64;
                struct {
                        unsigned int items;     /* R: number of items */
                        unsigned int item;      /* W: item number */
                        char name[64];          /* R: value name */
                        __u64 names_ptr;        /* W: names list (ELEM_ADD only) */
                        unsigned int names_length;
                } enumerated;
                unsigned char reserved[128];
        } value;
        union {
                unsigned short d[4];            /* dimensions */
                unsigned short *d_ptr;          /* indirect - obsoleted */
        } dimen;
        unsigned char reserved[64-4*sizeof(unsigned short)];
};

在main()函数中进行传参判断之前,需要先用mixer_open()打开mixer再进行操作,而在mixer_open()里,主要是调用了mixer_hw_open(),我们直接来看这个函数

int mixer_hw_open(unsigned int card, void **data,
                  struct mixer_ops **ops)
{
    struct mixer_hw_data *hw_data;
    int fd;
    char fn[256];

    snprintf(fn, sizeof(fn), "/dev/snd/controlC%u", card);
    fd = open(fn, O_RDWR);
    if (fd < 0)
        return fd;

    hw_data = calloc(1, sizeof(*hw_data));
    if (!hw_data) {
        close(fd);
        return -1;
    }

    hw_data->card = card;
    hw_data->fd = fd;
    *data = hw_data;
    *ops = &mixer_hw_ops;

    return fd;
}

可以发现其实就是打开control设备节点并返回

三部分在tinymix中对应的代码是:

if (argc == optind) {
    printf("Mixer name: '%s'\n", mixer_get_name(mixer));
    tinymix_list_controls(mixer);
} else if (argc == optind + 1) {
    ret = tinymix_detail_control(mixer, argv[optind], !g_value_only, !g_value_only);
} else if (argc >= optind + 2) {
    ret = tinymix_set_value(mixer, argv[optind], &argv[optind + 1], argc - optind - 1);
}

第一部分-tinymix -D x:查看声卡x的所有control

这部分的工作其实就是打印信息,所以最重要的是如何从control节点拿到这些信息。答案如下:

struct mixer_ctl *ctl;
unsigned int num_ctls;
unsigned int i;

num_ctls = mixer_get_num_ctls(mixer);
...
for (i = 0; i < num_ctls; i++) {
	ctl = mixer_get_ctl(mixer, i);

	mixer_ctl_get_name(ctl)
	mixer_ctl_get_type_string(ctl)
	mixer_ctl_get__num_values(ctl)
	...
	printf(...);
	...
}

后面三个函数都是直接读取ctl->info里面的信息,所以我们重点来看mixer_get_ctl():

struct mixer_ctl *mixer_get_ctl(struct mixer *mixer, unsigned int id)
{
    struct mixer_ctl *ctl;
    unsigned int hw_ctl_count, virt_ctl_count;

    if (!mixer || (id >= mixer->total_ctl_count))
        return NULL;

    hw_ctl_count = mixer_grp_get_count(mixer->hw_grp);
    virt_ctl_count = mixer_grp_get_count(mixer->virt_grp);

    if (id < hw_ctl_count)
        ctl = mixer->hw_grp->ctl + id;
    else if ((id - hw_ctl_count) < virt_ctl_count)
        ctl = mixer->virt_grp->ctl + (id - hw_ctl_count);
    else
        return NULL;

    if (!mixer_ctl_get_elem_info(ctl))
        return NULL;

    return ctl;
}

可以发现前面的部分都是为了定位到ctl,主要是最后的mixer_ctl_get_elem_info(),在该函数里,主要是通过 grp->ops->ioctl(grp->data, SNDRV_CTL_IOCTL_ELEM_INFO, &tmp) 来读取数据

第二部分-tinymix -D x y:查看声卡x的num为y的control信息

直接来看tinymix_detail_control()

/* 通过两种方式拿到ctl */
if (isnumber(control))
    ctl = mixer_get_ctl(mixer, atoi(control));
else
    ctl = mixer_get_ctl_by_name(mixer, control);
...
/* 拿到相关信息 */
type = mixer_ctl_get_type(ctl);
num_values = mixer_ctl_get_num_values(ctl);
...
/* 后面的内容都是打印相关信息了,主要是以下几类 */
/* 1. int类型,打印当前值以及合法范围 */
/* 2. bool类型,只有on和off */
/* 3. enum类型,打印所有选项并标记当前选项 */
/* 4. byte类型, tlv相关*/
...

第三部分-tinymix -D x y z:把声卡x的y控件设置为z

该部分在tinymix_set_value()函数中实现,其工作原理和前者类似:

  1. 先拿到相应的ctl

  2. 再根据它的类型选择对应的函数对其操作,根据类型不同有以下函数:

  • mixer_ctl_set_array()

  • mixer_ctl_set_value()

  • mixer_ctl_set_enum_by_string()

猜你喜欢

转载自blog.csdn.net/hhx123456798/article/details/123605796