cannam@85
|
1 /*
|
cannam@85
|
2 ** Copyright (C) 1999-2011 Erik de Castro Lopo <erikd@mega-nerd.com>
|
cannam@85
|
3 **
|
cannam@85
|
4 ** All rights reserved.
|
cannam@85
|
5 **
|
cannam@85
|
6 ** Redistribution and use in source and binary forms, with or without
|
cannam@85
|
7 ** modification, are permitted provided that the following conditions are
|
cannam@85
|
8 ** met:
|
cannam@85
|
9 **
|
cannam@85
|
10 ** * Redistributions of source code must retain the above copyright
|
cannam@85
|
11 ** notice, this list of conditions and the following disclaimer.
|
cannam@85
|
12 ** * Redistributions in binary form must reproduce the above copyright
|
cannam@85
|
13 ** notice, this list of conditions and the following disclaimer in
|
cannam@85
|
14 ** the documentation and/or other materials provided with the
|
cannam@85
|
15 ** distribution.
|
cannam@85
|
16 ** * Neither the author nor the names of any contributors may be used
|
cannam@85
|
17 ** to endorse or promote products derived from this software without
|
cannam@85
|
18 ** specific prior written permission.
|
cannam@85
|
19 **
|
cannam@85
|
20 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
cannam@85
|
21 ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
cannam@85
|
22 ** TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
cannam@85
|
23 ** PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
|
cannam@85
|
24 ** CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
cannam@85
|
25 ** EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
cannam@85
|
26 ** PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
cannam@85
|
27 ** OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
cannam@85
|
28 ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
cannam@85
|
29 ** OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
cannam@85
|
30 ** ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
cannam@85
|
31 */
|
cannam@85
|
32
|
cannam@85
|
33 #include <stdio.h>
|
cannam@85
|
34 #include <stdlib.h>
|
cannam@85
|
35 #include <string.h>
|
cannam@85
|
36 #include <ctype.h>
|
cannam@85
|
37
|
cannam@85
|
38 #include <sndfile.h>
|
cannam@85
|
39
|
cannam@85
|
40 #include "common.h"
|
cannam@85
|
41
|
cannam@85
|
42 #define BUFFER_LEN (1<<16)
|
cannam@85
|
43
|
cannam@85
|
44
|
cannam@85
|
45 static void concat_data_fp (SNDFILE *wfile, SNDFILE *rofile, int channels) ;
|
cannam@85
|
46 static void concat_data_int (SNDFILE *wfile, SNDFILE *rofile, int channels) ;
|
cannam@85
|
47
|
cannam@85
|
48 static void
|
cannam@85
|
49 usage_exit (const char *progname)
|
cannam@85
|
50 {
|
cannam@85
|
51 printf ("\nUsage : %s <infile1> <infile2> ... <outfile>\n\n", progname) ;
|
cannam@85
|
52 puts (
|
cannam@85
|
53 " Create a new output file <outfile> containing the concatenated\n"
|
cannam@85
|
54 " audio data from froms <infile1> <infile2> ....\n"
|
cannam@85
|
55 "\n"
|
cannam@85
|
56 " The joined file will be encoded in the same format as the data\n"
|
cannam@85
|
57 " in infile1, with all the data in subsequent files automatically\n"
|
cannam@85
|
58 " converted to the correct encoding.\n"
|
cannam@85
|
59 "\n"
|
cannam@85
|
60 " The only restriction is that the two files must have the same\n"
|
cannam@85
|
61 " number of channels.\n"
|
cannam@85
|
62 ) ;
|
cannam@85
|
63
|
cannam@85
|
64 exit (0) ;
|
cannam@85
|
65 } /* usage_exit */
|
cannam@85
|
66
|
cannam@85
|
67 int
|
cannam@85
|
68 main (int argc, char *argv [])
|
cannam@85
|
69 { const char *progname, *outfilename ;
|
cannam@85
|
70 SNDFILE *outfile, **infiles ;
|
cannam@85
|
71 SF_INFO sfinfo_out, sfinfo_in ;
|
cannam@85
|
72 void (*func) (SNDFILE*, SNDFILE*, int) ;
|
cannam@85
|
73 int k ;
|
cannam@85
|
74
|
cannam@85
|
75 progname = program_name (argv [0]) ;
|
cannam@85
|
76
|
cannam@85
|
77 if (argc < 4)
|
cannam@85
|
78 usage_exit (progname) ;
|
cannam@85
|
79
|
cannam@85
|
80 argv ++ ;
|
cannam@85
|
81 argc -- ;
|
cannam@85
|
82
|
cannam@85
|
83 argc -- ;
|
cannam@85
|
84 outfilename = argv [argc] ;
|
cannam@85
|
85
|
cannam@85
|
86 if ((infiles = calloc (argc, sizeof (SNDFILE*))) == NULL)
|
cannam@85
|
87 { printf ("\nError : Malloc failed.\n\n") ;
|
cannam@85
|
88 exit (1) ;
|
cannam@85
|
89 } ;
|
cannam@85
|
90
|
cannam@85
|
91 memset (&sfinfo_in, 0, sizeof (sfinfo_in)) ;
|
cannam@85
|
92
|
cannam@85
|
93 if ((infiles [0] = sf_open (argv [0], SFM_READ, &sfinfo_in)) == NULL)
|
cannam@85
|
94 { printf ("\nError : failed to open file '%s'.\n\n", argv [0]) ;
|
cannam@85
|
95 exit (1) ;
|
cannam@85
|
96 } ;
|
cannam@85
|
97
|
cannam@85
|
98 sfinfo_out = sfinfo_in ;
|
cannam@85
|
99
|
cannam@85
|
100 for (k = 1 ; k < argc ; k++)
|
cannam@85
|
101 { if ((infiles [k] = sf_open (argv [k], SFM_READ, &sfinfo_in)) == NULL)
|
cannam@85
|
102 { printf ("\nError : failed to open file '%s'.\n\n", argv [k]) ;
|
cannam@85
|
103 exit (1) ;
|
cannam@85
|
104 } ;
|
cannam@85
|
105
|
cannam@85
|
106 if (sfinfo_in.channels != sfinfo_out.channels)
|
cannam@85
|
107 { printf ("\nError : File '%s' has %d channels (should have %d).\n\n", argv [k], sfinfo_in.channels, sfinfo_out.channels) ;
|
cannam@85
|
108 exit (1) ;
|
cannam@85
|
109 } ;
|
cannam@85
|
110 } ;
|
cannam@85
|
111
|
cannam@85
|
112 if ((outfile = sf_open (outfilename, SFM_WRITE, &sfinfo_out)) == NULL)
|
cannam@85
|
113 { printf ("\nError : Not able to open input file %s.\n", outfilename) ;
|
cannam@85
|
114 puts (sf_strerror (NULL)) ;
|
cannam@85
|
115 exit (1) ;
|
cannam@85
|
116 } ;
|
cannam@85
|
117
|
cannam@85
|
118 if ((sfinfo_out.format & SF_FORMAT_SUBMASK) == SF_FORMAT_DOUBLE ||
|
cannam@85
|
119 (sfinfo_out.format & SF_FORMAT_SUBMASK) == SF_FORMAT_FLOAT)
|
cannam@85
|
120 func = concat_data_fp ;
|
cannam@85
|
121 else
|
cannam@85
|
122 func = concat_data_int ;
|
cannam@85
|
123
|
cannam@85
|
124 for (k = 0 ; k < argc ; k++)
|
cannam@85
|
125 { func (outfile, infiles [k], sfinfo_out.channels) ;
|
cannam@85
|
126 sf_close (infiles [k]) ;
|
cannam@85
|
127 } ;
|
cannam@85
|
128
|
cannam@85
|
129 sf_close (outfile) ;
|
cannam@85
|
130
|
cannam@85
|
131 return 0 ;
|
cannam@85
|
132 } /* main */
|
cannam@85
|
133
|
cannam@85
|
134 static void
|
cannam@85
|
135 concat_data_fp (SNDFILE *wfile, SNDFILE *rofile, int channels)
|
cannam@85
|
136 { static double data [BUFFER_LEN] ;
|
cannam@85
|
137 int frames, readcount ;
|
cannam@85
|
138
|
cannam@85
|
139 frames = BUFFER_LEN / channels ;
|
cannam@85
|
140 readcount = frames ;
|
cannam@85
|
141
|
cannam@85
|
142 sf_seek (wfile, 0, SEEK_END) ;
|
cannam@85
|
143
|
cannam@85
|
144 while (readcount > 0)
|
cannam@85
|
145 { readcount = sf_readf_double (rofile, data, frames) ;
|
cannam@85
|
146 sf_writef_double (wfile, data, readcount) ;
|
cannam@85
|
147 } ;
|
cannam@85
|
148
|
cannam@85
|
149 return ;
|
cannam@85
|
150 } /* concat_data_fp */
|
cannam@85
|
151
|
cannam@85
|
152 static void
|
cannam@85
|
153 concat_data_int (SNDFILE *wfile, SNDFILE *rofile, int channels)
|
cannam@85
|
154 { static int data [BUFFER_LEN] ;
|
cannam@85
|
155 int frames, readcount ;
|
cannam@85
|
156
|
cannam@85
|
157 frames = BUFFER_LEN / channels ;
|
cannam@85
|
158 readcount = frames ;
|
cannam@85
|
159
|
cannam@85
|
160 sf_seek (wfile, 0, SEEK_END) ;
|
cannam@85
|
161
|
cannam@85
|
162 while (readcount > 0)
|
cannam@85
|
163 { readcount = sf_readf_int (rofile, data, frames) ;
|
cannam@85
|
164 sf_writef_int (wfile, data, readcount) ;
|
cannam@85
|
165 } ;
|
cannam@85
|
166
|
cannam@85
|
167 return ;
|
cannam@85
|
168 } /* concat_data_int */
|
cannam@85
|
169
|