1 /*
2 * CDDL HEADER START
3 *
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License, Version 1.0 only
6 * (the "License"). You may not use this file except in compliance
7 * with the License.
8 *
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
13 *
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
19 *
20 * CDDL HEADER END
21 */
22 /*
23 * Copyright (c) 1992-2001 by Sun Microsystems, Inc.
24 * All rights reserved.
25 */
26
27 #include <stdlib.h>
28 #include <memory.h>
29 #include <math.h>
30
31 #include <AudioTypeG72X.h>
32
33 // class AudioTypeG72X methods
34 // G.721 & G.723 compress/decompress
35
36 // Constructor
37 AudioTypeG72X::
AudioTypeG72X()38 AudioTypeG72X()
39 {
40 initialized = FALSE;
41 }
42
43 // Destructor
44 AudioTypeG72X::
~AudioTypeG72X()45 ~AudioTypeG72X()
46 {
47 }
48
49 // Test conversion possibilities.
50 // Return TRUE if conversion to/from the specified type is possible.
51 Boolean AudioTypeG72X::
CanConvert(AudioHdr h) const52 CanConvert(
53 AudioHdr h) const // target header
54 {
55 // g72x conversion code handles mono 16-bit pcm, ulaw, alaw
56 if (h.channels != 1)
57 return (FALSE);
58
59 switch (h.encoding) {
60 case LINEAR:
61 if ((h.samples_per_unit != 1) ||
62 (h.bytes_per_unit != 2))
63 return (FALSE);
64 break;
65 case ALAW:
66 case ULAW:
67 if ((h.samples_per_unit != 1) ||
68 (h.bytes_per_unit != 1))
69 return (FALSE);
70 break;
71 case G721:
72 if ((h.samples_per_unit != 2) ||
73 (h.bytes_per_unit != 1))
74 return (FALSE);
75 break;
76 case G723:
77 if (h.samples_per_unit != 8)
78 return (FALSE);
79
80 // XXX - 5-bit G.722 not supported yet
81 if (h.bytes_per_unit != 3)
82 return (FALSE);
83 break;
84 case FLOAT:
85 default:
86 return (FALSE);
87 }
88 return (TRUE);
89 }
90
91 // Convert buffer to the specified type
92 // May replace the buffer with a new one, if necessary
93 AudioError AudioTypeG72X::
Convert(AudioBuffer * & inbuf,AudioHdr outhdr)94 Convert(
95 AudioBuffer*& inbuf, // data buffer to process
96 AudioHdr outhdr) // target header
97 {
98 AudioBuffer* outbuf;
99 AudioHdr inhdr;
100 Audio_hdr chdr; // C struct for g72x convert code
101 Double length;
102 Double pad;
103 size_t nbytes;
104 int cnt;
105 unsigned char *inptr;
106 unsigned char *outptr;
107 AudioError err;
108
109 inhdr = inbuf->GetHeader();
110 length = inbuf->GetLength();
111
112 if (Undefined(length)) {
113 return (AUDIO_ERR_BADARG);
114 }
115
116 // Make sure we're not being asked to do the impossible
117 if ((err = inhdr.Validate()) || (err = outhdr.Validate())) {
118 return (err);
119 }
120
121 if (!CanConvert(inhdr) || !CanConvert(outhdr) ||
122 (inhdr.sample_rate != outhdr.sample_rate) ||
123 (inhdr.channels != outhdr.channels))
124 return (AUDIO_ERR_HDRINVAL);
125
126 // if conversion is a no-op, just return success
127 if ((inhdr.encoding == outhdr.encoding) &&
128 (inhdr.bytes_per_unit == outhdr.bytes_per_unit)) {
129 return (AUDIO_SUCCESS);
130 }
131
132 // Add some padding to the output buffer
133 pad = outhdr.Samples_to_Time(
134 4 * outhdr.bytes_per_unit * outhdr.channels);
135
136 // Allocate a new buffer
137 outbuf = new AudioBuffer(length + pad, "(G72x conversion buffer)");
138 if (outbuf == 0)
139 return (AUDIO_UNIXERROR);
140 err = outbuf->SetHeader(outhdr);
141 if (err != AUDIO_SUCCESS) {
142 delete outbuf;
143 return (err);
144 }
145
146 // Convert from the input type to the output type
147 inptr = (unsigned char *)inbuf->GetAddress();
148 outptr = (unsigned char *)outbuf->GetAddress();
149 nbytes = (size_t)inhdr.Time_to_Bytes(length);
150 if (nbytes == 0)
151 goto cleanup;
152
153 switch (inhdr.encoding) {
154 case ALAW:
155 case ULAW:
156 case LINEAR:
157 switch (outhdr.encoding) {
158 case G721:
159 chdr = (Audio_hdr)inhdr;
160 if (!initialized) {
161 g721_init_state(&g72x_state);
162 initialized = TRUE;
163 }
164 err = g721_encode((void*)inptr, nbytes, &chdr,
165 outptr, &cnt, &g72x_state);
166 length = outhdr.Bytes_to_Time(cnt);
167 break;
168 case G723:
169 chdr = (Audio_hdr)inhdr;
170 if (!initialized) {
171 g723_init_state(&g72x_state);
172 initialized = TRUE;
173 }
174 err = g723_encode((void*)inptr, nbytes, &chdr,
175 outptr, &cnt, &g72x_state);
176 length = outhdr.Bytes_to_Time(cnt);
177 break;
178 default:
179 err = AUDIO_ERR_HDRINVAL; break;
180 }
181 break;
182 case G721:
183 switch (outhdr.encoding) {
184 case ALAW:
185 case ULAW:
186 case LINEAR:
187 chdr = (Audio_hdr)outhdr;
188 if (!initialized) {
189 g721_init_state(&g72x_state);
190 initialized = TRUE;
191 }
192 err = g721_decode(inptr, nbytes, &chdr,
193 (void*)outptr, &cnt, &g72x_state);
194 length = outhdr.Samples_to_Time(cnt);
195 break;
196 default:
197 err = AUDIO_ERR_HDRINVAL; break;
198 }
199 break;
200 case G723:
201 switch (outhdr.encoding) {
202 case ALAW:
203 case ULAW:
204 case LINEAR:
205 chdr = (Audio_hdr)outhdr;
206 if (!initialized) {
207 g723_init_state(&g72x_state);
208 initialized = TRUE;
209 }
210 err = g723_decode(inptr, nbytes, &chdr,
211 (void*)outptr, &cnt, &g72x_state);
212 length = outhdr.Samples_to_Time(cnt);
213 break;
214 default:
215 err = AUDIO_ERR_HDRINVAL; break;
216 }
217 break;
218 default:
219 err = AUDIO_ERR_HDRINVAL; break;
220 }
221 if (err) {
222 if (outbuf != inbuf)
223 delete outbuf;
224 return (err);
225 }
226 cleanup:
227 // This will delete the buffer
228 inbuf->Reference();
229 inbuf->Dereference();
230
231 // Set the valid data length
232 outbuf->SetLength(length);
233 inbuf = outbuf;
234
235 return (AUDIO_SUCCESS);
236 }
237
238 // Flush out any leftover state, appending to supplied buffer
239 AudioError AudioTypeG72X::
Flush(AudioBuffer * & outbuf)240 Flush(
241 AudioBuffer*& outbuf)
242 {
243 AudioHdr h;
244 Double pos;
245 size_t cnt;
246 AudioError err;
247 unsigned char tmpbuf[32];
248
249 if (!initialized)
250 return (AUDIO_SUCCESS);
251 initialized = FALSE;
252 if (outbuf == NULL)
253 return (AUDIO_SUCCESS);
254
255 h = outbuf->GetHeader();
256
257 switch (h.encoding) {
258 case G721:
259 case G723:
260 switch (h.encoding) {
261 case G721:
262 err = g721_encode(NULL, 0, NULL,
263 tmpbuf, (int *)&cnt, &g72x_state);
264 break;
265 case G723:
266 err = g723_encode(NULL, 0, NULL,
267 tmpbuf, (int *)&cnt, &g72x_state);
268 break;
269 }
270 // Copy to the supplied buffer
271 if (cnt > 0) {
272 pos = outbuf->GetLength();
273 err = outbuf->AppendData(tmpbuf, cnt, pos);
274 if (err)
275 return (err);
276 }
277 break;
278 default:
279 break;
280 }
281 return (AUDIO_SUCCESS);
282 }
283