xref: /aosp_15_r20/external/tinyalsa_new/src/mixer_hw.c (revision 02e95f1a335b55495d41ca67eaf42361f13704fa)
1*02e95f1aSMarcin Radomski /* mixer_hw.c
2*02e95f1aSMarcin Radomski ** Copyright (c) 2019, The Linux Foundation.
3*02e95f1aSMarcin Radomski **
4*02e95f1aSMarcin Radomski ** Redistribution and use in source and binary forms, with or without
5*02e95f1aSMarcin Radomski ** modification, are permitted provided that the following conditions are
6*02e95f1aSMarcin Radomski ** met:
7*02e95f1aSMarcin Radomski **   * Redistributions of source code must retain the above copyright
8*02e95f1aSMarcin Radomski **     notice, this list of conditions and the following disclaimer.
9*02e95f1aSMarcin Radomski **   * Redistributions in binary form must reproduce the above
10*02e95f1aSMarcin Radomski **     copyright notice, this list of conditions and the following
11*02e95f1aSMarcin Radomski **     disclaimer in the documentation and/or other materials provided
12*02e95f1aSMarcin Radomski **     with the distribution.
13*02e95f1aSMarcin Radomski **   * Neither the name of The Linux Foundation nor the names of its
14*02e95f1aSMarcin Radomski **     contributors may be used to endorse or promote products derived
15*02e95f1aSMarcin Radomski **     from this software without specific prior written permission.
16*02e95f1aSMarcin Radomski **
17*02e95f1aSMarcin Radomski ** THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18*02e95f1aSMarcin Radomski ** WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*02e95f1aSMarcin Radomski ** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20*02e95f1aSMarcin Radomski ** ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21*02e95f1aSMarcin Radomski ** BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22*02e95f1aSMarcin Radomski ** CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23*02e95f1aSMarcin Radomski ** SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24*02e95f1aSMarcin Radomski ** BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25*02e95f1aSMarcin Radomski ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26*02e95f1aSMarcin Radomski ** OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*02e95f1aSMarcin Radomski ** IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*02e95f1aSMarcin Radomski **/
29*02e95f1aSMarcin Radomski 
30*02e95f1aSMarcin Radomski #include <stdio.h>
31*02e95f1aSMarcin Radomski #include <stdlib.h>
32*02e95f1aSMarcin Radomski #include <stdarg.h>
33*02e95f1aSMarcin Radomski #include <stdint.h>
34*02e95f1aSMarcin Radomski #include <stdbool.h>
35*02e95f1aSMarcin Radomski #include <string.h>
36*02e95f1aSMarcin Radomski #include <unistd.h>
37*02e95f1aSMarcin Radomski #include <fcntl.h>
38*02e95f1aSMarcin Radomski #include <errno.h>
39*02e95f1aSMarcin Radomski #include <ctype.h>
40*02e95f1aSMarcin Radomski #include <poll.h>
41*02e95f1aSMarcin Radomski 
42*02e95f1aSMarcin Radomski #include <sys/ioctl.h>
43*02e95f1aSMarcin Radomski 
44*02e95f1aSMarcin Radomski #include <linux/ioctl.h>
45*02e95f1aSMarcin Radomski #include <time.h>
46*02e95f1aSMarcin Radomski #include <sound/asound.h>
47*02e95f1aSMarcin Radomski 
48*02e95f1aSMarcin Radomski #include "mixer_io.h"
49*02e95f1aSMarcin Radomski 
50*02e95f1aSMarcin Radomski /** Store the hardware (kernel interface) mixer data */
51*02e95f1aSMarcin Radomski struct mixer_hw_data {
52*02e95f1aSMarcin Radomski     /* Card number for the mixer */
53*02e95f1aSMarcin Radomski     unsigned int card;
54*02e95f1aSMarcin Radomski     /* File descriptor of the mixer device node */
55*02e95f1aSMarcin Radomski     int fd;
56*02e95f1aSMarcin Radomski };
57*02e95f1aSMarcin Radomski 
mixer_hw_close(void * data)58*02e95f1aSMarcin Radomski static void mixer_hw_close(void *data)
59*02e95f1aSMarcin Radomski {
60*02e95f1aSMarcin Radomski     struct mixer_hw_data *hw_data = data;
61*02e95f1aSMarcin Radomski 
62*02e95f1aSMarcin Radomski     if (!hw_data)
63*02e95f1aSMarcin Radomski         return;
64*02e95f1aSMarcin Radomski 
65*02e95f1aSMarcin Radomski     if (hw_data->fd >= 0)
66*02e95f1aSMarcin Radomski         close(hw_data->fd);
67*02e95f1aSMarcin Radomski 
68*02e95f1aSMarcin Radomski     hw_data->fd = -1;
69*02e95f1aSMarcin Radomski     free(hw_data);
70*02e95f1aSMarcin Radomski     hw_data = NULL;
71*02e95f1aSMarcin Radomski }
72*02e95f1aSMarcin Radomski 
mixer_hw_ioctl(void * data,unsigned int cmd,...)73*02e95f1aSMarcin Radomski static int mixer_hw_ioctl(void *data, unsigned int cmd, ...)
74*02e95f1aSMarcin Radomski {
75*02e95f1aSMarcin Radomski     struct mixer_hw_data *hw_data = data;
76*02e95f1aSMarcin Radomski     va_list ap;
77*02e95f1aSMarcin Radomski     void *arg;
78*02e95f1aSMarcin Radomski 
79*02e95f1aSMarcin Radomski     va_start(ap, cmd);
80*02e95f1aSMarcin Radomski     arg = va_arg(ap, void *);
81*02e95f1aSMarcin Radomski     va_end(ap);
82*02e95f1aSMarcin Radomski 
83*02e95f1aSMarcin Radomski     return ioctl(hw_data->fd, cmd, arg);
84*02e95f1aSMarcin Radomski }
85*02e95f1aSMarcin Radomski 
mixer_hw_read_event(void * data,struct snd_ctl_event * ev,size_t size)86*02e95f1aSMarcin Radomski static ssize_t mixer_hw_read_event(void *data, struct snd_ctl_event *ev,
87*02e95f1aSMarcin Radomski                                    size_t size)
88*02e95f1aSMarcin Radomski {
89*02e95f1aSMarcin Radomski     struct mixer_hw_data *hw_data = data;
90*02e95f1aSMarcin Radomski 
91*02e95f1aSMarcin Radomski     return read(hw_data->fd, ev, size);
92*02e95f1aSMarcin Radomski }
93*02e95f1aSMarcin Radomski 
94*02e95f1aSMarcin Radomski static const struct mixer_ops mixer_hw_ops = {
95*02e95f1aSMarcin Radomski     .close = mixer_hw_close,
96*02e95f1aSMarcin Radomski     .ioctl = mixer_hw_ioctl,
97*02e95f1aSMarcin Radomski     .read_event = mixer_hw_read_event,
98*02e95f1aSMarcin Radomski };
99*02e95f1aSMarcin Radomski 
mixer_hw_open(unsigned int card,void ** data,const struct mixer_ops ** ops)100*02e95f1aSMarcin Radomski int mixer_hw_open(unsigned int card, void **data,
101*02e95f1aSMarcin Radomski                   const struct mixer_ops **ops)
102*02e95f1aSMarcin Radomski {
103*02e95f1aSMarcin Radomski     struct mixer_hw_data *hw_data;
104*02e95f1aSMarcin Radomski     int fd;
105*02e95f1aSMarcin Radomski     char fn[256];
106*02e95f1aSMarcin Radomski 
107*02e95f1aSMarcin Radomski     snprintf(fn, sizeof(fn), "/dev/snd/controlC%u", card);
108*02e95f1aSMarcin Radomski     fd = open(fn, O_RDWR);
109*02e95f1aSMarcin Radomski     if (fd < 0)
110*02e95f1aSMarcin Radomski         return fd;
111*02e95f1aSMarcin Radomski 
112*02e95f1aSMarcin Radomski     hw_data = calloc(1, sizeof(*hw_data));
113*02e95f1aSMarcin Radomski     if (!hw_data)
114*02e95f1aSMarcin Radomski         return -1;
115*02e95f1aSMarcin Radomski 
116*02e95f1aSMarcin Radomski     hw_data->card = card;
117*02e95f1aSMarcin Radomski     hw_data->fd = fd;
118*02e95f1aSMarcin Radomski     *data = hw_data;
119*02e95f1aSMarcin Radomski     *ops = &mixer_hw_ops;
120*02e95f1aSMarcin Radomski 
121*02e95f1aSMarcin Radomski     return fd;
122*02e95f1aSMarcin Radomski }
123