]> code.delx.au - pulseaudio/blob - src/pulse/format.c
af24eb2e08074845b7f66802ad5379ac36487b64
[pulseaudio] / src / pulse / format.c
1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2011 Intel Corporation
5 Copyright 2011 Collabora Multimedia
6 Copyright 2011 Arun Raghavan <arun.raghavan@collabora.co.uk>
7
8 PulseAudio is free software; you can redistribute it and/or modify
9 it under the terms of the GNU Lesser General Public License as published
10 by the Free Software Foundation; either version 2.1 of the License,
11 or (at your option) any later version.
12
13 PulseAudio is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 General Public License for more details.
17
18 You should have received a copy of the GNU Lesser General Public License
19 along with PulseAudio; if not, write to the Free Software
20 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
21 USA.
22 ***/
23
24 #ifdef HAVE_CONFIG_H
25 #include <config.h>
26 #endif
27
28 #include <pulse/internal.h>
29 #include <pulse/xmalloc.h>
30
31 #include <pulsecore/core-util.h>
32 #include <pulsecore/macro.h>
33
34 #include "format.h"
35
36 pa_format_info* pa_format_info_new(void) {
37 pa_format_info *f = pa_xnew(pa_format_info, 1);
38
39 f->encoding = PA_ENCODING_INVALID;
40 f->plist = pa_proplist_new();
41
42 return f;
43 }
44
45 pa_format_info* pa_format_info_copy(const pa_format_info *src) {
46 pa_format_info *dest;
47
48 pa_assert(src);
49
50 dest = pa_xnew(pa_format_info, 1);
51
52 dest->encoding = src->encoding;
53
54 if (src->plist)
55 dest->plist = pa_proplist_copy(src->plist);
56 else
57 dest->plist = NULL;
58
59 return dest;
60 }
61
62 void pa_format_info_free(pa_format_info *f) {
63 pa_assert(f);
64
65 pa_proplist_free(f->plist);
66 pa_xfree(f);
67 }
68
69 int pa_format_info_valid(pa_format_info *f) {
70 return (f->encoding >= 0 && f->encoding < PA_ENCODING_MAX && f->plist != NULL);
71 }
72
73 int pa_format_info_is_pcm(pa_format_info *f) {
74 return f->encoding == PA_ENCODING_PCM;
75 }
76
77 pa_bool_t pa_format_info_is_compatible(pa_format_info *first, pa_format_info *second) {
78 const char *key;
79 void *state = NULL;
80
81 pa_assert(first);
82 pa_assert(second);
83
84 if (first->encoding != second->encoding)
85 return FALSE;
86
87 while ((key = pa_proplist_iterate(first->plist, &state))) {
88 const char *value_one, *value_two;
89
90 value_one = pa_proplist_gets(first->plist, key);
91 value_two = pa_proplist_gets(second->plist, key);
92
93 if (!value_two || !pa_streq(value_one, value_two))
94 return FALSE;
95 }
96
97 return TRUE;
98 }
99
100 pa_format_info* pa_format_info_from_sample_spec(pa_sample_spec *ss, pa_channel_map *map) {
101 char cm[PA_CHANNEL_MAP_SNPRINT_MAX];
102 pa_format_info *f;
103
104 pa_assert(ss && pa_sample_spec_valid(ss));
105 pa_assert(!map || pa_channel_map_valid(map));
106
107 f = pa_format_info_new();
108 f->encoding = PA_ENCODING_PCM;
109
110 pa_proplist_sets(f->plist, PA_PROP_FORMAT_SAMPLE_FORMAT, pa_sample_format_to_string(ss->format));
111 pa_proplist_setf(f->plist, PA_PROP_FORMAT_RATE, "%u", (unsigned int) ss->rate);
112 pa_proplist_setf(f->plist, PA_PROP_FORMAT_CHANNELS, "%u", (unsigned int) ss->channels);
113
114 if (map) {
115 pa_channel_map_snprint(cm, sizeof(cm), map);
116 pa_proplist_setf(f->plist, PA_PROP_FORMAT_CHANNEL_MAP, "%s", cm);
117 }
118
119 return f;
120 }
121
122 /* For PCM streams */
123 pa_bool_t pa_format_info_to_sample_spec(pa_format_info *f, pa_sample_spec *ss, pa_channel_map *map) {
124 const char *sf, *r, *ch;
125 uint32_t channels;
126
127 pa_assert(f);
128 pa_assert(ss);
129 pa_return_val_if_fail(f->encoding == PA_ENCODING_PCM, FALSE);
130
131 pa_return_val_if_fail(sf = pa_proplist_gets(f->plist, PA_PROP_FORMAT_SAMPLE_FORMAT), FALSE);
132 pa_return_val_if_fail(r = pa_proplist_gets(f->plist, PA_PROP_FORMAT_RATE), FALSE);
133 pa_return_val_if_fail(ch = pa_proplist_gets(f->plist, PA_PROP_FORMAT_CHANNELS), FALSE);
134
135 pa_return_val_if_fail((ss->format = pa_parse_sample_format(sf)) != PA_SAMPLE_INVALID, FALSE);
136 pa_return_val_if_fail(pa_atou(r, &ss->rate) == 0, FALSE);
137 pa_return_val_if_fail(pa_atou(ch, &channels) == 0, FALSE);
138 ss->channels = (uint8_t) channels;
139
140 if (map) {
141 const char *m = pa_proplist_gets(f->plist, PA_PROP_FORMAT_CHANNEL_MAP);
142 pa_channel_map_init(map);
143
144 if (m)
145 pa_return_val_if_fail(pa_channel_map_parse(map, m) != NULL, FALSE);
146 }
147
148 return TRUE;
149 }
150
151 /* For compressed streams */
152 pa_bool_t pa_format_info_to_sample_spec_fake(pa_format_info *f, pa_sample_spec *ss) {
153 const char *r;
154
155 pa_assert(f);
156 pa_assert(ss);
157 pa_return_val_if_fail(f->encoding != PA_ENCODING_PCM, FALSE);
158
159 ss->format = PA_SAMPLE_S16LE;
160 ss->channels = 2;
161
162 pa_return_val_if_fail(r = pa_proplist_gets(f->plist, PA_PROP_FORMAT_RATE), FALSE);
163 pa_return_val_if_fail(pa_atou(r, &ss->rate) == 0, FALSE);
164
165 return TRUE;
166 }