xref: /illumos-gate/usr/src/cmd/audio/utilities/AudioTypeG72X.cc (revision 7d98732934f32a5f80fe624c9689e8f54dd17ffe)
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