1b815c7f3Sopenharmony_ci/*
2b815c7f3Sopenharmony_ci** Copyright (C) 2007-2011 Erik de Castro Lopo <erikd@mega-nerd.com>
3b815c7f3Sopenharmony_ci**
4b815c7f3Sopenharmony_ci** This program is free software; you can redistribute it and/or modify
5b815c7f3Sopenharmony_ci** it under the terms of the GNU Lesser General Public License as published by
6b815c7f3Sopenharmony_ci** the Free Software Foundation; either version 2.1 of the License, or
7b815c7f3Sopenharmony_ci** (at your option) any later version.
8b815c7f3Sopenharmony_ci**
9b815c7f3Sopenharmony_ci** This program is distributed in the hope that it will be useful,
10b815c7f3Sopenharmony_ci** but WITHOUT ANY WARRANTY; without even the implied warranty of
11b815c7f3Sopenharmony_ci** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12b815c7f3Sopenharmony_ci** GNU Lesser General Public License for more details.
13b815c7f3Sopenharmony_ci**
14b815c7f3Sopenharmony_ci** You should have received a copy of the GNU Lesser General Public License
15b815c7f3Sopenharmony_ci** along with this program; if not, write to the Free Software
16b815c7f3Sopenharmony_ci** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17b815c7f3Sopenharmony_ci*/
18b815c7f3Sopenharmony_ci
19b815c7f3Sopenharmony_ci#include <octave/oct.h>
20b815c7f3Sopenharmony_ci
21b815c7f3Sopenharmony_ci#include "sndfile.h"
22b815c7f3Sopenharmony_ci
23b815c7f3Sopenharmony_ci#define FOUR_GIG 		(0x100000000LL)
24b815c7f3Sopenharmony_ci#define	BUFFER_FRAMES	8192
25b815c7f3Sopenharmony_ci
26b815c7f3Sopenharmony_ci
27b815c7f3Sopenharmony_cistatic int format_of_str (const std::string & fmt) ;
28b815c7f3Sopenharmony_cistatic void string_of_format (std::string & fmt, int format) ;
29b815c7f3Sopenharmony_ci
30b815c7f3Sopenharmony_ci
31b815c7f3Sopenharmony_ciDEFUN_DLD (sfversion, args, nargout ,
32b815c7f3Sopenharmony_ci"-*- texinfo -*-\n\
33b815c7f3Sopenharmony_ci@deftypefn {Loadable Function} {@var{version} =} sfversion ()\n\
34b815c7f3Sopenharmony_ci@cindex Reading sound files\n\
35b815c7f3Sopenharmony_ciReturn a string containing the libsndfile version.\n\
36b815c7f3Sopenharmony_ci@seealso{sfread, sfwrite}\n\
37b815c7f3Sopenharmony_ci@end deftypefn")
38b815c7f3Sopenharmony_ci{	char buffer [256] ;
39b815c7f3Sopenharmony_ci	octave_value_list retval ;
40b815c7f3Sopenharmony_ci
41b815c7f3Sopenharmony_ci	/* Bail out if the input parameters are bad. */
42b815c7f3Sopenharmony_ci	if (args.length () != 0 || nargout > 1)
43b815c7f3Sopenharmony_ci	{	print_usage () ;
44b815c7f3Sopenharmony_ci		return retval ;
45b815c7f3Sopenharmony_ci		} ;
46b815c7f3Sopenharmony_ci
47b815c7f3Sopenharmony_ci	sf_command (NULL, SFC_GET_LIB_VERSION, buffer, sizeof (buffer)) ;
48b815c7f3Sopenharmony_ci
49b815c7f3Sopenharmony_ci	std::string version (buffer) ;
50b815c7f3Sopenharmony_ci
51b815c7f3Sopenharmony_ci	retval.append (version) ;
52b815c7f3Sopenharmony_ci	return retval ;
53b815c7f3Sopenharmony_ci} /* sfversion */
54b815c7f3Sopenharmony_ci
55b815c7f3Sopenharmony_ci
56b815c7f3Sopenharmony_ciDEFUN_DLD (sfread, args, nargout ,
57b815c7f3Sopenharmony_ci"-*- texinfo -*-\n\
58b815c7f3Sopenharmony_ci@deftypefn {Loadable Function} {@var{data},@var{srate},@var{format} =} sfread (@var{filename})\n\
59b815c7f3Sopenharmony_ci@cindex Reading sound files\n\
60b815c7f3Sopenharmony_ciRead a sound file from disk using libsndfile.\n\
61b815c7f3Sopenharmony_ci@seealso{sfversion, sfwrite}\n\
62b815c7f3Sopenharmony_ci@end deftypefn")
63b815c7f3Sopenharmony_ci{	SNDFILE * file ;
64b815c7f3Sopenharmony_ci	SF_INFO sfinfo ;
65b815c7f3Sopenharmony_ci
66b815c7f3Sopenharmony_ci	octave_value_list retval ;
67b815c7f3Sopenharmony_ci
68b815c7f3Sopenharmony_ci	int nargin  = args.length () ;
69b815c7f3Sopenharmony_ci
70b815c7f3Sopenharmony_ci	/* Bail out if the input parameters are bad. */
71b815c7f3Sopenharmony_ci	if ((nargin != 1) || !args (0) .is_string () || nargout < 1 || nargout > 3)
72b815c7f3Sopenharmony_ci	{	print_usage () ;
73b815c7f3Sopenharmony_ci		return retval ;
74b815c7f3Sopenharmony_ci		} ;
75b815c7f3Sopenharmony_ci
76b815c7f3Sopenharmony_ci	memset (&sfinfo, 0, sizeof (sfinfo)) ;
77b815c7f3Sopenharmony_ci
78b815c7f3Sopenharmony_ci	std::string filename = args (0).string_value () ;
79b815c7f3Sopenharmony_ci
80b815c7f3Sopenharmony_ci	if ((file = sf_open (filename.c_str (), SFM_READ, &sfinfo)) == NULL)
81b815c7f3Sopenharmony_ci	{	error ("sfread: couldn't open file %s : %s", filename.c_str (), sf_strerror (NULL)) ;
82b815c7f3Sopenharmony_ci		return retval ;
83b815c7f3Sopenharmony_ci		} ;
84b815c7f3Sopenharmony_ci
85b815c7f3Sopenharmony_ci	if (sfinfo.frames > FOUR_GIG)
86b815c7f3Sopenharmony_ci		printf ("This is a really huge file (%lld frames).\nYou may run out of memory trying to load it.\n", (long long) sfinfo.frames) ;
87b815c7f3Sopenharmony_ci
88b815c7f3Sopenharmony_ci	dim_vector dim = dim_vector () ;
89b815c7f3Sopenharmony_ci	dim.resize (2) ;
90b815c7f3Sopenharmony_ci	dim (0) = sfinfo.frames ;
91b815c7f3Sopenharmony_ci	dim (1) = sfinfo.channels ;
92b815c7f3Sopenharmony_ci
93b815c7f3Sopenharmony_ci	/* Should I be using Matrix instead? */
94b815c7f3Sopenharmony_ci	NDArray out (dim, 0.0) ;
95b815c7f3Sopenharmony_ci
96b815c7f3Sopenharmony_ci	float buffer [BUFFER_FRAMES * sfinfo.channels] ;
97b815c7f3Sopenharmony_ci	int readcount ;
98b815c7f3Sopenharmony_ci	sf_count_t total = 0 ;
99b815c7f3Sopenharmony_ci
100b815c7f3Sopenharmony_ci	do
101b815c7f3Sopenharmony_ci	{	readcount = sf_readf_float (file, buffer, BUFFER_FRAMES) ;
102b815c7f3Sopenharmony_ci
103b815c7f3Sopenharmony_ci		/* Make sure we don't read more frames than we allocated. */
104b815c7f3Sopenharmony_ci		if (total + readcount > sfinfo.frames)
105b815c7f3Sopenharmony_ci			readcount = sfinfo.frames - total ;
106b815c7f3Sopenharmony_ci
107b815c7f3Sopenharmony_ci		for (int ch = 0 ; ch < sfinfo.channels ; ch++)
108b815c7f3Sopenharmony_ci		{	for (int k = 0 ; k < readcount ; k++)
109b815c7f3Sopenharmony_ci				out (total + k, ch) = buffer [k * sfinfo.channels + ch] ;
110b815c7f3Sopenharmony_ci			} ;
111b815c7f3Sopenharmony_ci
112b815c7f3Sopenharmony_ci		total += readcount ;
113b815c7f3Sopenharmony_ci	} while (readcount > 0 && total < sfinfo.frames) ;
114b815c7f3Sopenharmony_ci
115b815c7f3Sopenharmony_ci	retval.append (out.squeeze ()) ;
116b815c7f3Sopenharmony_ci
117b815c7f3Sopenharmony_ci	if (nargout >= 2)
118b815c7f3Sopenharmony_ci		retval.append ((octave_uint32) sfinfo.samplerate) ;
119b815c7f3Sopenharmony_ci
120b815c7f3Sopenharmony_ci	if (nargout >= 3)
121b815c7f3Sopenharmony_ci	{	std::string fmt ("") ;
122b815c7f3Sopenharmony_ci		string_of_format (fmt, sfinfo.format) ;
123b815c7f3Sopenharmony_ci		retval.append (fmt) ;
124b815c7f3Sopenharmony_ci		} ;
125b815c7f3Sopenharmony_ci
126b815c7f3Sopenharmony_ci	/* Clean up. */
127b815c7f3Sopenharmony_ci	sf_close (file) ;
128b815c7f3Sopenharmony_ci
129b815c7f3Sopenharmony_ci	return retval ;
130b815c7f3Sopenharmony_ci} /* sfread */
131b815c7f3Sopenharmony_ci
132b815c7f3Sopenharmony_ciDEFUN_DLD (sfwrite, args, nargout ,
133b815c7f3Sopenharmony_ci"-*- texinfo -*-\n\
134b815c7f3Sopenharmony_ci@deftypefn {Function File} sfwrite (@var{filename},@var{data},@var{srate},@var{format})\n\
135b815c7f3Sopenharmony_ciWrite a sound file to disk using libsndfile.\n\
136b815c7f3Sopenharmony_ci@seealso{sfread, sfversion}\n\
137b815c7f3Sopenharmony_ci@end deftypefn\n\
138b815c7f3Sopenharmony_ci")
139b815c7f3Sopenharmony_ci{	SNDFILE * file ;
140b815c7f3Sopenharmony_ci	SF_INFO sfinfo ;
141b815c7f3Sopenharmony_ci
142b815c7f3Sopenharmony_ci    octave_value_list retval ;
143b815c7f3Sopenharmony_ci
144b815c7f3Sopenharmony_ci    int nargin  = args.length () ;
145b815c7f3Sopenharmony_ci
146b815c7f3Sopenharmony_ci    /* Bail out if the input parameters are bad. */
147b815c7f3Sopenharmony_ci    if (nargin != 4 || !args (0).is_string () || !args (1).is_real_matrix ()
148b815c7f3Sopenharmony_ci			|| !args (2).is_real_scalar () || !args (3).is_string ()
149b815c7f3Sopenharmony_ci			|| nargout != 0)
150b815c7f3Sopenharmony_ci	{	print_usage () ;
151b815c7f3Sopenharmony_ci		return retval ;
152b815c7f3Sopenharmony_ci    	} ;
153b815c7f3Sopenharmony_ci
154b815c7f3Sopenharmony_ci    std::string filename = args (0).string_value () ;
155b815c7f3Sopenharmony_ci    std::string format = args (3).string_value () ;
156b815c7f3Sopenharmony_ci
157b815c7f3Sopenharmony_ci	memset (&sfinfo, 0, sizeof (sfinfo)) ;
158b815c7f3Sopenharmony_ci
159b815c7f3Sopenharmony_ci	sfinfo.format = format_of_str (format) ;
160b815c7f3Sopenharmony_ci	if (sfinfo.format == 0)
161b815c7f3Sopenharmony_ci	{	error ("Bad format '%s'", format.c_str ()) ;
162b815c7f3Sopenharmony_ci		return retval ;
163b815c7f3Sopenharmony_ci		} ;
164b815c7f3Sopenharmony_ci
165b815c7f3Sopenharmony_ci	sfinfo.samplerate = lrint (args (2).scalar_value ()) ;
166b815c7f3Sopenharmony_ci	if (sfinfo.samplerate < 1)
167b815c7f3Sopenharmony_ci	{	error ("Bad sample rate : %d.\n", sfinfo.samplerate) ;
168b815c7f3Sopenharmony_ci		return retval ;
169b815c7f3Sopenharmony_ci		} ;
170b815c7f3Sopenharmony_ci
171b815c7f3Sopenharmony_ci	Matrix data = args (1).matrix_value () ;
172b815c7f3Sopenharmony_ci	long rows = args (1).rows () ;
173b815c7f3Sopenharmony_ci	long cols = args (1).columns () ;
174b815c7f3Sopenharmony_ci
175b815c7f3Sopenharmony_ci	if (cols > rows)
176b815c7f3Sopenharmony_ci	{	error ("Audio data should have one column per channel, but supplied data "
177b815c7f3Sopenharmony_ci				"has %ld rows and %ld columns.\n", rows, cols) ;
178b815c7f3Sopenharmony_ci		return retval ;
179b815c7f3Sopenharmony_ci		} ;
180b815c7f3Sopenharmony_ci
181b815c7f3Sopenharmony_ci	sfinfo.channels = cols ;
182b815c7f3Sopenharmony_ci
183b815c7f3Sopenharmony_ci    if ((file = sf_open (filename.c_str (), SFM_WRITE, &sfinfo)) == NULL)
184b815c7f3Sopenharmony_ci	{	error ("Couldn't open file %s : %s", filename.c_str (), sf_strerror (NULL)) ;
185b815c7f3Sopenharmony_ci		return retval ;
186b815c7f3Sopenharmony_ci    	} ;
187b815c7f3Sopenharmony_ci
188b815c7f3Sopenharmony_ci	float buffer [BUFFER_FRAMES * sfinfo.channels] ;
189b815c7f3Sopenharmony_ci	int writecount ;
190b815c7f3Sopenharmony_ci	long total = 0 ;
191b815c7f3Sopenharmony_ci
192b815c7f3Sopenharmony_ci	do
193b815c7f3Sopenharmony_ci	{
194b815c7f3Sopenharmony_ci		writecount = BUFFER_FRAMES ;
195b815c7f3Sopenharmony_ci
196b815c7f3Sopenharmony_ci		/* Make sure we don't read more frames than we allocated. */
197b815c7f3Sopenharmony_ci		if (total + writecount > rows)
198b815c7f3Sopenharmony_ci			writecount = rows - total ;
199b815c7f3Sopenharmony_ci
200b815c7f3Sopenharmony_ci		for (int ch = 0 ; ch < sfinfo.channels ; ch++)
201b815c7f3Sopenharmony_ci		{	for (int k = 0 ; k < writecount ; k++)
202b815c7f3Sopenharmony_ci				buffer [k * sfinfo.channels + ch] = data (total + k, ch) ;
203b815c7f3Sopenharmony_ci			} ;
204b815c7f3Sopenharmony_ci
205b815c7f3Sopenharmony_ci		if (writecount > 0)
206b815c7f3Sopenharmony_ci			sf_writef_float (file, buffer, writecount) ;
207b815c7f3Sopenharmony_ci
208b815c7f3Sopenharmony_ci		total += writecount ;
209b815c7f3Sopenharmony_ci	} while (writecount > 0 && total < rows) ;
210b815c7f3Sopenharmony_ci
211b815c7f3Sopenharmony_ci    /* Clean up. */
212b815c7f3Sopenharmony_ci    sf_close (file) ;
213b815c7f3Sopenharmony_ci
214b815c7f3Sopenharmony_ci    return retval ;
215b815c7f3Sopenharmony_ci} /* sfwrite */
216b815c7f3Sopenharmony_ci
217b815c7f3Sopenharmony_ci
218b815c7f3Sopenharmony_cistatic void
219b815c7f3Sopenharmony_cistr_split (const std::string & str, const std::string & delim, std::vector <std::string> & output)
220b815c7f3Sopenharmony_ci{
221b815c7f3Sopenharmony_ci    unsigned int offset = 0 ;
222b815c7f3Sopenharmony_ci    size_t delim_index = 0 ;
223b815c7f3Sopenharmony_ci
224b815c7f3Sopenharmony_ci    delim_index = str.find (delim, offset) ;
225b815c7f3Sopenharmony_ci
226b815c7f3Sopenharmony_ci    while (delim_index != std::string::npos)
227b815c7f3Sopenharmony_ci    {
228b815c7f3Sopenharmony_ci        output.push_back (str.substr(offset, delim_index - offset)) ;
229b815c7f3Sopenharmony_ci        offset += delim_index - offset + delim.length () ;
230b815c7f3Sopenharmony_ci        delim_index = str.find (delim, offset) ;
231b815c7f3Sopenharmony_ci    }
232b815c7f3Sopenharmony_ci
233b815c7f3Sopenharmony_ci    output.push_back (str.substr (offset)) ;
234b815c7f3Sopenharmony_ci} /* str_split */
235b815c7f3Sopenharmony_ci
236b815c7f3Sopenharmony_cistatic int
237b815c7f3Sopenharmony_cihash_of_str (const std::string & str)
238b815c7f3Sopenharmony_ci{
239b815c7f3Sopenharmony_ci	int hash = 0 ;
240b815c7f3Sopenharmony_ci
241b815c7f3Sopenharmony_ci	for (unsigned k = 0 ; k < str.length () ; k++)
242b815c7f3Sopenharmony_ci		hash = (hash * 3) + tolower (str [k]) ;
243b815c7f3Sopenharmony_ci
244b815c7f3Sopenharmony_ci	return hash ;
245b815c7f3Sopenharmony_ci} /* hash_of_str */
246b815c7f3Sopenharmony_ci
247b815c7f3Sopenharmony_cistatic int
248b815c7f3Sopenharmony_cimajor_format_of_hash (const std::string & str)
249b815c7f3Sopenharmony_ci{	int hash ;
250b815c7f3Sopenharmony_ci
251b815c7f3Sopenharmony_ci	hash = hash_of_str (str) ;
252b815c7f3Sopenharmony_ci
253b815c7f3Sopenharmony_ci	switch (hash)
254b815c7f3Sopenharmony_ci	{
255b815c7f3Sopenharmony_ci		case 0x5c8 : /* 'wav' */ return SF_FORMAT_WAV ;
256b815c7f3Sopenharmony_ci		case 0xf84 : /* 'aiff' */ return SF_FORMAT_AIFF ;
257b815c7f3Sopenharmony_ci		case 0x198 : /* 'au' */ return SF_FORMAT_AU ;
258b815c7f3Sopenharmony_ci		case 0x579 : /* 'paf' */ return SF_FORMAT_PAF ;
259b815c7f3Sopenharmony_ci		case 0x5e5 : /* 'svx' */ return SF_FORMAT_SVX ;
260b815c7f3Sopenharmony_ci		case 0x1118 : /* 'nist' */ return SF_FORMAT_NIST ;
261b815c7f3Sopenharmony_ci		case 0x5d6 : /* 'voc' */ return SF_FORMAT_VOC ;
262b815c7f3Sopenharmony_ci		case 0x324a : /* 'ircam' */ return SF_FORMAT_IRCAM ;
263b815c7f3Sopenharmony_ci		case 0x505 : /* 'w64' */ return SF_FORMAT_W64 ;
264b815c7f3Sopenharmony_ci		case 0x1078 : /* 'mat4' */ return SF_FORMAT_MAT4 ;
265b815c7f3Sopenharmony_ci		case 0x1079 : /* 'mat5' */ return SF_FORMAT_MAT5 ;
266b815c7f3Sopenharmony_ci		case 0x5b8 : /* 'pvf' */ return SF_FORMAT_PVF ;
267b815c7f3Sopenharmony_ci		case 0x1d1 : /* 'xi' */ return SF_FORMAT_XI ;
268b815c7f3Sopenharmony_ci		case 0x56f : /* 'htk' */ return SF_FORMAT_HTK ;
269b815c7f3Sopenharmony_ci		case 0x5aa : /* 'sds' */ return SF_FORMAT_SDS ;
270b815c7f3Sopenharmony_ci		case 0x53d : /* 'avr' */ return SF_FORMAT_AVR ;
271b815c7f3Sopenharmony_ci		case 0x11d0 : /* 'wavx' */ return SF_FORMAT_WAVEX ;
272b815c7f3Sopenharmony_ci		case 0x569 : /* 'sd2' */ return SF_FORMAT_SD2 ;
273b815c7f3Sopenharmony_ci		case 0x1014 : /* 'flac' */ return SF_FORMAT_FLAC ;
274b815c7f3Sopenharmony_ci		case 0x504 : /* 'caf' */ return SF_FORMAT_CAF ;
275b815c7f3Sopenharmony_ci		case 0x5f6 : /* 'wve' */ return SF_FORMAT_WVE ;
276b815c7f3Sopenharmony_ci		default : break ;
277b815c7f3Sopenharmony_ci		} ;
278b815c7f3Sopenharmony_ci
279b815c7f3Sopenharmony_ci	printf ("%s : hash '%s' -> 0x%x\n", __func__, str.c_str (), hash) ;
280b815c7f3Sopenharmony_ci
281b815c7f3Sopenharmony_ci	return 0 ;
282b815c7f3Sopenharmony_ci} /* major_format_of_hash */
283b815c7f3Sopenharmony_ci
284b815c7f3Sopenharmony_cistatic int
285b815c7f3Sopenharmony_ciminor_format_of_hash (const std::string & str)
286b815c7f3Sopenharmony_ci{	int hash ;
287b815c7f3Sopenharmony_ci
288b815c7f3Sopenharmony_ci	hash = hash_of_str (str) ;
289b815c7f3Sopenharmony_ci
290b815c7f3Sopenharmony_ci	switch (hash)
291b815c7f3Sopenharmony_ci	{
292b815c7f3Sopenharmony_ci		case 0x1085 : /* 'int8' */ return SF_FORMAT_PCM_S8 ;
293b815c7f3Sopenharmony_ci		case 0x358a : /* 'uint8' */ return SF_FORMAT_PCM_U8 ;
294b815c7f3Sopenharmony_ci		case 0x31b0 : /* 'int16' */ return SF_FORMAT_PCM_16 ;
295b815c7f3Sopenharmony_ci		case 0x31b1 : /* 'int24' */ return SF_FORMAT_PCM_24 ;
296b815c7f3Sopenharmony_ci		case 0x31b2 : /* 'int32' */ return SF_FORMAT_PCM_32 ;
297b815c7f3Sopenharmony_ci		case 0x3128 : /* 'float' */ return SF_FORMAT_FLOAT ;
298b815c7f3Sopenharmony_ci		case 0x937d : /* 'double' */ return SF_FORMAT_DOUBLE ;
299b815c7f3Sopenharmony_ci		case 0x11bd : /* 'ulaw' */ return SF_FORMAT_ULAW ;
300b815c7f3Sopenharmony_ci		case 0xfa1 : /* 'alaw' */ return SF_FORMAT_ALAW ;
301b815c7f3Sopenharmony_ci		case 0xfc361 : /* 'ima_adpcm' */ return SF_FORMAT_IMA_ADPCM ;
302b815c7f3Sopenharmony_ci		case 0x5739a : /* 'ms_adpcm' */ return SF_FORMAT_MS_ADPCM ;
303b815c7f3Sopenharmony_ci		case 0x9450 : /* 'gsm610' */ return SF_FORMAT_GSM610 ;
304b815c7f3Sopenharmony_ci		case 0x172a3 : /* 'g721_32' */ return SF_FORMAT_G721_32 ;
305b815c7f3Sopenharmony_ci		case 0x172d8 : /* 'g723_24' */ return SF_FORMAT_G723_24 ;
306b815c7f3Sopenharmony_ci		case 0x172da : /* 'g723_40' */ return SF_FORMAT_G723_40 ;
307b815c7f3Sopenharmony_ci		default : break ;
308b815c7f3Sopenharmony_ci		} ;
309b815c7f3Sopenharmony_ci
310b815c7f3Sopenharmony_ci	printf ("%s : hash '%s' -> 0x%x\n", __func__, str.c_str (), hash) ;
311b815c7f3Sopenharmony_ci
312b815c7f3Sopenharmony_ci	return 0 ;
313b815c7f3Sopenharmony_ci} /* minor_format_of_hash */
314b815c7f3Sopenharmony_ci
315b815c7f3Sopenharmony_ci
316b815c7f3Sopenharmony_cistatic const char *
317b815c7f3Sopenharmony_cistring_of_major_format (int format)
318b815c7f3Sopenharmony_ci{
319b815c7f3Sopenharmony_ci	switch (format & SF_FORMAT_TYPEMASK)
320b815c7f3Sopenharmony_ci	{
321b815c7f3Sopenharmony_ci		case SF_FORMAT_WAV : return "wav" ;
322b815c7f3Sopenharmony_ci		case SF_FORMAT_AIFF : return "aiff" ;
323b815c7f3Sopenharmony_ci		case SF_FORMAT_AU : return "au" ;
324b815c7f3Sopenharmony_ci		case SF_FORMAT_PAF : return "paf" ;
325b815c7f3Sopenharmony_ci		case SF_FORMAT_SVX : return "svx" ;
326b815c7f3Sopenharmony_ci		case SF_FORMAT_NIST : return "nist" ;
327b815c7f3Sopenharmony_ci		case SF_FORMAT_VOC : return "voc" ;
328b815c7f3Sopenharmony_ci		case SF_FORMAT_IRCAM : return "ircam" ;
329b815c7f3Sopenharmony_ci		case SF_FORMAT_W64 : return "w64" ;
330b815c7f3Sopenharmony_ci		case SF_FORMAT_MAT4 : return "mat4" ;
331b815c7f3Sopenharmony_ci		case SF_FORMAT_MAT5 : return "mat5" ;
332b815c7f3Sopenharmony_ci		case SF_FORMAT_PVF : return "pvf" ;
333b815c7f3Sopenharmony_ci		case SF_FORMAT_XI : return "xi" ;
334b815c7f3Sopenharmony_ci		case SF_FORMAT_HTK : return "htk" ;
335b815c7f3Sopenharmony_ci		case SF_FORMAT_SDS : return "sds" ;
336b815c7f3Sopenharmony_ci		case SF_FORMAT_AVR : return "avr" ;
337b815c7f3Sopenharmony_ci		case SF_FORMAT_WAVEX : return "wavx" ;
338b815c7f3Sopenharmony_ci		case SF_FORMAT_SD2 : return "sd2" ;
339b815c7f3Sopenharmony_ci		case SF_FORMAT_FLAC : return "flac" ;
340b815c7f3Sopenharmony_ci		case SF_FORMAT_CAF : return "caf" ;
341b815c7f3Sopenharmony_ci		case SF_FORMAT_WVE : return "wfe" ;
342b815c7f3Sopenharmony_ci		default : break ;
343b815c7f3Sopenharmony_ci		} ;
344b815c7f3Sopenharmony_ci
345b815c7f3Sopenharmony_ci	return "unknown" ;
346b815c7f3Sopenharmony_ci} /* string_of_major_format */
347b815c7f3Sopenharmony_ci
348b815c7f3Sopenharmony_cistatic const char *
349b815c7f3Sopenharmony_cistring_of_minor_format (int format)
350b815c7f3Sopenharmony_ci{
351b815c7f3Sopenharmony_ci	switch (format & SF_FORMAT_SUBMASK)
352b815c7f3Sopenharmony_ci	{
353b815c7f3Sopenharmony_ci		case SF_FORMAT_PCM_S8 : return "int8" ;
354b815c7f3Sopenharmony_ci		case SF_FORMAT_PCM_U8 : return "uint8" ;
355b815c7f3Sopenharmony_ci		case SF_FORMAT_PCM_16 : return "int16" ;
356b815c7f3Sopenharmony_ci		case SF_FORMAT_PCM_24 : return "int24" ;
357b815c7f3Sopenharmony_ci		case SF_FORMAT_PCM_32 : return "int32" ;
358b815c7f3Sopenharmony_ci		case SF_FORMAT_FLOAT : return "float" ;
359b815c7f3Sopenharmony_ci		case SF_FORMAT_DOUBLE : return "double" ;
360b815c7f3Sopenharmony_ci		case SF_FORMAT_ULAW : return "ulaw" ;
361b815c7f3Sopenharmony_ci		case SF_FORMAT_ALAW : return "alaw" ;
362b815c7f3Sopenharmony_ci		case SF_FORMAT_IMA_ADPCM : return "ima_adpcm" ;
363b815c7f3Sopenharmony_ci		case SF_FORMAT_MS_ADPCM : return "ms_adpcm" ;
364b815c7f3Sopenharmony_ci		case SF_FORMAT_GSM610 : return "gsm610" ;
365b815c7f3Sopenharmony_ci		case SF_FORMAT_G721_32 : return "g721_32" ;
366b815c7f3Sopenharmony_ci		case SF_FORMAT_G723_24 : return "g723_24" ;
367b815c7f3Sopenharmony_ci		case SF_FORMAT_G723_40 : return "g723_40" ;
368b815c7f3Sopenharmony_ci		default : break ;
369b815c7f3Sopenharmony_ci		} ;
370b815c7f3Sopenharmony_ci
371b815c7f3Sopenharmony_ci	return "unknown" ;
372b815c7f3Sopenharmony_ci} /* string_of_minor_format */
373b815c7f3Sopenharmony_ci
374b815c7f3Sopenharmony_cistatic int
375b815c7f3Sopenharmony_ciformat_of_str (const std::string & fmt)
376b815c7f3Sopenharmony_ci{
377b815c7f3Sopenharmony_ci	std::vector <std::string> split ;
378b815c7f3Sopenharmony_ci
379b815c7f3Sopenharmony_ci	str_split (fmt, "-", split) ;
380b815c7f3Sopenharmony_ci
381b815c7f3Sopenharmony_ci	if (split.size () != 2)
382b815c7f3Sopenharmony_ci		return 0 ;
383b815c7f3Sopenharmony_ci
384b815c7f3Sopenharmony_ci	int major_fmt = major_format_of_hash (split.at (0)) ;
385b815c7f3Sopenharmony_ci	if (major_fmt == 0)
386b815c7f3Sopenharmony_ci		return 0 ;
387b815c7f3Sopenharmony_ci
388b815c7f3Sopenharmony_ci	int minor_fmt = minor_format_of_hash (split.at (1)) ;
389b815c7f3Sopenharmony_ci	if (minor_fmt == 0)
390b815c7f3Sopenharmony_ci		return 0 ;
391b815c7f3Sopenharmony_ci
392b815c7f3Sopenharmony_ci	return major_fmt | minor_fmt ;
393b815c7f3Sopenharmony_ci} /* format_of_str */
394b815c7f3Sopenharmony_ci
395b815c7f3Sopenharmony_cistatic void
396b815c7f3Sopenharmony_cistring_of_format (std::string & fmt, int format)
397b815c7f3Sopenharmony_ci{
398b815c7f3Sopenharmony_ci	char buffer [64] ;
399b815c7f3Sopenharmony_ci
400b815c7f3Sopenharmony_ci	snprintf (buffer, sizeof (buffer), "%s-%s", string_of_major_format (format), string_of_minor_format (format)) ;
401b815c7f3Sopenharmony_ci
402b815c7f3Sopenharmony_ci	fmt = buffer ;
403b815c7f3Sopenharmony_ci
404b815c7f3Sopenharmony_ci	return ;
405b815c7f3Sopenharmony_ci} /* string_of_format */
406