xref: /aosp_15_r20/external/tinyalsa_new/examples/sndcardparser/sample_sndcardparser.c (revision 02e95f1a335b55495d41ca67eaf42361f13704fa)
1*02e95f1aSMarcin Radomski /* sample_sndcardparser.c
2*02e95f1aSMarcin Radomski **
3*02e95f1aSMarcin Radomski ** Copyright (c) 2021, The Linux Foundation. All rights reserved.
4*02e95f1aSMarcin Radomski **
5*02e95f1aSMarcin Radomski ** Redistribution and use in source and binary forms, with or without
6*02e95f1aSMarcin Radomski ** modification, are permitted provided that the following conditions are
7*02e95f1aSMarcin Radomski ** met:
8*02e95f1aSMarcin Radomski **   * Redistributions of source code must retain the above copyright
9*02e95f1aSMarcin Radomski **     notice, this list of conditions and the following disclaimer.
10*02e95f1aSMarcin Radomski **   * Redistributions in binary form must reproduce the above
11*02e95f1aSMarcin Radomski **     copyright notice, this list of conditions and the following
12*02e95f1aSMarcin Radomski **     disclaimer in the documentation and/or other materials provided
13*02e95f1aSMarcin Radomski **     with the distribution.
14*02e95f1aSMarcin Radomski **   * Neither the name of The Linux Foundation nor the names of its
15*02e95f1aSMarcin Radomski **     contributors may be used to endorse or promote products derived
16*02e95f1aSMarcin Radomski **     from this software without specific prior written permission.
17*02e95f1aSMarcin Radomski **
18*02e95f1aSMarcin Radomski ** THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
19*02e95f1aSMarcin Radomski ** WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
20*02e95f1aSMarcin Radomski ** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
21*02e95f1aSMarcin Radomski ** ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
22*02e95f1aSMarcin Radomski ** BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23*02e95f1aSMarcin Radomski ** CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24*02e95f1aSMarcin Radomski ** SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
25*02e95f1aSMarcin Radomski ** BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
26*02e95f1aSMarcin Radomski ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
27*02e95f1aSMarcin Radomski ** OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
28*02e95f1aSMarcin Radomski ** IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29*02e95f1aSMarcin Radomski **/
30*02e95f1aSMarcin Radomski 
31*02e95f1aSMarcin Radomski #include <errno.h>
32*02e95f1aSMarcin Radomski #include <stdio.h>
33*02e95f1aSMarcin Radomski #include <stdlib.h>
34*02e95f1aSMarcin Radomski #include <string.h>
35*02e95f1aSMarcin Radomski 
36*02e95f1aSMarcin Radomski #define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
37*02e95f1aSMarcin Radomski 
38*02e95f1aSMarcin Radomski #define VIRTUAL_SND_CARD_ID 100
39*02e95f1aSMarcin Radomski #define MAX_PATH 256
40*02e95f1aSMarcin Radomski #define BUF_SIZE 1024
41*02e95f1aSMarcin Radomski 
42*02e95f1aSMarcin Radomski enum snd_node_type {
43*02e95f1aSMarcin Radomski     NODE_TYPE_HW = 0,
44*02e95f1aSMarcin Radomski     NODE_TYPE_PLUGIN,
45*02e95f1aSMarcin Radomski     NODE_TYPE_INVALID,
46*02e95f1aSMarcin Radomski };
47*02e95f1aSMarcin Radomski 
48*02e95f1aSMarcin Radomski enum {
49*02e95f1aSMarcin Radomski     NODE_PCM,
50*02e95f1aSMarcin Radomski     NODE_COMPR,
51*02e95f1aSMarcin Radomski     NODE_MIXER,
52*02e95f1aSMarcin Radomski     NODE_MAX,
53*02e95f1aSMarcin Radomski };
54*02e95f1aSMarcin Radomski 
55*02e95f1aSMarcin Radomski struct snd_node_ops {
56*02e95f1aSMarcin Radomski     /** Function pointer to get card definition */
57*02e95f1aSMarcin Radomski     void* (*open_card)(unsigned int card);
58*02e95f1aSMarcin Radomski     /** Function pointer to release card definition */
59*02e95f1aSMarcin Radomski     void (*close_card)(void *card);
60*02e95f1aSMarcin Radomski     /** Get interger type properties from device definition */
61*02e95f1aSMarcin Radomski     int (*get_int)(void *node, const char *prop, int *val);
62*02e95f1aSMarcin Radomski     /** Get string type properties from device definition */
63*02e95f1aSMarcin Radomski     int (*get_str)(void *node, const char *prop, char **val);
64*02e95f1aSMarcin Radomski     /** Function pointer to get mixer definition */
65*02e95f1aSMarcin Radomski     void* (*get_mixer)(void *card);
66*02e95f1aSMarcin Radomski     /** Function pointer to get PCM definition */
67*02e95f1aSMarcin Radomski     void* (*get_pcm)(void *card, unsigned int id);
68*02e95f1aSMarcin Radomski     /** Function pointer to get COMPRESS definition */
69*02e95f1aSMarcin Radomski     void* (*get_compress)(void *card, unsigned int id);
70*02e95f1aSMarcin Radomski };
71*02e95f1aSMarcin Radomski 
72*02e95f1aSMarcin Radomski struct snd_dev_def {
73*02e95f1aSMarcin Radomski     unsigned int device;
74*02e95f1aSMarcin Radomski     int type;
75*02e95f1aSMarcin Radomski     const char *name;
76*02e95f1aSMarcin Radomski     const char *so_name;
77*02e95f1aSMarcin Radomski     int playback; //used only for pcm node
78*02e95f1aSMarcin Radomski     int capture;  //used only for pcm node
79*02e95f1aSMarcin Radomski     /* add custom props here */
80*02e95f1aSMarcin Radomski };
81*02e95f1aSMarcin Radomski 
82*02e95f1aSMarcin Radomski struct snd_dev_def_card {
83*02e95f1aSMarcin Radomski     unsigned int card;
84*02e95f1aSMarcin Radomski     char *name;
85*02e95f1aSMarcin Radomski 
86*02e95f1aSMarcin Radomski     /* child device details */
87*02e95f1aSMarcin Radomski     int num_pcm_nodes;
88*02e95f1aSMarcin Radomski     struct snd_dev_def *pcm_dev_def;
89*02e95f1aSMarcin Radomski 
90*02e95f1aSMarcin Radomski     struct snd_dev_def *mixer_dev_def;
91*02e95f1aSMarcin Radomski };
92*02e95f1aSMarcin Radomski 
93*02e95f1aSMarcin Radomski struct snd_dev_def pcm_devs[] = {
94*02e95f1aSMarcin Radomski     {100, NODE_TYPE_PLUGIN, "PCM100", "libtinyalsav2_example_plugin_pcm.so", 1, 0},
95*02e95f1aSMarcin Radomski     /* Add other plugin info here */
96*02e95f1aSMarcin Radomski };
97*02e95f1aSMarcin Radomski 
98*02e95f1aSMarcin Radomski struct snd_dev_def mixer_dev =
99*02e95f1aSMarcin Radomski     {VIRTUAL_SND_CARD_ID, NODE_TYPE_PLUGIN, "virtual-snd-card", "libtinyalsav2_example_plugin_mixer.so", 0, 0};
100*02e95f1aSMarcin Radomski 
snd_card_def_open_card(unsigned int card)101*02e95f1aSMarcin Radomski void *snd_card_def_open_card(unsigned int card)
102*02e95f1aSMarcin Radomski {
103*02e95f1aSMarcin Radomski     struct snd_dev_def_card *card_def = NULL;
104*02e95f1aSMarcin Radomski     struct snd_dev_def *pcm_dev_def = NULL;
105*02e95f1aSMarcin Radomski     struct snd_dev_def *mixer_dev_def = NULL;
106*02e95f1aSMarcin Radomski     int num_pcm = ARRAY_SIZE(pcm_devs);
107*02e95f1aSMarcin Radomski     int i;
108*02e95f1aSMarcin Radomski 
109*02e95f1aSMarcin Radomski     if (card != VIRTUAL_SND_CARD_ID)
110*02e95f1aSMarcin Radomski         return NULL;
111*02e95f1aSMarcin Radomski 
112*02e95f1aSMarcin Radomski     card_def = calloc(1, sizeof(struct snd_dev_def_card));
113*02e95f1aSMarcin Radomski     if (!card_def)
114*02e95f1aSMarcin Radomski         return card_def;
115*02e95f1aSMarcin Radomski 
116*02e95f1aSMarcin Radomski     card_def->card = card;
117*02e95f1aSMarcin Radomski     card_def->name = strdup("virtual-snd-card");
118*02e95f1aSMarcin Radomski 
119*02e95f1aSMarcin Radomski     /* fill pcm device node info */
120*02e95f1aSMarcin Radomski     card_def->num_pcm_nodes = num_pcm;
121*02e95f1aSMarcin Radomski     pcm_dev_def = calloc(num_pcm, sizeof(struct snd_dev_def));
122*02e95f1aSMarcin Radomski     if (!pcm_dev_def)
123*02e95f1aSMarcin Radomski         goto free_card_def;
124*02e95f1aSMarcin Radomski 
125*02e95f1aSMarcin Radomski     for (i = 0; i < num_pcm; i++)
126*02e95f1aSMarcin Radomski         memcpy(&pcm_dev_def[i], &pcm_devs[i], sizeof(struct snd_dev_def));
127*02e95f1aSMarcin Radomski 
128*02e95f1aSMarcin Radomski     card_def->pcm_dev_def = pcm_dev_def;
129*02e95f1aSMarcin Radomski 
130*02e95f1aSMarcin Radomski     /* fill mixer device node info */
131*02e95f1aSMarcin Radomski     mixer_dev_def = calloc(1, sizeof(struct snd_dev_def));
132*02e95f1aSMarcin Radomski     if (!mixer_dev_def)
133*02e95f1aSMarcin Radomski         goto free_pcm_dev;
134*02e95f1aSMarcin Radomski 
135*02e95f1aSMarcin Radomski     memcpy(mixer_dev_def, &mixer_dev, sizeof(struct snd_dev_def));
136*02e95f1aSMarcin Radomski 
137*02e95f1aSMarcin Radomski     card_def->mixer_dev_def = mixer_dev_def;
138*02e95f1aSMarcin Radomski     return card_def;
139*02e95f1aSMarcin Radomski free_pcm_dev:
140*02e95f1aSMarcin Radomski     free(pcm_dev_def);
141*02e95f1aSMarcin Radomski free_card_def:
142*02e95f1aSMarcin Radomski     free(card_def->name);
143*02e95f1aSMarcin Radomski     free(card_def);
144*02e95f1aSMarcin Radomski     return NULL;
145*02e95f1aSMarcin Radomski }
146*02e95f1aSMarcin Radomski 
snd_card_def_close_card(void * card_node)147*02e95f1aSMarcin Radomski void snd_card_def_close_card(void *card_node)
148*02e95f1aSMarcin Radomski {
149*02e95f1aSMarcin Radomski     struct snd_dev_def_card *defs = (struct snd_dev_def_card *)card_node;
150*02e95f1aSMarcin Radomski     struct snd_dev_def *pcm_dev_def = NULL;
151*02e95f1aSMarcin Radomski     struct snd_dev_def *mixer_dev_def = NULL;
152*02e95f1aSMarcin Radomski 
153*02e95f1aSMarcin Radomski     if (!defs)
154*02e95f1aSMarcin Radomski         return;
155*02e95f1aSMarcin Radomski 
156*02e95f1aSMarcin Radomski     pcm_dev_def = defs->pcm_dev_def;
157*02e95f1aSMarcin Radomski     if (pcm_dev_def)
158*02e95f1aSMarcin Radomski         free(pcm_dev_def);
159*02e95f1aSMarcin Radomski 
160*02e95f1aSMarcin Radomski     mixer_dev_def = defs->mixer_dev_def;
161*02e95f1aSMarcin Radomski     if (!mixer_dev_def)
162*02e95f1aSMarcin Radomski          goto free_defs;
163*02e95f1aSMarcin Radomski 
164*02e95f1aSMarcin Radomski     free(mixer_dev_def);
165*02e95f1aSMarcin Radomski free_defs:
166*02e95f1aSMarcin Radomski     free(defs->name);
167*02e95f1aSMarcin Radomski     free(defs);
168*02e95f1aSMarcin Radomski }
169*02e95f1aSMarcin Radomski 
snd_card_def_get_node(void * card_node,unsigned int id,int type)170*02e95f1aSMarcin Radomski void *snd_card_def_get_node(void *card_node, unsigned int id, int type)
171*02e95f1aSMarcin Radomski {
172*02e95f1aSMarcin Radomski     struct snd_dev_def_card *card_def = (struct snd_dev_def_card *)card_node;
173*02e95f1aSMarcin Radomski     struct snd_dev_def *dev_def = NULL;
174*02e95f1aSMarcin Radomski     int i;
175*02e95f1aSMarcin Radomski 
176*02e95f1aSMarcin Radomski     if (!card_def)
177*02e95f1aSMarcin Radomski         return NULL;
178*02e95f1aSMarcin Radomski 
179*02e95f1aSMarcin Radomski     if (type >= NODE_MAX)
180*02e95f1aSMarcin Radomski         return NULL;
181*02e95f1aSMarcin Radomski 
182*02e95f1aSMarcin Radomski     if (type == NODE_MIXER)
183*02e95f1aSMarcin Radomski         return card_def->mixer_dev_def;
184*02e95f1aSMarcin Radomski 
185*02e95f1aSMarcin Radomski     if (type == NODE_PCM)
186*02e95f1aSMarcin Radomski         dev_def = card_def->pcm_dev_def;
187*02e95f1aSMarcin Radomski 
188*02e95f1aSMarcin Radomski     for (i = 0; i < card_def->num_pcm_nodes; i++) {
189*02e95f1aSMarcin Radomski         if (dev_def[i].device == id) {
190*02e95f1aSMarcin Radomski             return &dev_def[i];
191*02e95f1aSMarcin Radomski         }
192*02e95f1aSMarcin Radomski     }
193*02e95f1aSMarcin Radomski 
194*02e95f1aSMarcin Radomski     return NULL;
195*02e95f1aSMarcin Radomski }
196*02e95f1aSMarcin Radomski 
snd_card_def_get_int(void * node,const char * prop,int * val)197*02e95f1aSMarcin Radomski int snd_card_def_get_int(void *node, const char *prop, int *val)
198*02e95f1aSMarcin Radomski {
199*02e95f1aSMarcin Radomski     struct snd_dev_def *dev_def = (struct snd_dev_def *)node;
200*02e95f1aSMarcin Radomski     int ret = -EINVAL;
201*02e95f1aSMarcin Radomski 
202*02e95f1aSMarcin Radomski     if (!dev_def || !prop || !val)
203*02e95f1aSMarcin Radomski         return ret;
204*02e95f1aSMarcin Radomski 
205*02e95f1aSMarcin Radomski     if (!strcmp(prop, "type")) {
206*02e95f1aSMarcin Radomski         *val = dev_def->type;
207*02e95f1aSMarcin Radomski         return 0;
208*02e95f1aSMarcin Radomski     } else if (!strcmp(prop, "id")) {
209*02e95f1aSMarcin Radomski         *val = dev_def->device;
210*02e95f1aSMarcin Radomski         return 0;
211*02e95f1aSMarcin Radomski     } else if (!strcmp(prop, "playback")) {
212*02e95f1aSMarcin Radomski         *val = dev_def->playback;
213*02e95f1aSMarcin Radomski         return 0;
214*02e95f1aSMarcin Radomski     } else if (!strcmp(prop, "capture")) {
215*02e95f1aSMarcin Radomski         *val = dev_def->capture;
216*02e95f1aSMarcin Radomski         return 0;
217*02e95f1aSMarcin Radomski     }
218*02e95f1aSMarcin Radomski 
219*02e95f1aSMarcin Radomski     return ret;
220*02e95f1aSMarcin Radomski }
221*02e95f1aSMarcin Radomski 
snd_card_def_get_str(void * node,const char * prop,char ** val)222*02e95f1aSMarcin Radomski int snd_card_def_get_str(void *node, const char *prop, char **val)
223*02e95f1aSMarcin Radomski {
224*02e95f1aSMarcin Radomski     struct snd_dev_def *dev_def = (struct snd_dev_def *)node;
225*02e95f1aSMarcin Radomski     int ret = -EINVAL;
226*02e95f1aSMarcin Radomski 
227*02e95f1aSMarcin Radomski     if (!dev_def || !prop)
228*02e95f1aSMarcin Radomski         return ret;
229*02e95f1aSMarcin Radomski 
230*02e95f1aSMarcin Radomski     if (!strcmp(prop, "so-name")) {
231*02e95f1aSMarcin Radomski         if (dev_def->so_name) {
232*02e95f1aSMarcin Radomski             *val = (char *)dev_def->so_name;
233*02e95f1aSMarcin Radomski             return 0;
234*02e95f1aSMarcin Radomski         }
235*02e95f1aSMarcin Radomski     }
236*02e95f1aSMarcin Radomski 
237*02e95f1aSMarcin Radomski     if (!strcmp(prop, "name")) {
238*02e95f1aSMarcin Radomski         if (dev_def->name) {
239*02e95f1aSMarcin Radomski             *val = (char *)dev_def->name;
240*02e95f1aSMarcin Radomski             return 0;
241*02e95f1aSMarcin Radomski         }
242*02e95f1aSMarcin Radomski     }
243*02e95f1aSMarcin Radomski 
244*02e95f1aSMarcin Radomski     return ret;
245*02e95f1aSMarcin Radomski }
246*02e95f1aSMarcin Radomski 
snd_card_def_get_pcm(void * card_node,unsigned int id)247*02e95f1aSMarcin Radomski void *snd_card_def_get_pcm(void *card_node, unsigned int id)
248*02e95f1aSMarcin Radomski {
249*02e95f1aSMarcin Radomski     return snd_card_def_get_node(card_node, id, NODE_PCM);
250*02e95f1aSMarcin Radomski }
251*02e95f1aSMarcin Radomski 
snd_card_def_get_compress(void * card_node,unsigned int id)252*02e95f1aSMarcin Radomski void *snd_card_def_get_compress(void *card_node, unsigned int id)
253*02e95f1aSMarcin Radomski {
254*02e95f1aSMarcin Radomski     return snd_card_def_get_node(card_node, id, NODE_COMPR);
255*02e95f1aSMarcin Radomski }
256*02e95f1aSMarcin Radomski 
snd_card_def_get_mixer(void * card_node)257*02e95f1aSMarcin Radomski void *snd_card_def_get_mixer(void *card_node)
258*02e95f1aSMarcin Radomski {
259*02e95f1aSMarcin Radomski     return snd_card_def_get_node(card_node, 1, NODE_MIXER);
260*02e95f1aSMarcin Radomski }
261*02e95f1aSMarcin Radomski 
262*02e95f1aSMarcin Radomski struct snd_node_ops snd_card_ops = {
263*02e95f1aSMarcin Radomski     .open_card = snd_card_def_open_card,
264*02e95f1aSMarcin Radomski     .close_card = snd_card_def_close_card,
265*02e95f1aSMarcin Radomski     .get_int = snd_card_def_get_int,
266*02e95f1aSMarcin Radomski     .get_str = snd_card_def_get_str,
267*02e95f1aSMarcin Radomski     .get_pcm = snd_card_def_get_pcm,
268*02e95f1aSMarcin Radomski     .get_compress = snd_card_def_get_compress,
269*02e95f1aSMarcin Radomski     .get_mixer = snd_card_def_get_mixer,
270*02e95f1aSMarcin Radomski };
271