add a lot of validity checking
[platform/upstream/pulseaudio.git] / src / pulse / channelmap.c
1 /***
2   This file is part of PulseAudio.
3
4   Copyright 2005-2006 Lennart Poettering
5   Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB
6
7   PulseAudio is free software; you can redistribute it and/or modify
8   it under the terms of the GNU Lesser General Public License as published
9   by the Free Software Foundation; either version 2 of the License,
10   or (at your option) any later version.
11
12   PulseAudio is distributed in the hope that it will be useful, but
13   WITHOUT ANY WARRANTY; without even the implied warranty of
14   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15   General Public License for more details.
16
17   You should have received a copy of the GNU Lesser General Public License
18   along with PulseAudio; if not, write to the Free Software
19   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
20   USA.
21 ***/
22
23 #ifdef HAVE_CONFIG_H
24 #include <config.h>
25 #endif
26
27 #include <stdlib.h>
28 #include <stdio.h>
29 #include <string.h>
30
31 #include <pulse/xmalloc.h>
32 #include <pulse/i18n.h>
33 #include <pulsecore/core-util.h>
34 #include <pulsecore/macro.h>
35 #include <pulsecore/bitset.h>
36
37 #include "channelmap.h"
38
39 const char *const table[PA_CHANNEL_POSITION_MAX] = {
40     [PA_CHANNEL_POSITION_MONO] = "mono",
41
42     [PA_CHANNEL_POSITION_FRONT_CENTER] = "front-center",
43     [PA_CHANNEL_POSITION_FRONT_LEFT] = "front-left",
44     [PA_CHANNEL_POSITION_FRONT_RIGHT] = "front-right",
45
46     [PA_CHANNEL_POSITION_REAR_CENTER] = "rear-center",
47     [PA_CHANNEL_POSITION_REAR_LEFT] = "rear-left",
48     [PA_CHANNEL_POSITION_REAR_RIGHT] = "rear-right",
49
50     [PA_CHANNEL_POSITION_LFE] = "lfe",
51
52     [PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER] = "front-left-of-center",
53     [PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER] = "front-right-of-center",
54
55     [PA_CHANNEL_POSITION_SIDE_LEFT] = "side-left",
56     [PA_CHANNEL_POSITION_SIDE_RIGHT] = "side-right",
57
58     [PA_CHANNEL_POSITION_AUX0] = "aux0",
59     [PA_CHANNEL_POSITION_AUX1] = "aux1",
60     [PA_CHANNEL_POSITION_AUX2] = "aux2",
61     [PA_CHANNEL_POSITION_AUX3] = "aux3",
62     [PA_CHANNEL_POSITION_AUX4] = "aux4",
63     [PA_CHANNEL_POSITION_AUX5] = "aux5",
64     [PA_CHANNEL_POSITION_AUX6] = "aux6",
65     [PA_CHANNEL_POSITION_AUX7] = "aux7",
66     [PA_CHANNEL_POSITION_AUX8] = "aux8",
67     [PA_CHANNEL_POSITION_AUX9] = "aux9",
68     [PA_CHANNEL_POSITION_AUX10] = "aux10",
69     [PA_CHANNEL_POSITION_AUX11] = "aux11",
70     [PA_CHANNEL_POSITION_AUX12] = "aux12",
71     [PA_CHANNEL_POSITION_AUX13] = "aux13",
72     [PA_CHANNEL_POSITION_AUX14] = "aux14",
73     [PA_CHANNEL_POSITION_AUX15] = "aux15",
74     [PA_CHANNEL_POSITION_AUX16] = "aux16",
75     [PA_CHANNEL_POSITION_AUX17] = "aux17",
76     [PA_CHANNEL_POSITION_AUX18] = "aux18",
77     [PA_CHANNEL_POSITION_AUX19] = "aux19",
78     [PA_CHANNEL_POSITION_AUX20] = "aux20",
79     [PA_CHANNEL_POSITION_AUX21] = "aux21",
80     [PA_CHANNEL_POSITION_AUX22] = "aux22",
81     [PA_CHANNEL_POSITION_AUX23] = "aux23",
82     [PA_CHANNEL_POSITION_AUX24] = "aux24",
83     [PA_CHANNEL_POSITION_AUX25] = "aux25",
84     [PA_CHANNEL_POSITION_AUX26] = "aux26",
85     [PA_CHANNEL_POSITION_AUX27] = "aux27",
86     [PA_CHANNEL_POSITION_AUX28] = "aux28",
87     [PA_CHANNEL_POSITION_AUX29] = "aux29",
88     [PA_CHANNEL_POSITION_AUX30] = "aux30",
89     [PA_CHANNEL_POSITION_AUX31] = "aux31",
90
91     [PA_CHANNEL_POSITION_TOP_CENTER] = "top-center",
92
93     [PA_CHANNEL_POSITION_TOP_FRONT_CENTER] = "top-front-center",
94     [PA_CHANNEL_POSITION_TOP_FRONT_LEFT] = "top-front-left",
95     [PA_CHANNEL_POSITION_TOP_FRONT_RIGHT] = "top-front-right",
96
97     [PA_CHANNEL_POSITION_TOP_REAR_CENTER] = "top-rear-center",
98     [PA_CHANNEL_POSITION_TOP_REAR_LEFT] = "top-rear-left",
99     [PA_CHANNEL_POSITION_TOP_REAR_RIGHT] = "top-rear-right"
100 };
101
102 const char *const pretty_table[PA_CHANNEL_POSITION_MAX] = {
103     [PA_CHANNEL_POSITION_MONO] = N_("Mono"),
104
105     [PA_CHANNEL_POSITION_FRONT_CENTER] = N_("Front Center"),
106     [PA_CHANNEL_POSITION_FRONT_LEFT] = N_("Front Left"),
107     [PA_CHANNEL_POSITION_FRONT_RIGHT] = N_("Front Right"),
108
109     [PA_CHANNEL_POSITION_REAR_CENTER] = N_("Rear Center"),
110     [PA_CHANNEL_POSITION_REAR_LEFT] = N_("Rear Left"),
111     [PA_CHANNEL_POSITION_REAR_RIGHT] = N_("Rear Right"),
112
113     [PA_CHANNEL_POSITION_LFE] = N_("Low Frequency Emmiter"),
114
115     [PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER] = N_("Front Left-of-center"),
116     [PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER] = N_("Front Right-of-center"),
117
118     [PA_CHANNEL_POSITION_SIDE_LEFT] = N_("Side Left"),
119     [PA_CHANNEL_POSITION_SIDE_RIGHT] = N_("Side Right"),
120
121     [PA_CHANNEL_POSITION_AUX0] = N_("Auxiliary 0"),
122     [PA_CHANNEL_POSITION_AUX1] = N_("Auxiliary 1"),
123     [PA_CHANNEL_POSITION_AUX2] = N_("Auxiliary 2"),
124     [PA_CHANNEL_POSITION_AUX3] = N_("Auxiliary 3"),
125     [PA_CHANNEL_POSITION_AUX4] = N_("Auxiliary 4"),
126     [PA_CHANNEL_POSITION_AUX5] = N_("Auxiliary 5"),
127     [PA_CHANNEL_POSITION_AUX6] = N_("Auxiliary 6"),
128     [PA_CHANNEL_POSITION_AUX7] = N_("Auxiliary 7"),
129     [PA_CHANNEL_POSITION_AUX8] = N_("Auxiliary 8"),
130     [PA_CHANNEL_POSITION_AUX9] = N_("Auxiliary 9"),
131     [PA_CHANNEL_POSITION_AUX10] = N_("Auxiliary 10"),
132     [PA_CHANNEL_POSITION_AUX11] = N_("Auxiliary 11"),
133     [PA_CHANNEL_POSITION_AUX12] = N_("Auxiliary 12"),
134     [PA_CHANNEL_POSITION_AUX13] = N_("Auxiliary 13"),
135     [PA_CHANNEL_POSITION_AUX14] = N_("Auxiliary 14"),
136     [PA_CHANNEL_POSITION_AUX15] = N_("Auxiliary 15"),
137     [PA_CHANNEL_POSITION_AUX16] = N_("Auxiliary 16"),
138     [PA_CHANNEL_POSITION_AUX17] = N_("Auxiliary 17"),
139     [PA_CHANNEL_POSITION_AUX18] = N_("Auxiliary 18"),
140     [PA_CHANNEL_POSITION_AUX19] = N_("Auxiliary 19"),
141     [PA_CHANNEL_POSITION_AUX20] = N_("Auxiliary 20"),
142     [PA_CHANNEL_POSITION_AUX21] = N_("Auxiliary 21"),
143     [PA_CHANNEL_POSITION_AUX22] = N_("Auxiliary 22"),
144     [PA_CHANNEL_POSITION_AUX23] = N_("Auxiliary 23"),
145     [PA_CHANNEL_POSITION_AUX24] = N_("Auxiliary 24"),
146     [PA_CHANNEL_POSITION_AUX25] = N_("Auxiliary 25"),
147     [PA_CHANNEL_POSITION_AUX26] = N_("Auxiliary 26"),
148     [PA_CHANNEL_POSITION_AUX27] = N_("Auxiliary 27"),
149     [PA_CHANNEL_POSITION_AUX28] = N_("Auxiliary 28"),
150     [PA_CHANNEL_POSITION_AUX29] = N_("Auxiliary 29"),
151     [PA_CHANNEL_POSITION_AUX30] = N_("Auxiliary 30"),
152     [PA_CHANNEL_POSITION_AUX31] = N_("Auxiliary 31"),
153
154     [PA_CHANNEL_POSITION_TOP_CENTER] = N_("Top Center"),
155
156     [PA_CHANNEL_POSITION_TOP_FRONT_CENTER] = N_("Top Front Center"),
157     [PA_CHANNEL_POSITION_TOP_FRONT_LEFT] = N_("Top Front Left"),
158     [PA_CHANNEL_POSITION_TOP_FRONT_RIGHT] = N_("Top Front Right"),
159
160     [PA_CHANNEL_POSITION_TOP_REAR_CENTER] = N_("Top Rear Center"),
161     [PA_CHANNEL_POSITION_TOP_REAR_LEFT] = N_("Top Rear Left"),
162     [PA_CHANNEL_POSITION_TOP_REAR_RIGHT] = N_("Top Rear Right")
163 };
164
165 pa_channel_map* pa_channel_map_init(pa_channel_map *m) {
166     unsigned c;
167     pa_assert(m);
168
169     m->channels = 0;
170
171     for (c = 0; c < PA_CHANNELS_MAX; c++)
172         m->map[c] = PA_CHANNEL_POSITION_INVALID;
173
174     return m;
175 }
176
177 pa_channel_map* pa_channel_map_init_mono(pa_channel_map *m) {
178     pa_assert(m);
179
180     pa_channel_map_init(m);
181
182     m->channels = 1;
183     m->map[0] = PA_CHANNEL_POSITION_MONO;
184     return m;
185 }
186
187 pa_channel_map* pa_channel_map_init_stereo(pa_channel_map *m) {
188     pa_assert(m);
189
190     pa_channel_map_init(m);
191
192     m->channels = 2;
193     m->map[0] = PA_CHANNEL_POSITION_LEFT;
194     m->map[1] = PA_CHANNEL_POSITION_RIGHT;
195     return m;
196 }
197
198 pa_channel_map* pa_channel_map_init_auto(pa_channel_map *m, unsigned channels, pa_channel_map_def_t def) {
199     pa_assert(m);
200     pa_assert(channels > 0);
201     pa_assert(channels <= PA_CHANNELS_MAX);
202     pa_assert(def < PA_CHANNEL_MAP_DEF_MAX);
203
204     pa_channel_map_init(m);
205
206     m->channels = (uint8_t) channels;
207
208     switch (def) {
209         case PA_CHANNEL_MAP_AIFF:
210
211             /* This is somewhat compatible with RFC3551 */
212
213             switch (channels) {
214                 case 1:
215                     m->map[0] = PA_CHANNEL_POSITION_MONO;
216                     return m;
217
218                 case 6:
219                     m->map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
220                     m->map[1] = PA_CHANNEL_POSITION_SIDE_LEFT;
221                     m->map[2] = PA_CHANNEL_POSITION_FRONT_CENTER;
222                     m->map[3] = PA_CHANNEL_POSITION_FRONT_RIGHT;
223                     m->map[4] = PA_CHANNEL_POSITION_SIDE_RIGHT;
224                     m->map[5] = PA_CHANNEL_POSITION_LFE;
225                     return m;
226
227                 case 5:
228                     m->map[2] = PA_CHANNEL_POSITION_FRONT_CENTER;
229                     m->map[3] = PA_CHANNEL_POSITION_REAR_LEFT;
230                     m->map[4] = PA_CHANNEL_POSITION_REAR_RIGHT;
231                     /* Fall through */
232
233                 case 2:
234                     m->map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
235                     m->map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
236                     return m;
237
238                 case 3:
239                     m->map[0] = PA_CHANNEL_POSITION_LEFT;
240                     m->map[1] = PA_CHANNEL_POSITION_RIGHT;
241                     m->map[2] = PA_CHANNEL_POSITION_CENTER;
242                     return m;
243
244                 case 4:
245                     m->map[0] = PA_CHANNEL_POSITION_LEFT;
246                     m->map[1] = PA_CHANNEL_POSITION_CENTER;
247                     m->map[2] = PA_CHANNEL_POSITION_RIGHT;
248                     m->map[3] = PA_CHANNEL_POSITION_LFE;
249                     return m;
250
251                 default:
252                     return NULL;
253             }
254
255         case PA_CHANNEL_MAP_ALSA:
256
257             switch (channels) {
258                 case 1:
259                     m->map[0] = PA_CHANNEL_POSITION_MONO;
260                     return m;
261
262                 case 8:
263                     m->map[6] = PA_CHANNEL_POSITION_SIDE_LEFT;
264                     m->map[7] = PA_CHANNEL_POSITION_SIDE_RIGHT;
265                     /* Fall through */
266
267                 case 6:
268                     m->map[5] = PA_CHANNEL_POSITION_LFE;
269                     /* Fall through */
270
271                 case 5:
272                     m->map[4] = PA_CHANNEL_POSITION_FRONT_CENTER;
273                     /* Fall through */
274
275                 case 4:
276                     m->map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
277                     m->map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
278                     /* Fall through */
279
280                 case 2:
281                     m->map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
282                     m->map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
283                     return m;
284
285                 default:
286                     return NULL;
287             }
288
289         case PA_CHANNEL_MAP_AUX: {
290             unsigned i;
291
292             for (i = 0; i < channels; i++)
293                 m->map[i] = PA_CHANNEL_POSITION_AUX0 + i;
294
295             return m;
296         }
297
298         case PA_CHANNEL_MAP_WAVEEX:
299
300             switch (channels) {
301                 case 1:
302                     m->map[0] = PA_CHANNEL_POSITION_MONO;
303                     return m;
304
305                 case 18:
306                     m->map[15] = PA_CHANNEL_POSITION_TOP_REAR_LEFT;
307                     m->map[16] = PA_CHANNEL_POSITION_TOP_REAR_CENTER;
308                     m->map[17] = PA_CHANNEL_POSITION_TOP_REAR_RIGHT;
309                     /* Fall through */
310
311                 case 15:
312                     m->map[12] = PA_CHANNEL_POSITION_TOP_FRONT_LEFT;
313                     m->map[13] = PA_CHANNEL_POSITION_TOP_FRONT_CENTER;
314                     m->map[14] = PA_CHANNEL_POSITION_TOP_FRONT_RIGHT;
315                     /* Fall through */
316
317                 case 12:
318                     m->map[11] = PA_CHANNEL_POSITION_TOP_CENTER;
319                     /* Fall through */
320
321                 case 11:
322                     m->map[9] = PA_CHANNEL_POSITION_SIDE_LEFT;
323                     m->map[10] = PA_CHANNEL_POSITION_SIDE_RIGHT;
324                     /* Fall through */
325
326                 case 9:
327                     m->map[8] = PA_CHANNEL_POSITION_REAR_CENTER;
328                     /* Fall through */
329
330                 case 8:
331                     m->map[6] = PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER;
332                     m->map[7] = PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER;
333                     /* Fall through */
334
335                 case 6:
336                     m->map[4] = PA_CHANNEL_POSITION_REAR_LEFT;
337                     m->map[5] = PA_CHANNEL_POSITION_REAR_RIGHT;
338                     /* Fall through */
339
340                 case 4:
341                     m->map[3] = PA_CHANNEL_POSITION_LFE;
342                     /* Fall through */
343
344                 case 3:
345                     m->map[2] = PA_CHANNEL_POSITION_FRONT_CENTER;
346                     /* Fall through */
347
348                 case 2:
349                     m->map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
350                     m->map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
351                     return m;
352
353                 default:
354                     return NULL;
355             }
356
357         case PA_CHANNEL_MAP_OSS:
358
359             switch (channels) {
360                 case 1:
361                     m->map[0] = PA_CHANNEL_POSITION_MONO;
362                     return m;
363
364                 case 8:
365                     m->map[6] = PA_CHANNEL_POSITION_REAR_LEFT;
366                     m->map[7] = PA_CHANNEL_POSITION_REAR_RIGHT;
367                     /* Fall through */
368
369                 case 6:
370                     m->map[4] = PA_CHANNEL_POSITION_SIDE_LEFT;
371                     m->map[5] = PA_CHANNEL_POSITION_SIDE_RIGHT;
372                     /* Fall through */
373
374                 case 4:
375                     m->map[3] = PA_CHANNEL_POSITION_LFE;
376                     /* Fall through */
377
378                 case 3:
379                     m->map[2] = PA_CHANNEL_POSITION_FRONT_CENTER;
380                     /* Fall through */
381
382                 case 2:
383                     m->map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
384                     m->map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
385                     return m;
386
387                 default:
388                     return NULL;
389             }
390
391
392         default:
393             pa_assert_not_reached();
394     }
395 }
396
397 pa_channel_map* pa_channel_map_init_extend(pa_channel_map *m, unsigned channels, pa_channel_map_def_t def) {
398     unsigned c;
399
400     pa_assert(m);
401     pa_assert(channels > 0);
402     pa_assert(channels <= PA_CHANNELS_MAX);
403     pa_assert(def < PA_CHANNEL_MAP_DEF_MAX);
404
405     pa_channel_map_init(m);
406
407     for (c = channels; c > 0; c--) {
408
409         if (pa_channel_map_init_auto(m, c, def)) {
410             unsigned i = 0;
411
412             for (; c < channels; c++) {
413
414                 m->map[c] = PA_CHANNEL_POSITION_AUX0 + i;
415                 i++;
416             }
417
418             m->channels = (uint8_t) channels;
419
420             return m;
421         }
422     }
423
424     return NULL;
425 }
426
427 const char* pa_channel_position_to_string(pa_channel_position_t pos) {
428
429     if (pos < 0 || pos >= PA_CHANNEL_POSITION_MAX)
430         return NULL;
431
432     return table[pos];
433 }
434
435 const char* pa_channel_position_to_pretty_string(pa_channel_position_t pos) {
436
437     if (pos < 0 || pos >= PA_CHANNEL_POSITION_MAX)
438         return NULL;
439
440     pa_init_i18n();
441
442     return _(pretty_table[pos]);
443 }
444
445 int pa_channel_map_equal(const pa_channel_map *a, const pa_channel_map *b) {
446     unsigned c;
447
448     pa_assert(a);
449     pa_assert(b);
450
451     pa_return_val_if_fail(pa_channel_map_valid(a), 0);
452     pa_return_val_if_fail(pa_channel_map_valid(b), 0);
453
454     if (a->channels != b->channels)
455         return 0;
456
457     for (c = 0; c < a->channels; c++)
458         if (a->map[c] != b->map[c])
459             return 0;
460
461     return 1;
462 }
463
464 char* pa_channel_map_snprint(char *s, size_t l, const pa_channel_map *map) {
465     unsigned channel;
466     pa_bool_t first = TRUE;
467     char *e;
468
469     pa_assert(s);
470     pa_assert(l > 0);
471     pa_assert(map);
472
473     pa_init_i18n();
474
475     if (!pa_channel_map_valid(map)) {
476         pa_snprintf(s, l, _("(invalid)"));
477         return s;
478     }
479
480     *(e = s) = 0;
481
482     for (channel = 0; channel < map->channels && l > 1; channel++) {
483         l -= pa_snprintf(e, l, "%s%s",
484                       first ? "" : ",",
485                       pa_channel_position_to_string(map->map[channel]));
486
487         e = strchr(e, 0);
488         first = FALSE;
489     }
490
491     return s;
492 }
493
494 pa_channel_map *pa_channel_map_parse(pa_channel_map *rmap, const char *s) {
495     const char *state;
496     pa_channel_map map;
497     char *p;
498
499     pa_assert(rmap);
500     pa_assert(s);
501
502     pa_channel_map_init(&map);
503
504     /* We don't need to match against the well known channel mapping
505      * "mono" here explicitly, because that can be understood as
506      * listing with one channel called "mono". */
507
508     if (pa_streq(s, "stereo")) {
509         map.channels = 2;
510         map.map[0] = PA_CHANNEL_POSITION_LEFT;
511         map.map[1] = PA_CHANNEL_POSITION_RIGHT;
512         goto finish;
513     } else if (pa_streq(s, "surround-40")) {
514         map.channels = 4;
515         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
516         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
517         map.map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
518         map.map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
519         goto finish;
520     } else if (pa_streq(s, "surround-41")) {
521         map.channels = 5;
522         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
523         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
524         map.map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
525         map.map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
526         map.map[4] = PA_CHANNEL_POSITION_LFE;
527         goto finish;
528     } else if (pa_streq(s, "surround-50")) {
529         map.channels = 5;
530         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
531         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
532         map.map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
533         map.map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
534         map.map[4] = PA_CHANNEL_POSITION_FRONT_CENTER;
535         goto finish;
536     } else if (pa_streq(s, "surround-51")) {
537         map.channels = 6;
538         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
539         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
540         map.map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
541         map.map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
542         map.map[4] = PA_CHANNEL_POSITION_FRONT_CENTER;
543         map.map[5] = PA_CHANNEL_POSITION_LFE;
544         goto finish;
545     } else if (pa_streq(s, "surround-71")) {
546         map.channels = 8;
547         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
548         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
549         map.map[2] = PA_CHANNEL_POSITION_REAR_LEFT;
550         map.map[3] = PA_CHANNEL_POSITION_REAR_RIGHT;
551         map.map[4] = PA_CHANNEL_POSITION_FRONT_CENTER;
552         map.map[5] = PA_CHANNEL_POSITION_LFE;
553         map.map[6] = PA_CHANNEL_POSITION_SIDE_LEFT;
554         map.map[7] = PA_CHANNEL_POSITION_SIDE_RIGHT;
555         goto finish;
556     }
557
558     state = NULL;
559     map.channels = 0;
560
561     while ((p = pa_split(s, ",", &state))) {
562
563         if (map.channels >= PA_CHANNELS_MAX) {
564             pa_xfree(p);
565             return NULL;
566         }
567
568         /* Some special aliases */
569         if (pa_streq(p, "left"))
570             map.map[map.channels++] = PA_CHANNEL_POSITION_LEFT;
571         else if (pa_streq(p, "right"))
572             map.map[map.channels++] = PA_CHANNEL_POSITION_RIGHT;
573         else if (pa_streq(p, "center"))
574             map.map[map.channels++] = PA_CHANNEL_POSITION_CENTER;
575         else if (pa_streq(p, "subwoofer"))
576             map.map[map.channels++] = PA_CHANNEL_POSITION_SUBWOOFER;
577         else {
578             pa_channel_position_t i;
579
580             for (i = 0; i < PA_CHANNEL_POSITION_MAX; i++)
581                 if (strcmp(p, table[i]) == 0) {
582                     map.map[map.channels++] = i;
583                     break;
584                 }
585
586             if (i >= PA_CHANNEL_POSITION_MAX) {
587                 pa_xfree(p);
588                 return NULL;
589             }
590         }
591
592         pa_xfree(p);
593     }
594
595 finish:
596
597     if (!pa_channel_map_valid(&map))
598         return NULL;
599
600     *rmap = map;
601     return rmap;
602 }
603
604 int pa_channel_map_valid(const pa_channel_map *map) {
605     unsigned c;
606
607     pa_assert(map);
608
609     if (map->channels <= 0 || map->channels > PA_CHANNELS_MAX)
610         return 0;
611
612     for (c = 0; c < map->channels; c++)
613         if (map->map[c] < 0 || map->map[c] >= PA_CHANNEL_POSITION_MAX)
614             return 0;
615
616     return 1;
617 }
618
619 int pa_channel_map_compatible(const pa_channel_map *map, const pa_sample_spec *ss) {
620     pa_assert(map);
621     pa_assert(ss);
622
623     pa_return_val_if_fail(pa_channel_map_valid(map), 0);
624     pa_return_val_if_fail(pa_sample_spec_valid(ss), 0);
625
626     return map->channels == ss->channels;
627 }
628
629 int pa_channel_map_superset(const pa_channel_map *a, const pa_channel_map *b) {
630     pa_bitset_t in_a[PA_BITSET_ELEMENTS(PA_CHANNEL_POSITION_MAX)];
631     unsigned i;
632
633     pa_assert(a);
634     pa_assert(b);
635
636     pa_return_val_if_fail(pa_channel_map_valid(a), 0);
637     pa_return_val_if_fail(pa_channel_map_valid(b), 0);
638
639     memset(in_a, 0, sizeof(in_a));
640
641     for (i = 0; i < a->channels; i++)
642         pa_bitset_set(in_a, a->map[i], TRUE);
643
644     for (i = 0; i < b->channels; i++)
645         if (!pa_bitset_get(in_a, b->map[i]))
646             return 0;
647
648     return 1;
649 }
650
651 int pa_channel_map_can_balance(const pa_channel_map *map) {
652     unsigned c;
653     pa_bool_t left = FALSE, right = FALSE;
654
655     pa_assert(map);
656
657     pa_return_val_if_fail(pa_channel_map_valid(map), 0);
658
659     for (c = 0; c < map->channels; c++) {
660
661         switch (map->map[c]) {
662             case PA_CHANNEL_POSITION_LEFT:
663             case PA_CHANNEL_POSITION_REAR_LEFT:
664             case PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER:
665             case PA_CHANNEL_POSITION_SIDE_LEFT:
666             case PA_CHANNEL_POSITION_TOP_FRONT_LEFT:
667             case PA_CHANNEL_POSITION_TOP_REAR_LEFT:
668                 left = TRUE;
669                 break;
670
671             case PA_CHANNEL_POSITION_RIGHT:
672             case PA_CHANNEL_POSITION_REAR_RIGHT:
673             case PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER:
674             case PA_CHANNEL_POSITION_SIDE_RIGHT:
675             case PA_CHANNEL_POSITION_TOP_FRONT_RIGHT:
676             case PA_CHANNEL_POSITION_TOP_REAR_RIGHT:
677                 right = TRUE;
678                 break;
679
680             default:
681                 ;
682         }
683
684         if (left && right)
685             return 1;
686     }
687
688     return 0;
689 }
690
691 int pa_channel_map_can_fade(const pa_channel_map *map) {
692     unsigned c;
693     pa_bool_t front = FALSE, rear = FALSE;
694
695     pa_assert(map);
696
697     pa_return_val_if_fail(pa_channel_map_valid(map), 0);
698
699     for (c = 0; c < map->channels; c++) {
700
701         switch (map->map[c]) {
702             case PA_CHANNEL_POSITION_FRONT_LEFT:
703             case PA_CHANNEL_POSITION_FRONT_RIGHT:
704             case PA_CHANNEL_POSITION_FRONT_CENTER:
705             case PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER:
706             case PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER:
707             case PA_CHANNEL_POSITION_TOP_FRONT_LEFT:
708             case PA_CHANNEL_POSITION_TOP_FRONT_RIGHT:
709             case PA_CHANNEL_POSITION_TOP_FRONT_CENTER:
710                 front = TRUE;
711                 break;
712
713             case PA_CHANNEL_POSITION_REAR_LEFT:
714             case PA_CHANNEL_POSITION_REAR_RIGHT:
715             case PA_CHANNEL_POSITION_REAR_CENTER:
716             case PA_CHANNEL_POSITION_TOP_REAR_LEFT:
717             case PA_CHANNEL_POSITION_TOP_REAR_RIGHT:
718             case PA_CHANNEL_POSITION_TOP_REAR_CENTER:
719                 rear = TRUE;
720                 break;
721
722             default:
723                 ;
724         }
725
726         if (front && rear)
727             return 1;
728     }
729
730     return 0;
731 }
732
733 const char* pa_channel_map_to_name(const pa_channel_map *map) {
734     pa_bitset_t in_map[PA_BITSET_ELEMENTS(PA_CHANNEL_POSITION_MAX)];
735     unsigned c;
736
737     pa_assert(map);
738
739     pa_return_val_if_fail(pa_channel_map_valid(map), NULL);
740
741     memset(in_map, 0, sizeof(in_map));
742
743     for (c = 0; c < map->channels; c++)
744         pa_bitset_set(in_map, map->map[c], TRUE);
745
746     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
747                          PA_CHANNEL_POSITION_MONO, -1))
748         return "mono";
749
750     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
751                          PA_CHANNEL_POSITION_LEFT, PA_CHANNEL_POSITION_RIGHT, -1))
752         return "stereo";
753
754     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
755                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
756                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT, -1))
757         return "surround-40";
758
759     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
760                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
761                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
762                          PA_CHANNEL_POSITION_LFE, -1))
763         return "surround-41";
764
765     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
766                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
767                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
768                          PA_CHANNEL_POSITION_FRONT_CENTER, -1))
769         return "surround-50";
770
771     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
772                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
773                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
774                          PA_CHANNEL_POSITION_FRONT_CENTER, PA_CHANNEL_POSITION_LFE, -1))
775         return "surround-51";
776
777     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
778                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
779                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
780                          PA_CHANNEL_POSITION_FRONT_CENTER, PA_CHANNEL_POSITION_LFE,
781                          PA_CHANNEL_POSITION_SIDE_LEFT, PA_CHANNEL_POSITION_SIDE_RIGHT, -1))
782         return "surround-71";
783
784     return NULL;
785 }
786
787 const char* pa_channel_map_to_pretty_name(const pa_channel_map *map) {
788     pa_bitset_t in_map[PA_BITSET_ELEMENTS(PA_CHANNEL_POSITION_MAX)];
789     unsigned c;
790
791     pa_assert(map);
792
793     pa_return_val_if_fail(pa_channel_map_valid(map), NULL);
794
795     memset(in_map, 0, sizeof(in_map));
796
797     for (c = 0; c < map->channels; c++)
798         pa_bitset_set(in_map, map->map[c], TRUE);
799
800     pa_init_i18n();
801
802     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
803                          PA_CHANNEL_POSITION_MONO, -1))
804         return _("Mono");
805
806     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
807                          PA_CHANNEL_POSITION_LEFT, PA_CHANNEL_POSITION_RIGHT, -1))
808         return _("Stereo");
809
810     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
811                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
812                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT, -1))
813         return _("Surround 4.0");
814
815     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
816                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
817                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
818                          PA_CHANNEL_POSITION_LFE, -1))
819         return _("Surround 4.1");
820
821     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
822                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
823                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
824                          PA_CHANNEL_POSITION_FRONT_CENTER, -1))
825         return _("Surround 5.0");
826
827     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
828                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
829                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
830                          PA_CHANNEL_POSITION_FRONT_CENTER, PA_CHANNEL_POSITION_LFE, -1))
831         return _("Surround 5.1");
832
833     if (pa_bitset_equals(in_map, PA_CHANNEL_POSITION_MAX,
834                          PA_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_RIGHT,
835                          PA_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_RIGHT,
836                          PA_CHANNEL_POSITION_FRONT_CENTER, PA_CHANNEL_POSITION_LFE,
837                          PA_CHANNEL_POSITION_SIDE_LEFT, PA_CHANNEL_POSITION_SIDE_RIGHT, -1))
838         return _("Surround 7.1");
839
840     return NULL;
841 }