annotate src/libsndfile-1.0.27/programs/sndfile-convert.c @ 127:7867fa7e1b6b

Current fftw source
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 18 Oct 2016 13:40:26 +0100
parents cd6cdf86811e
children
rev   line source
cannam@125 1 /*
cannam@125 2 ** Copyright (C) 1999-2014 Erik de Castro Lopo <erikd@mega-nerd.com>
cannam@125 3 **
cannam@125 4 ** All rights reserved.
cannam@125 5 **
cannam@125 6 ** Redistribution and use in source and binary forms, with or without
cannam@125 7 ** modification, are permitted provided that the following conditions are
cannam@125 8 ** met:
cannam@125 9 **
cannam@125 10 ** * Redistributions of source code must retain the above copyright
cannam@125 11 ** notice, this list of conditions and the following disclaimer.
cannam@125 12 ** * Redistributions in binary form must reproduce the above copyright
cannam@125 13 ** notice, this list of conditions and the following disclaimer in
cannam@125 14 ** the documentation and/or other materials provided with the
cannam@125 15 ** distribution.
cannam@125 16 ** * Neither the author nor the names of any contributors may be used
cannam@125 17 ** to endorse or promote products derived from this software without
cannam@125 18 ** specific prior written permission.
cannam@125 19 **
cannam@125 20 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
cannam@125 21 ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
cannam@125 22 ** TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
cannam@125 23 ** PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
cannam@125 24 ** CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
cannam@125 25 ** EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
cannam@125 26 ** PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
cannam@125 27 ** OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
cannam@125 28 ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
cannam@125 29 ** OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
cannam@125 30 ** ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
cannam@125 31 */
cannam@125 32
cannam@125 33 #include <stdio.h>
cannam@125 34 #include <stdlib.h>
cannam@125 35 #include <string.h>
cannam@125 36 #include <ctype.h>
cannam@125 37
cannam@125 38 #include <sndfile.h>
cannam@125 39
cannam@125 40 #include "common.h"
cannam@125 41
cannam@125 42
cannam@125 43 typedef struct
cannam@125 44 { char *infilename, *outfilename ;
cannam@125 45 SF_INFO infileinfo, outfileinfo ;
cannam@125 46 } OptionData ;
cannam@125 47
cannam@125 48 static void copy_metadata (SNDFILE *outfile, SNDFILE *infile, int channels) ;
cannam@125 49
cannam@125 50 static void
cannam@125 51 usage_exit (const char *progname)
cannam@125 52 {
cannam@125 53 printf ("\nUsage : %s [options] [encoding] <input file> <output file>\n", progname) ;
cannam@125 54 puts ("\n"
cannam@125 55 " where [option] may be:\n\n"
cannam@125 56 " -override-sample-rate=X : force sample rate of input to X\n"
cannam@125 57 " -endian=little : force output file to little endian data\n"
cannam@125 58 " -endian=big : force output file to big endian data\n"
cannam@125 59 " -endian=cpu : force output file same endian-ness as the CPU\n"
cannam@125 60 " -normalize : normalize the data in the output file\n"
cannam@125 61 ) ;
cannam@125 62
cannam@125 63 puts (
cannam@125 64 " where [encoding] may be one of the following:\n\n"
cannam@125 65 " -pcms8 : signed 8 bit pcm\n"
cannam@125 66 " -pcmu8 : unsigned 8 bit pcm\n"
cannam@125 67 " -pcm16 : 16 bit pcm\n"
cannam@125 68 " -pcm24 : 24 bit pcm\n"
cannam@125 69 " -pcm32 : 32 bit pcm\n"
cannam@125 70 " -float32 : 32 bit floating point\n"
cannam@125 71 " -ulaw : ULAW\n"
cannam@125 72 " -alaw : ALAW\n"
cannam@125 73 " -alac16 : 16 bit ALAC (CAF only)\n"
cannam@125 74 " -alac20 : 20 bit ALAC (CAF only)\n"
cannam@125 75 " -alac24 : 24 bit ALAC (CAF only)\n"
cannam@125 76 " -alac32 : 32 bit ALAC (CAF only)\n"
cannam@125 77 " -ima-adpcm : IMA ADPCM (WAV only)\n"
cannam@125 78 " -ms-adpcm : MS ADPCM (WAV only)\n"
cannam@125 79 " -gsm610 : GSM6.10 (WAV only)\n"
cannam@125 80 " -dwvw12 : 12 bit DWVW (AIFF only)\n"
cannam@125 81 " -dwvw16 : 16 bit DWVW (AIFF only)\n"
cannam@125 82 " -dwvw24 : 24 bit DWVW (AIFF only)\n"
cannam@125 83 " -vorbis : Vorbis (OGG only)\n"
cannam@125 84 ) ;
cannam@125 85
cannam@125 86 puts (
cannam@125 87 " If no encoding is specified, the program will try to use the encoding\n"
cannam@125 88 " of the input file in the output file. This will not always work as\n"
cannam@125 89 " most container formats (eg WAV, AIFF etc) only support a small subset\n"
cannam@125 90 " of codec formats (eg 16 bit PCM, a-law, Vorbis etc).\n"
cannam@125 91 ) ;
cannam@125 92
cannam@125 93 puts (
cannam@125 94 " The format of the output file is determined by the file extension of the\n"
cannam@125 95 " output file name. The following extensions are currently understood:\n"
cannam@125 96 ) ;
cannam@125 97
cannam@125 98 sfe_dump_format_map () ;
cannam@125 99
cannam@125 100 puts ("") ;
cannam@125 101 exit (1) ;
cannam@125 102 } /* usage_exit */
cannam@125 103
cannam@125 104 static void
cannam@125 105 report_format_error_exit (const char * argv0, SF_INFO * sfinfo)
cannam@125 106 { int old_format = sfinfo->format ;
cannam@125 107 int endian = sfinfo->format & SF_FORMAT_ENDMASK ;
cannam@125 108
cannam@125 109 sfinfo->format = old_format & (SF_FORMAT_TYPEMASK | SF_FORMAT_SUBMASK) ;
cannam@125 110
cannam@125 111 if (endian && sf_format_check (sfinfo))
cannam@125 112 { printf ("Error : output file format does not support %s endian-ness.\n", sfe_endian_name (endian)) ;
cannam@125 113 exit (1) ;
cannam@125 114 } ;
cannam@125 115
cannam@125 116 printf ("\n"
cannam@125 117 "Error : output file format is invalid.\n"
cannam@125 118 "The '%s' container does not support '%s' codec data.\n"
cannam@125 119 "Run '%s --help' for clues.\n\n",
cannam@125 120 sfe_container_name (sfinfo->format), sfe_codec_name (sfinfo->format), program_name (argv0)) ;
cannam@125 121 exit (1) ;
cannam@125 122 } /* report_format_error_exit */
cannam@125 123
cannam@125 124 int
cannam@125 125 main (int argc, char * argv [])
cannam@125 126 { const char *progname, *infilename, *outfilename ;
cannam@125 127 SNDFILE *infile = NULL, *outfile = NULL ;
cannam@125 128 SF_INFO sfinfo ;
cannam@125 129 int k, outfilemajor, outfileminor = 0, infileminor ;
cannam@125 130 int override_sample_rate = 0 ; /* assume no sample rate override. */
cannam@125 131 int endian = SF_ENDIAN_FILE, normalize = SF_FALSE ;
cannam@125 132
cannam@125 133 progname = program_name (argv [0]) ;
cannam@125 134
cannam@125 135 if (argc < 3 || argc > 5)
cannam@125 136 usage_exit (progname) ;
cannam@125 137
cannam@125 138 infilename = argv [argc-2] ;
cannam@125 139 outfilename = argv [argc-1] ;
cannam@125 140
cannam@125 141 if (strcmp (infilename, outfilename) == 0)
cannam@125 142 { printf ("Error : Input and output filenames are the same.\n\n") ;
cannam@125 143 usage_exit (progname) ;
cannam@125 144 } ;
cannam@125 145
cannam@125 146 if (strlen (infilename) > 1 && infilename [0] == '-')
cannam@125 147 { printf ("Error : Input filename (%s) looks like an option.\n\n", infilename) ;
cannam@125 148 usage_exit (progname) ;
cannam@125 149 } ;
cannam@125 150
cannam@125 151 if (outfilename [0] == '-')
cannam@125 152 { printf ("Error : Output filename (%s) looks like an option.\n\n", outfilename) ;
cannam@125 153 usage_exit (progname) ;
cannam@125 154 } ;
cannam@125 155
cannam@125 156 for (k = 1 ; k < argc - 2 ; k++)
cannam@125 157 { if (! strcmp (argv [k], "-pcms8"))
cannam@125 158 { outfileminor = SF_FORMAT_PCM_S8 ;
cannam@125 159 continue ;
cannam@125 160 } ;
cannam@125 161 if (! strcmp (argv [k], "-pcmu8"))
cannam@125 162 { outfileminor = SF_FORMAT_PCM_U8 ;
cannam@125 163 continue ;
cannam@125 164 } ;
cannam@125 165 if (! strcmp (argv [k], "-pcm16"))
cannam@125 166 { outfileminor = SF_FORMAT_PCM_16 ;
cannam@125 167 continue ;
cannam@125 168 } ;
cannam@125 169 if (! strcmp (argv [k], "-pcm24"))
cannam@125 170 { outfileminor = SF_FORMAT_PCM_24 ;
cannam@125 171 continue ;
cannam@125 172 } ;
cannam@125 173 if (! strcmp (argv [k], "-pcm32"))
cannam@125 174 { outfileminor = SF_FORMAT_PCM_32 ;
cannam@125 175 continue ;
cannam@125 176 } ;
cannam@125 177 if (! strcmp (argv [k], "-float32"))
cannam@125 178 { outfileminor = SF_FORMAT_FLOAT ;
cannam@125 179 continue ;
cannam@125 180 } ;
cannam@125 181 if (! strcmp (argv [k], "-ulaw"))
cannam@125 182 { outfileminor = SF_FORMAT_ULAW ;
cannam@125 183 continue ;
cannam@125 184 } ;
cannam@125 185 if (! strcmp (argv [k], "-alaw"))
cannam@125 186 { outfileminor = SF_FORMAT_ALAW ;
cannam@125 187 continue ;
cannam@125 188 } ;
cannam@125 189 if (! strcmp (argv [k], "-alac16"))
cannam@125 190 { outfileminor = SF_FORMAT_ALAC_16 ;
cannam@125 191 continue ;
cannam@125 192 } ;
cannam@125 193 if (! strcmp (argv [k], "-alac20"))
cannam@125 194 { outfileminor = SF_FORMAT_ALAC_20 ;
cannam@125 195 continue ;
cannam@125 196 } ;
cannam@125 197 if (! strcmp (argv [k], "-alac24"))
cannam@125 198 { outfileminor = SF_FORMAT_ALAC_24 ;
cannam@125 199 continue ;
cannam@125 200 } ;
cannam@125 201 if (! strcmp (argv [k], "-alac32"))
cannam@125 202 { outfileminor = SF_FORMAT_ALAC_32 ;
cannam@125 203 continue ;
cannam@125 204 } ;
cannam@125 205 if (! strcmp (argv [k], "-ima-adpcm"))
cannam@125 206 { outfileminor = SF_FORMAT_IMA_ADPCM ;
cannam@125 207 continue ;
cannam@125 208 } ;
cannam@125 209 if (! strcmp (argv [k], "-ms-adpcm"))
cannam@125 210 { outfileminor = SF_FORMAT_MS_ADPCM ;
cannam@125 211 continue ;
cannam@125 212 } ;
cannam@125 213 if (! strcmp (argv [k], "-gsm610"))
cannam@125 214 { outfileminor = SF_FORMAT_GSM610 ;
cannam@125 215 continue ;
cannam@125 216 } ;
cannam@125 217 if (! strcmp (argv [k], "-dwvw12"))
cannam@125 218 { outfileminor = SF_FORMAT_DWVW_12 ;
cannam@125 219 continue ;
cannam@125 220 } ;
cannam@125 221 if (! strcmp (argv [k], "-dwvw16"))
cannam@125 222 { outfileminor = SF_FORMAT_DWVW_16 ;
cannam@125 223 continue ;
cannam@125 224 } ;
cannam@125 225 if (! strcmp (argv [k], "-dwvw24"))
cannam@125 226 { outfileminor = SF_FORMAT_DWVW_24 ;
cannam@125 227 continue ;
cannam@125 228 } ;
cannam@125 229 if (! strcmp (argv [k], "-vorbis"))
cannam@125 230 { outfileminor = SF_FORMAT_VORBIS ;
cannam@125 231 continue ;
cannam@125 232 } ;
cannam@125 233
cannam@125 234 if (strstr (argv [k], "-override-sample-rate=") == argv [k])
cannam@125 235 { const char *ptr ;
cannam@125 236
cannam@125 237 ptr = argv [k] + strlen ("-override-sample-rate=") ;
cannam@125 238 override_sample_rate = atoi (ptr) ;
cannam@125 239 continue ;
cannam@125 240 } ;
cannam@125 241
cannam@125 242 if (! strcmp (argv [k], "-endian=little"))
cannam@125 243 { endian = SF_ENDIAN_LITTLE ;
cannam@125 244 continue ;
cannam@125 245 } ;
cannam@125 246
cannam@125 247 if (! strcmp (argv [k], "-endian=big"))
cannam@125 248 { endian = SF_ENDIAN_BIG ;
cannam@125 249 continue ;
cannam@125 250 } ;
cannam@125 251
cannam@125 252 if (! strcmp (argv [k], "-endian=cpu"))
cannam@125 253 { endian = SF_ENDIAN_CPU ;
cannam@125 254 continue ;
cannam@125 255 } ;
cannam@125 256
cannam@125 257 if (! strcmp (argv [k], "-endian=file"))
cannam@125 258 { endian = SF_ENDIAN_FILE ;
cannam@125 259 continue ;
cannam@125 260 } ;
cannam@125 261
cannam@125 262 if (! strcmp (argv [k], "-normalize"))
cannam@125 263 { normalize = SF_TRUE ;
cannam@125 264 continue ;
cannam@125 265 } ;
cannam@125 266
cannam@125 267 printf ("Error : Not able to decode argunment '%s'.\n", argv [k]) ;
cannam@125 268 exit (1) ;
cannam@125 269 } ;
cannam@125 270
cannam@125 271 memset (&sfinfo, 0, sizeof (sfinfo)) ;
cannam@125 272
cannam@125 273 if ((infile = sf_open (infilename, SFM_READ, &sfinfo)) == NULL)
cannam@125 274 { printf ("Not able to open input file %s.\n", infilename) ;
cannam@125 275 puts (sf_strerror (NULL)) ;
cannam@125 276 return 1 ;
cannam@125 277 } ;
cannam@125 278
cannam@125 279 /* Update sample rate if forced to something else. */
cannam@125 280 if (override_sample_rate)
cannam@125 281 sfinfo.samplerate = override_sample_rate ;
cannam@125 282
cannam@125 283 infileminor = sfinfo.format & SF_FORMAT_SUBMASK ;
cannam@125 284
cannam@125 285 if ((sfinfo.format = sfe_file_type_of_ext (outfilename, sfinfo.format)) == 0)
cannam@125 286 { printf ("Error : Not able to determine output file type for %s.\n", outfilename) ;
cannam@125 287 return 1 ;
cannam@125 288 } ;
cannam@125 289
cannam@125 290 outfilemajor = sfinfo.format & (SF_FORMAT_TYPEMASK | SF_FORMAT_ENDMASK) ;
cannam@125 291
cannam@125 292 if (outfileminor == 0)
cannam@125 293 outfileminor = sfinfo.format & SF_FORMAT_SUBMASK ;
cannam@125 294
cannam@125 295 if (outfileminor != 0)
cannam@125 296 sfinfo.format = outfilemajor | outfileminor ;
cannam@125 297 else
cannam@125 298 sfinfo.format = outfilemajor | (sfinfo.format & SF_FORMAT_SUBMASK) ;
cannam@125 299
cannam@125 300 sfinfo.format |= endian ;
cannam@125 301
cannam@125 302 if ((sfinfo.format & SF_FORMAT_TYPEMASK) == SF_FORMAT_XI)
cannam@125 303 switch (sfinfo.format & SF_FORMAT_SUBMASK)
cannam@125 304 { case SF_FORMAT_PCM_16 :
cannam@125 305 sfinfo.format = outfilemajor | SF_FORMAT_DPCM_16 ;
cannam@125 306 break ;
cannam@125 307
cannam@125 308 case SF_FORMAT_PCM_S8 :
cannam@125 309 case SF_FORMAT_PCM_U8 :
cannam@125 310 sfinfo.format = outfilemajor | SF_FORMAT_DPCM_8 ;
cannam@125 311 break ;
cannam@125 312 } ;
cannam@125 313
cannam@125 314 if (sf_format_check (&sfinfo) == 0)
cannam@125 315 report_format_error_exit (argv [0], &sfinfo) ;
cannam@125 316
cannam@125 317 if ((sfinfo.format & SF_FORMAT_SUBMASK) == SF_FORMAT_GSM610 && sfinfo.samplerate != 8000)
cannam@125 318 { printf (
cannam@125 319 "WARNING: GSM 6.10 data format only supports 8kHz sample rate. The converted\n"
cannam@125 320 "ouput file will contain the input data converted to the GSM 6.10 data format\n"
cannam@125 321 "but not re-sampled.\n"
cannam@125 322 ) ;
cannam@125 323 } ;
cannam@125 324
cannam@125 325 /* Open the output file. */
cannam@125 326 if ((outfile = sf_open (outfilename, SFM_WRITE, &sfinfo)) == NULL)
cannam@125 327 { printf ("Not able to open output file %s : %s\n", outfilename, sf_strerror (NULL)) ;
cannam@125 328 return 1 ;
cannam@125 329 } ;
cannam@125 330
cannam@125 331 /* Copy the metadata */
cannam@125 332 copy_metadata (outfile, infile, sfinfo.channels) ;
cannam@125 333
cannam@125 334 if (normalize
cannam@125 335 || (outfileminor == SF_FORMAT_DOUBLE) || (outfileminor == SF_FORMAT_FLOAT)
cannam@125 336 || (infileminor == SF_FORMAT_DOUBLE) || (infileminor == SF_FORMAT_FLOAT)
cannam@125 337 || (infileminor == SF_FORMAT_VORBIS) || (outfileminor == SF_FORMAT_VORBIS))
cannam@125 338 sfe_copy_data_fp (outfile, infile, sfinfo.channels, normalize) ;
cannam@125 339 else
cannam@125 340 sfe_copy_data_int (outfile, infile, sfinfo.channels) ;
cannam@125 341
cannam@125 342 sf_close (infile) ;
cannam@125 343 sf_close (outfile) ;
cannam@125 344
cannam@125 345 return 0 ;
cannam@125 346 } /* main */
cannam@125 347
cannam@125 348 static void
cannam@125 349 copy_metadata (SNDFILE *outfile, SNDFILE *infile, int channels)
cannam@125 350 { SF_INSTRUMENT inst ;
cannam@125 351 SF_CUES cues ;
cannam@125 352 SF_BROADCAST_INFO_2K binfo ;
cannam@125 353 const char *str ;
cannam@125 354 int k, chanmap [256] ;
cannam@125 355
cannam@125 356 for (k = SF_STR_FIRST ; k <= SF_STR_LAST ; k++)
cannam@125 357 { str = sf_get_string (infile, k) ;
cannam@125 358 if (str != NULL)
cannam@125 359 sf_set_string (outfile, k, str) ;
cannam@125 360 } ;
cannam@125 361
cannam@125 362 memset (&inst, 0, sizeof (inst)) ;
cannam@125 363 memset (&cues, 0, sizeof (cues)) ;
cannam@125 364 memset (&binfo, 0, sizeof (binfo)) ;
cannam@125 365
cannam@125 366 if (channels < ARRAY_LEN (chanmap))
cannam@125 367 { size_t size = channels * sizeof (chanmap [0]) ;
cannam@125 368
cannam@125 369 if (sf_command (infile, SFC_GET_CHANNEL_MAP_INFO, chanmap, size) == SF_TRUE)
cannam@125 370 sf_command (outfile, SFC_SET_CHANNEL_MAP_INFO, chanmap, size) ;
cannam@125 371 } ;
cannam@125 372
cannam@125 373 if (sf_command (infile, SFC_GET_CUE, &cues, sizeof (cues)) == SF_TRUE)
cannam@125 374 sf_command (outfile, SFC_SET_CUE, &cues, sizeof (cues)) ;
cannam@125 375
cannam@125 376 if (sf_command (infile, SFC_GET_INSTRUMENT, &inst, sizeof (inst)) == SF_TRUE)
cannam@125 377 sf_command (outfile, SFC_SET_INSTRUMENT, &inst, sizeof (inst)) ;
cannam@125 378
cannam@125 379 if (sf_command (infile, SFC_GET_BROADCAST_INFO, &binfo, sizeof (binfo)) == SF_TRUE)
cannam@125 380 sf_command (outfile, SFC_SET_BROADCAST_INFO, &binfo, sizeof (binfo)) ;
cannam@125 381
cannam@125 382 } /* copy_metadata */
cannam@125 383