1 /* tinypcminfo.c
2 **
3 ** Copyright 2012, The Android Open Source Project
4 **
5 ** Redistribution and use in source and binary forms, with or without
6 ** modification, are permitted provided that the following conditions are met:
7 **     * Redistributions of source code must retain the above copyright
8 **       notice, this list of conditions and the following disclaimer.
9 **     * Redistributions in binary form must reproduce the above copyright
10 **       notice, this list of conditions and the following disclaimer in the
11 **       documentation and/or other materials provided with the distribution.
12 **     * Neither the name of The Android Open Source Project nor the names of
13 **       its contributors may be used to endorse or promote products derived
14 **       from this software without specific prior written permission.
15 **
16 ** THIS SOFTWARE IS PROVIDED BY The Android Open Source Project ``AS IS'' AND
17 ** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 ** ARE DISCLAIMED. IN NO EVENT SHALL The Android Open Source Project BE LIABLE
20 ** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22 ** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
23 ** CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 ** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 ** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
26 ** DAMAGE.
27 */
28 
29 #include <tinyalsa/asoundlib.h>
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <string.h>
33 
34 #ifndef ARRAY_SIZE
35 #define ARRAY_SIZE(x) (sizeof(x)/sizeof((x)[0]))
36 #endif
37 
38 /* The format_lookup is in order of SNDRV_PCM_FORMAT_##index and
39  * matches the grouping in sound/asound.h.  Note this is not
40  * continuous and has an empty gap from (25 - 30).
41  */
42 static const char *format_lookup[] = {
43         /*[0] =*/ "S8",
44         "U8",
45         "S16_LE",
46         "S16_BE",
47         "U16_LE",
48         "U16_BE",
49         "S24_LE",
50         "S24_BE",
51         "U24_LE",
52         "U24_BE",
53         "S32_LE",
54         "S32_BE",
55         "U32_LE",
56         "U32_BE",
57         "FLOAT_LE",
58         "FLOAT_BE",
59         "FLOAT64_LE",
60         "FLOAT64_BE",
61         "IEC958_SUBFRAME_LE",
62         "IEC958_SUBFRAME_BE",
63         "MU_LAW",
64         "A_LAW",
65         "IMA_ADPCM",
66         "MPEG",
67         /*[24] =*/ "GSM",
68         [31] = "SPECIAL",
69         "S24_3LE",
70         "S24_3BE",
71         "U24_3LE",
72         "U24_3BE",
73         "S20_3LE",
74         "S20_3BE",
75         "U20_3LE",
76         "U20_3BE",
77         "S18_3LE",
78         "S18_3BE",
79         "U18_3LE",
80         /*[43] =*/ "U18_3BE",
81 #if 0
82         /* recent additions, may not be present on local asound.h */
83         "G723_24",
84         "G723_24_1B",
85         "G723_40",
86         "G723_40_1B",
87         "DSD_U8",
88         "DSD_U16_LE",
89 #endif
90 };
91 
92 /* Returns a human readable name for the format associated with bit_index,
93  * NULL if bit_index is not known.
94  */
pcm_get_format_name(unsigned bit_index)95 static inline const char *pcm_get_format_name(unsigned bit_index)
96 {
97     return bit_index < ARRAY_SIZE(format_lookup) ? format_lookup[bit_index] : NULL;
98 }
99 
main(int argc,char ** argv)100 int main(int argc, char **argv)
101 {
102     unsigned int device = 0;
103     unsigned int card = 0;
104     int i;
105 
106     if (argc < 3) {
107         fprintf(stderr, "Usage: %s -D card -d device\n", argv[0]);
108         return 1;
109     }
110 
111     /* parse command line arguments */
112     argv += 1;
113     while (*argv) {
114         if (strcmp(*argv, "-D") == 0) {
115             argv++;
116             if (*argv)
117                 card = atoi(*argv);
118         }
119         if (strcmp(*argv, "-d") == 0) {
120             argv++;
121             if (*argv)
122                 device = atoi(*argv);
123         }
124         if (*argv)
125             argv++;
126     }
127 
128     printf("Info for card %u, device %u:\n", card, device);
129 
130     for (i = 0; i < 2; i++) {
131         struct pcm_params *params;
132         struct pcm_mask *m;
133         unsigned int min;
134         unsigned int max;
135 
136         printf("\nPCM %s:\n", i == 0 ? "out" : "in");
137 
138         params = pcm_params_get(card, device, i == 0 ? PCM_OUT : PCM_IN);
139         if (params == NULL) {
140             printf("Device does not exist.\n");
141             continue;
142         }
143 
144         m = pcm_params_get_mask(params, PCM_PARAM_ACCESS);
145         if (m) { /* bitmask, refer to SNDRV_PCM_ACCESS_*, generally interleaved */
146             printf("      Access:\t%#08x\n", m->bits[0]);
147         }
148         m = pcm_params_get_mask(params, PCM_PARAM_FORMAT);
149         if (m) { /* bitmask, refer to: SNDRV_PCM_FORMAT_* */
150             unsigned j, k, count = 0;
151             const unsigned bitcount = sizeof(m->bits[0]) * 8;
152 
153             /* we only check first two format masks (out of 8) - others are zero. */
154             printf("   Format[0]:\t%#08x\n", m->bits[0]);
155             printf("   Format[1]:\t%#08x\n", m->bits[1]);
156 
157             /* print friendly format names, if they exist */
158             for (k = 0; k < 2; ++k) {
159                 for (j = 0; j < bitcount; ++j) {
160                     const char *name;
161 
162                     if (m->bits[k] & (1 << j)) {
163                         name = pcm_get_format_name(j + k*bitcount);
164                         if (name) {
165                             if (count++ == 0) {
166                                 printf(" Format Name:\t");
167                             } else {
168                                 printf (", ");
169                             }
170                             printf("%s", name);
171                         }
172                     }
173                 }
174             }
175             if (count) {
176                 printf("\n");
177             }
178         }
179         m = pcm_params_get_mask(params, PCM_PARAM_SUBFORMAT);
180         if (m) { /* bitmask, should be 1: SNDRV_PCM_SUBFORMAT_STD */
181             printf("   Subformat:\t%#08x\n", m->bits[0]);
182         }
183         min = pcm_params_get_min(params, PCM_PARAM_RATE);
184         max = pcm_params_get_max(params, PCM_PARAM_RATE);
185         printf("        Rate:\tmin=%uHz\tmax=%uHz\n", min, max);
186         min = pcm_params_get_min(params, PCM_PARAM_CHANNELS);
187         max = pcm_params_get_max(params, PCM_PARAM_CHANNELS);
188         printf("    Channels:\tmin=%u\t\tmax=%u\n", min, max);
189         min = pcm_params_get_min(params, PCM_PARAM_SAMPLE_BITS);
190         max = pcm_params_get_max(params, PCM_PARAM_SAMPLE_BITS);
191         printf(" Sample bits:\tmin=%u\t\tmax=%u\n", min, max);
192         min = pcm_params_get_min(params, PCM_PARAM_PERIOD_SIZE);
193         max = pcm_params_get_max(params, PCM_PARAM_PERIOD_SIZE);
194         printf(" Period size:\tmin=%u\t\tmax=%u\n", min, max);
195         min = pcm_params_get_min(params, PCM_PARAM_PERIODS);
196         max = pcm_params_get_max(params, PCM_PARAM_PERIODS);
197         printf("Period count:\tmin=%u\t\tmax=%u\n", min, max);
198 
199         pcm_params_free(params);
200     }
201 
202     return 0;
203 }
204