annotate src/linuxplay.c @ 0:376386a54a3c r19-14

Import from CVS: tag r19-14
author cvs
date Mon, 13 Aug 2007 08:45:50 +0200
parents
children 8fc7fe29b841
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1 /* linuxplay.c - play a sound file on the speaker
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
2 **
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
3 ** Copyright (C) 1995,96 by Markus Gutschke (gutschk@math.uni-muenster.de)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
4 ** This is version 1.3 of linuxplay.c
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
5 **
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
6 ** Parts of this code were inspired by sunplay.c, which is copyright 1989 by
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
7 ** Jef Poskanzer and 1991,92 by Jamie Zawinski; c.f. sunplay.c for further
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
8 ** information.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
9 **
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
10 ** Permission to use, copy, modify, and distribute this software and its
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
11 ** documentation for any purpose and without fee is hereby granted, provided
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
12 ** that the above copyright notice appear in all copies and that both that
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
13 ** copyright notice and this permission notice appear in supporting
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
14 ** documentation. This software is provided "as is" without express or
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
15 ** implied warranty.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
16 **
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
17 ** Changelog:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
18 ** 1.0 -- first release; supports SunAudio, Wave and RAW file formats
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
19 ** detects (and rejects) VOC file format
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
20 ** tested with PC-Speaker driver only
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
21 ** 1.1 -- fixed bug with playback of stereo Wave files
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
22 ** fixed VOC file detection
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
23 ** fixed mono/8bit conversion
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
24 ** cleaned up mixer programming (c.f. VoxWare-SDK)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
25 ** tested with PC-Speaker driver and with PAS16 soundcard
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
26 ** 1.2 -- first (incompatible) attempt at fixing reliable signal handling
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
27 ** 1.3 -- changed signal handling to use reliable signals; this is done
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
28 ** by including "syssignal.h"; it fixes nasty program crashes
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
29 ** when using native sound in TTY mode.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
30 ** added support for DEC audio file format (this is basically the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
31 ** same as Sun audio, but uses little endian format, instead).
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
32 ** strip the header from Sun audio and DEC audio files in order to
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
33 ** prevent noise at beginning of samples (thanks to Thomas Pundt
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
34 ** <pundtt@math.uni-muenster.de> for pointing out this bug and
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
35 ** providing information on the file format).
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
36 ** added a few more conversion routines.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
37 ** made the code even more tolerant to the limits imposed by some
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
38 ** soundcards and try to accept soundfiles even if they are not
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
39 ** fully conformant to the standard.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
40 ** 1.4 -- increased header size to 256; I hope there is no sample software
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
41 ** that requires this much.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
42 ** added code for converting from signed to unsigned format as
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
43 ** some soundcards cannot handle signed 8bit data.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
44 */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
45
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
46 /* Synched up with: Not in FSF. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
47
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
48 #define HEADERSZ 256 /* has to be at least as big as the biggest header */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
49 #define SNDBUFSZ 2048 /* has to be at least as big as HEADERSZ */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
50
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
51 /* XEmacs beta testers say: undef this by default. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
52 #undef NOVOLUMECTRLFORMULAW /* Changing the volume for uLaw-encoded
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
53 samples sounds very poor; possibly,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
54 this is true only for the PC-Snd
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
55 driver, so undefine this symbol at your
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
56 discretion */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
57
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
58 #ifdef HAVE_CONFIG_H
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
59 #include <config.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
60 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
61
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
62 #include <errno.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
63 #include <fcntl.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
64 #include <linux/soundcard.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
65 #include <stdio.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
66 #include <stdlib.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
67 #include <string.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
68 #include <sys/fcntl.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
69 #include <sys/file.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
70 #include <sys/ioctl.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
71 #include <sys/signal.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
72 #include <unistd.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
73
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
74 #ifdef LINUXPLAYSTANDALONE
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
75 #define perror(str) fprintf(stderr,"audio: %s %s\n",str,strerror(errno));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
76 #define warn(str) fprintf(stderr,"audio: %s\n",str);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
77 #else
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
78 #include "lisp.h"
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
79 #include "syssignal.h"
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
80 #define perror(str) message("audio: %s, %s ",str,strerror(errno))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
81 #define warn(str) message("audio: %s ",GETTEXT(str))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
82 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
83
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
84 #ifdef __GNUC__
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
85 #define UNUSED(x) ((void)(x))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
86 #else
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
87 #define UNUSED(x)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
88 #define __inline__
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
89 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
90
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
91 static __sighandler_t sighup_handler;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
92 static __sighandler_t sigint_handler;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
93
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
94 /* Maintain global variable for keeping parser state information; this struct
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
95 is set to zero before the first invocation of the parser. The use of a
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
96 global variable prevents multiple concurrent executions of this code, but
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
97 this does not happen anyways... */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
98 static union {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
99 struct {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
100 int align;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
101 enum {wvMain,wvSubchunk,wvOutOfBlock,wvSkipChunk,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
102 wvSoundChunk,wvFatal,wvFatalNotify} state;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
103 size_t left;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
104 unsigned char leftover[HEADERSZ];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
105 signed long chunklength;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
106 } wave;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
107 struct {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
108 int align;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
109 int isdata;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
110 int skipping;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
111 size_t left;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
112 unsigned char leftover[HEADERSZ];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
113 } audio;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
114 } parsestate;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
115
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
116 /* Use a global buffer as scratch-pad for possible conversions of the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
117 sampling format */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
118 static unsigned char sndbuf[SNDBUFSZ];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
119
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
120 static int mix_fd = -1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
121 static int audio_vol = -1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
122 static int audio_fd = -1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
123 static char *audio_dev = "";
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
124
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
125 typedef enum {fmtIllegal,fmtRaw,fmtVoc,fmtWave,fmtSunAudio} fmtType;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
126
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
127 /* Intercept SIGINT and SIGHUP in order to close the audio and mixer
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
128 devices before terminating sound output; this requires reliable
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
129 signals as provided by "syssignal.h" */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
130 static void sighandler(int sig)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
131 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
132 if (mix_fd > 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
133 if (audio_vol >= 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
134 ioctl(mix_fd,SOUND_MIXER_WRITE_PCM,&audio_vol);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
135 audio_vol = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
136 if (mix_fd != audio_fd)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
137 close(mix_fd);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
138 mix_fd = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
139 if (audio_fd > 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
140 ioctl(audio_fd,SNDCTL_DSP_SYNC,NULL);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
141 ioctl(audio_fd,SNDCTL_DSP_RESET,NULL);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
142 close(audio_fd);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
143 audio_fd = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
144 if (sig == SIGHUP && sighup_handler) sighup_handler(sig);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
145 else if (sig == SIGINT && sigint_handler) sigint_handler(sig);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
146 else exit(1);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
147 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
148
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
149 /* There is no special treatment required for parsing raw data files; we
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
150 assume that these files contain data in 8bit unsigned format that
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
151 has been sampled at 8kHz; there is no extra header */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
152 static size_t parseraw(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
153 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
154 int rc = *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
155
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
156 *outbuf = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
157 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
158 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
159 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
160
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
161 /* Currently we cannot cope with files in VOC format; if you really need
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
162 to play these files, they should be converted by using SOX */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
163 static size_t parsevoc(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
164 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
165 UNUSED(data);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
166 UNUSED(sz);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
167 UNUSED(outbuf);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
168 return(0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
169 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
170
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
171 /* We need to perform some look-ahead in order to parse files in WAVE format;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
172 this might require re-partioning of the data segments if headers cross the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
173 boundaries between two read operations. This is done in a two-step way:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
174 first we request a certain amount of bytes... */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
175 static __inline__ int waverequire(void **data,size_t *sz,size_t rq)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
176 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
177 int rc = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
178
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
179 if (rq > HEADERSZ) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
180 warn("Header size exceeded while parsing WAVE file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
181 parsestate.wave.state = wvFatal;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
182 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
183 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
184 if ((rq -= parsestate.wave.left) <= 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
185 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
186 if (rq > *sz) {rq = *sz; rc = 0;}
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
187 memcpy(parsestate.wave.leftover+parsestate.wave.left,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
188 *data,rq);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
189 parsestate.wave.left += rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
190 ((unsigned char *)*data) += rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
191 *sz -= rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
192 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
193 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
194
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
195 /* ...and next we remove this many bytes from the buffer */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
196 static __inline__ void waveremove(size_t rq)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
197 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
198 if (parsestate.wave.left <= rq)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
199 parsestate.wave.left = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
200 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
201 parsestate.wave.left -= rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
202 memmove(parsestate.wave.leftover,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
203 parsestate.wave.leftover+rq,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
204 parsestate.wave.left); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
205 return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
206 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
207
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
208 /* Sound files in WAVE format can contain an arbitrary amount of tagged
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
209 chunks; this requires quite some effort for parsing the data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
210 static size_t parsewave(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
211 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
212 for (;;)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
213 switch (parsestate.wave.state) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
214 case wvMain:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
215 if (!waverequire(data,sz,20))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
216 return(0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
217 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
218 parsestate.wave.chunklength = parsestate.wave.leftover[16] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
219 256*(parsestate.wave.leftover[17] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
220 256*(parsestate.wave.leftover[18] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
221 256*parsestate.wave.leftover[19]));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
222 waveremove(20);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
223 parsestate.wave.state = wvSubchunk;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
224 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
225 case wvSubchunk:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
226 if (!waverequire(data,sz,parsestate.wave.chunklength))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
227 return(0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
228 parsestate.wave.align = parsestate.wave.chunklength < 14 ? 1
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
229 : parsestate.wave.leftover[12];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
230 if (parsestate.wave.align != 1 &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
231 parsestate.wave.align != 2 &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
232 parsestate.wave.align != 4) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
233 warn("Illegal datawidth detected while parsing WAVE file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
234 parsestate.wave.state = wvFatal; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
235 else
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
236 parsestate.wave.state = wvOutOfBlock;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
237 waveremove(parsestate.wave.chunklength);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
238 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
239 case wvOutOfBlock:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
240 if (!waverequire(data,sz,8))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
241 return(0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
242 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
243 parsestate.wave.chunklength = parsestate.wave.leftover[4] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
244 256*(parsestate.wave.leftover[5] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
245 256*(parsestate.wave.leftover[6] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
246 256*(parsestate.wave.leftover[7] & 0x7F)));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
247 if (memcmp(parsestate.wave.leftover,"data",4))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
248 parsestate.wave.state = wvSkipChunk;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
249 else
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
250 parsestate.wave.state = wvSoundChunk;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
251 waveremove(8);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
252 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
253 case wvSkipChunk:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
254 if (parsestate.wave.chunklength > 0 && *sz > 0 &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
255 (signed long)*sz < (signed long)parsestate.wave.chunklength) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
256 parsestate.wave.chunklength -= *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
257 *sz = 0; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
258 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
259 if (parsestate.wave.chunklength > 0 && *sz > 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
260 *sz -= parsestate.wave.chunklength;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
261 ((unsigned char *)*data) += parsestate.wave.chunklength; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
262 parsestate.wave.state = wvOutOfBlock; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
263 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
264 case wvSoundChunk: {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
265 size_t count,rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
266 if (parsestate.wave.left) { /* handle leftover bytes from last
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
267 alignment operation */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
268 count = parsestate.wave.left;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
269 rq = HEADERSZ-count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
270 if (rq > parsestate.wave.chunklength)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
271 rq = parsestate.wave.chunklength;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
272 if (!waverequire(data,sz,rq)) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
273 parsestate.wave.chunklength -= parsestate.wave.left - count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
274 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
275 parsestate.wave.chunklength -= rq;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
276 *outbuf = parsestate.wave.leftover;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
277 parsestate.wave.left = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
278 return(rq); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
279 if (*sz >= parsestate.wave.chunklength) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
280 count = parsestate.wave.chunklength;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
281 rq = 0; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
282 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
283 count = *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
284 count -= rq = count % parsestate.wave.align; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
285 *outbuf = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
286 ((unsigned char *)*data) += count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
287 *sz -= count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
288 if ((parsestate.wave.chunklength -= count) < parsestate.wave.align) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
289 parsestate.wave.state = wvOutOfBlock;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
290 /* Some broken software (e.g. SOX) attaches junk to the end of a sound
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
291 chunk; so, let's ignore this... */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
292 if (parsestate.wave.chunklength)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
293 parsestate.wave.state = wvSkipChunk; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
294 else if (rq)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
295 /* align data length to a multiple of datasize; keep additional data
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
296 in "leftover" buffer --- this is neccessary to ensure proper
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
297 functioning of the sndcnv... routines */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
298 waverequire(data,sz,rq);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
299 return(count); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
300 case wvFatalNotify:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
301 warn("Irrecoverable error while parsing WAVE file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
302 parsestate.wave.state = wvFatal;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
303 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
304 case wvFatal:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
305 default:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
306 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
307 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
308 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
309
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
310 /* Strip the header from files in Sun/DEC audio format; this requires some
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
311 extra processing as the header can be an arbitrary size and it might
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
312 result in alignment errors for subsequent conversions --- thus we do
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
313 some buffering, where needed */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
314 static size_t parsesundecaudio(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
315 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
316 /* There is data left over from the last invocation of this function; join
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
317 it with the new data and return a sound chunk that is as big as a
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
318 single entry */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
319 if (parsestate.audio.left) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
320 if (parsestate.audio.left + *sz > parsestate.audio.align) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
321 int count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
322 memmove(parsestate.audio.leftover + parsestate.audio.left,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
323 *data,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
324 count = parsestate.audio.align - parsestate.audio.left);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
325 *outbuf = parsestate.audio.leftover;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
326 *sz -= count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
327 *data = (*(char **)data) + count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
328 parsestate.audio.left = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
329 return(parsestate.audio.align); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
330 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
331 /* We need even more data in order to get one complete single entry! */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
332 memmove(parsestate.audio.leftover + parsestate.audio.left,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
333 *data,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
334 *sz);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
335 *data = (*(char **)data) + *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
336 parsestate.audio.left += *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
337 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
338 return(0); } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
339
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
340 /* This is the main sound chunk, strip of any extra data that does not fit
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
341 the alignment requirements and move these bytes into the leftover buffer*/
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
342 if (parsestate.audio.isdata) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
343 int rc = *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
344 *outbuf = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
345 if ((parsestate.audio.left = rc % parsestate.audio.align) != 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
346 memmove(parsestate.audio.leftover,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
347 (char *)*outbuf + rc - parsestate.audio.left,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
348 parsestate.audio.left);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
349 rc -= parsestate.audio.left; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
350 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
351 return(rc); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
352
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
353 /* This is the first invocation of this function; we need to parse the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
354 header information and determine how many bytes we need to skip until
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
355 the start of the sound chunk */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
356 if (!parsestate.audio.skipping) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
357 unsigned char *header = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
358 if (*sz < 8) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
359 warn("Irrecoverable error while parsing Sun/DEC audio file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
360 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
361 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
362 if (header[3]) { /* Sun audio (big endian) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
363 parsestate.audio.align = ((header[15] > 2)+1)*header[23];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
364 parsestate.audio.skipping = header[7]+256*(header[6]+256*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
365 (header[5]+256*header[4])); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
366 else { /* DEC audio (little endian) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
367 parsestate.audio.align = ((header[12] > 2)+1)*header[20];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
368 parsestate.audio.skipping = header[4]+256*(header[5]+256*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
369 (header[6]+256*header[7])); }}
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
370
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
371 /* We are skipping extra data that has been attached to header; most usually
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
372 this will be just a comment, such as the original filename and/or the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
373 creation date. Make sure that we do not return less than one single sound
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
374 sample entry to the caller; if this happens, rather decide to move those
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
375 few bytes into the leftover buffer and deal with it later */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
376 if (*sz >= parsestate.audio.skipping) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
377 /* Skip just the header information and return the sound chunk */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
378 int rc = *sz - parsestate.audio.skipping;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
379 *outbuf = (char *)*data + parsestate.audio.skipping;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
380 if ((parsestate.audio.left = rc % parsestate.audio.align) != 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
381 memmove(parsestate.audio.leftover,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
382 (char *)*outbuf + rc - parsestate.audio.left,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
383 parsestate.audio.left);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
384 rc -= parsestate.audio.left; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
385 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
386 parsestate.audio.skipping = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
387 parsestate.audio.isdata++;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
388 return(rc); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
389 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
390 /* Skip everything */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
391 parsestate.audio.skipping -= *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
392 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
393 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
394
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
395 /* If the soundcard could not be set to natively support the data format, we
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
396 try to do some limited on-the-fly conversion to a different format; if
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
397 no conversion is needed, though, we can output directly */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
398 static size_t sndcnvnop(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
399 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
400 int rc = *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
401
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
402 *outbuf = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
403 *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
404 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
405 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
406
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
407 /* Convert 8 bit unsigned stereo data to 8 bit unsigned mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
408 static size_t sndcnv8U_2mono(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
409 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
410 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
411 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
412 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
413
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
414 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
415 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
416 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
417 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
418 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
419 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
420 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
421 while (count--)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
422 *dest++ = (unsigned char)(((int)*((unsigned char *)src)++ +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
423 (int)*((unsigned char *)src)++) / 2);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
424 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
425 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
426 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
427
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
428 /* Convert 8 bit signed stereo data to 8 bit signed mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
429 static size_t sndcnv8S_2mono(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
430 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
431 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
432 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
433 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
434
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
435 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
436 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
437 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
438 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
439 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
440 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
441 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
442 while (count--)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
443 *dest++ = (unsigned char)(((int)*((signed char *)src)++ +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
444 (int)*((signed char *)src)++) / 2);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
445 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
446 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
447 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
448
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
449 /* Convert 8 bit signed stereo data to 8 bit unsigned mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
450 static size_t sndcnv2monounsigned(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
451 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
452 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
453 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
454 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
455
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
456 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
457 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
458 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
459 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
460 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
461 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
462 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
463 while (count--)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
464 *dest++ = (unsigned char)(((int)*((signed char *)src)++ +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
465 (int)*((signed char *)src)++) / 2) ^ 0x80;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
466 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
467 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
468 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
469
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
470 /* Convert 8 bit signed mono data to 8 bit unsigned mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
471 static size_t sndcnv2unsigned(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
472 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
473 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
474 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
475 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
476
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
477 count = *sz;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
478 if (count > SNDBUFSZ) { *sz -= SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
479 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
480 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
481 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
482 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
483 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
484 while (count--)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
485 *dest++ = *((unsigned char *)src)++ ^ 0x80;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
486 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
487 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
488 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
489
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
490 /* Convert a number in the range -32768..32767 to an 8 bit ulaw encoded
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
491 number --- I hope, I got this conversion right :-) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
492 static __inline__ signed char int2ulaw(int i)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
493 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
494 /* Lookup table for fast calculation of number of bits that need shifting*/
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
495 static short int t_bits[128] = {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
496 0,1,2,2,3,3,3,3,4,4,4,4,4,4,4,4,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
497 6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
498 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
499 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7};
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
500 register int bits,logi;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
501
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
502 /* unrolling this condition (hopefully) improves execution speed */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
503 if (i < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
504 if ((i = (132-i)) > 0x7FFF) i = 0x7FFF;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
505 logi = (i >> ((bits = t_bits[i/256])+4));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
506 return((bits << 4 | logi) ^ 0x7F); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
507 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
508 if ((i = 132+i) > 0x7FFF) i = 0x7FFF;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
509 logi = (i >> ((bits = t_bits[i/256])+4));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
510 return(~(bits << 4 | logi)); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
511 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
512
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
513 /* Convert 8 bit ulaw stereo data to 8 bit ulaw mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
514 static size_t sndcnvULaw_2mono(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
515 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
516
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
517 static short int ulaw2int[256] = {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
518 /* Precomputed lookup table for conversion from ulaw to 15 bit signed */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
519 -16062,-15550,-15038,-14526,-14014,-13502,-12990,-12478,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
520 -11966,-11454,-10942,-10430, -9918, -9406, -8894, -8382,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
521 -7998, -7742, -7486, -7230, -6974, -6718, -6462, -6206,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
522 -5950, -5694, -5438, -5182, -4926, -4670, -4414, -4158,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
523 -3966, -3838, -3710, -3582, -3454, -3326, -3198, -3070,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
524 -2942, -2814, -2686, -2558, -2430, -2302, -2174, -2046,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
525 -1950, -1886, -1822, -1758, -1694, -1630, -1566, -1502,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
526 -1438, -1374, -1310, -1246, -1182, -1118, -1054, -990,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
527 -942, -910, -878, -846, -814, -782, -750, -718,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
528 -686, -654, -622, -590, -558, -526, -494, -462,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
529 -438, -422, -406, -390, -374, -358, -342, -326,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
530 -310, -294, -278, -262, -246, -230, -214, -198,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
531 -186, -178, -170, -162, -154, -146, -138, -130,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
532 -122, -114, -106, -98, -90, -82, -74, -66,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
533 -60, -56, -52, -48, -44, -40, -36, -32,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
534 -28, -24, -20, -16, -12, -8, -4, +0,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
535 +16062,+15550,+15038,+14526,+14014,+13502,+12990,+12478,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
536 +11966,+11454,+10942,+10430, +9918, +9406, +8894, +8382,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
537 +7998, +7742, +7486, +7230, +6974, +6718, +6462, +6206,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
538 +5950, +5694, +5438, +5182, +4926, +4670, +4414, +4158,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
539 +3966, +3838, +3710, +3582, +3454, +3326, +3198, +3070,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
540 +2942, +2814, +2686, +2558, +2430, +2302, +2174, +2046,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
541 +1950, +1886, +1822, +1758, +1694, +1630, +1566, +1502,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
542 +1438, +1374, +1310, +1246, +1182, +1118, +1054, +990,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
543 +942, +910, +878, +846, +814, +782, +750, +718,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
544 +686, +654, +622, +590, +558, +526, +494, +462,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
545 +438, +422, +406, +390, +374, +358, +342, +326,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
546 +310, +294, +278, +262, +246, +230, +214, +198,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
547 +186, +178, +170, +162, +154, +146, +138, +130,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
548 +122, +114, +106, +98, +90, +82, +74, +66,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
549 +60, +56, +52, +48, +44, +40, +36, +32,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
550 +28, +24, +20, +16, +12, +8, +4, +0};
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
551
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
552 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
553 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
554 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
555
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
556 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
557 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
558 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
559 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
560 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
561 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
562 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
563 while (count--)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
564 /* it is not possible to directly interpolate between two ulaw encoded
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
565 data bytes, thus we need to convert to linear format first and later
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
566 we convert back to ulaw format */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
567 *dest++ = int2ulaw(ulaw2int[*((unsigned char *)src)++] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
568 ulaw2int[*((unsigned char *)src)++]);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
569 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
570 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
571 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
572
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
573 /* Convert 16 bit little endian signed stereo data to 16 bit little endian
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
574 signed mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
575 static size_t sndcnv16_2monoLE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
576 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
577 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
578 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
579 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
580 signed short i;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
581
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
582 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
583 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
584 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
585 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
586 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
587 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
588 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
589 for (count /= 2; count--; ) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
590 i = ((int)(((unsigned char *)src)[0]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
591 256*(int)(((unsigned char *)src)[1]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
592 (int)(((unsigned char *)src)[2]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
593 256*(int)(((unsigned char *)src)[3])) / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
594 src += 4;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
595 *dest++ = (unsigned char)(i & 0xFF);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
596 *dest++ = (unsigned char)((i / 256) & 0xFF); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
597 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
598 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
599 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
600
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
601 /* Convert 16 bit big endian signed stereo data to 16 bit big endian
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
602 signed mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
603 static size_t sndcnv16_2monoBE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
604 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
605 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
606 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
607 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
608 signed short i;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
609
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
610 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
611 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
612 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
613 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
614 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
615 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
616 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
617 for (count /= 2; count--; ) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
618 i = ((int)(((unsigned char *)src)[1]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
619 256*(int)(((unsigned char *)src)[0]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
620 (int)(((unsigned char *)src)[3]) +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
621 256*(int)(((unsigned char *)src)[2])) / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
622 src += 4;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
623 *dest++ = (unsigned char)((i / 256) & 0xFF);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
624 *dest++ = (unsigned char)(i & 0xFF); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
625 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
626 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
627 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
628
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
629 /* Convert 16 bit little endian signed data to 8 bit unsigned data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
630 static size_t sndcnv2byteLE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
631 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
632 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
633 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
634 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
635
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
636 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
637 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
638 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
639 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
640 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
641 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
642 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
643 while (count--) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
644 *dest++ = (unsigned char)(((signed char *)src)[1] ^ (signed char)0x80);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
645 ((char *)src) += 2; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
646 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
647 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
648 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
649
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
650 /* Convert 16 bit big endian signed data to 8 bit unsigned data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
651 static size_t sndcnv2byteBE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
652 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
653 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
654 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
655 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
656
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
657 count = *sz / 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
658 if (count > SNDBUFSZ) { *sz -= 2*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
659 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
660 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
661 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
662 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
663 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
664 while (count--) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
665 *dest++ = (unsigned char)(((signed char *)src)[0] ^ (signed char)0x80);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
666 ((char *)src) += 2; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
667 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
668 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
669 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
670
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
671 /* Convert 16 bit little endian signed stereo data to 8 bit unsigned
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
672 mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
673 static size_t sndcnv2monobyteLE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
674 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
675 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
676 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
677 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
678
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
679 count = *sz / 4;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
680 if (count > SNDBUFSZ) { *sz -= 4*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
681 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
682 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
683 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
684 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
685 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
686 while (count--) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
687 *dest++ = (unsigned char)(((int)((signed char *)src)[1] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
688 (int)((signed char *)src)[3]) / 2 ^ 0x80);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
689 ((char *)src) += 4; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
690 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
691 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
692 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
693
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
694 /* Convert 16 bit big endian signed stereo data to 8 bit unsigned
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
695 mono data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
696 static size_t sndcnv2monobyteBE(void **data,size_t *sz,void **outbuf)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
697 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
698 register unsigned char *src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
699 register unsigned char *dest;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
700 int rc,count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
701
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
702 count = *sz / 4;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
703 if (count > SNDBUFSZ) { *sz -= 4*SNDBUFSZ; count = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
704 else *sz = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
705 rc = count;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
706 src = *data;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
707 *outbuf =
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
708 dest = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
709 while (count--) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
710 *dest++ = (unsigned char)(((int)((signed char *)src)[0] +
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
711 (int)((signed char *)src)[2]) / 2 ^ 0x80);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
712 ((char *)src) += 4; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
713 *data = src;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
714 return(rc);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
715 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
716
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
717 /* Look at the header of the sound file and try to determine the format;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
718 we can recognize files in VOC, WAVE, and, Sun/DEC-audio format--- everything
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
719 else is assumed to be raw 8 bit unsigned data sampled at 8kHz */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
720 static fmtType analyze_format(unsigned char *format,int *fmt,int *speed,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
721 int *tracks,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
722 size_t (**parsesndfile)(void **,size_t *sz,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
723 void **))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
724 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
725 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
726 if (!memcmp(format,"Creative Voice File\x1A\x1A\x00",22) &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
727 (format[22]+256*format[23]) ==
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
728 ((0x1233-format[24]-256*format[25])&0xFFFF)) { /* VOC */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
729 *fmt = AFMT_U8;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
730 *speed = 8000;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
731 *tracks = 2;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
732 *parsesndfile = parsevoc;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
733 return(fmtVoc); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
734 else if (!memcmp(format,"RIFF",4) &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
735 !memcmp(format+8,"WAVEfmt ",8)) { /* WAVE */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
736 if (memcmp(format+20,"\001\000\001"/* PCM mono */,4) &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
737 memcmp(format+20,"\001\000\002"/* PCM stereo */,4))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
738 return(fmtIllegal);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
739 *fmt = (format[32]/(*tracks = format[22])) == 1 ?
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
740 AFMT_U8 : AFMT_S16_LE;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
741 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
742 *speed = format[24]+256*(format[25]+256*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
743 (format[26]+256*format[27]));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
744 *parsesndfile = parsewave;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
745 return(fmtWave); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
746 else if (!memcmp(format,".snd",4)) { /* Sun Audio (big endian) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
747 if (format[7]+256*(format[6]+256*(format[5]+256*format[4])) < 24) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
748 *fmt = AFMT_MU_LAW;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
749 *speed = 8000;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
750 *tracks = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
751 *parsesndfile = parsesundecaudio;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
752 return(fmtSunAudio); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
753 if (!memcmp(format+12,"\000\000\000\001",4)) *fmt = AFMT_MU_LAW;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
754 else if (!memcmp(format+12,"\000\000\000\002",4)) *fmt = AFMT_S8;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
755 else if (!memcmp(format+12,"\000\000\000\003",4)) *fmt = AFMT_S16_BE;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
756 else return(fmtIllegal);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
757 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
758 *speed = format[19]+256*(format[18]+256*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
759 (format[17]+256*format[16]));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
760 *tracks = format[23];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
761 *parsesndfile = parsesundecaudio;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
762 return(fmtSunAudio); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
763 else if (!memcmp(format,".sd",4)) { /* DEC Audio (little endian) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
764 if (format[4]+256*(format[5]+256*(format[6]+256*format[7])) < 24) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
765 *fmt = AFMT_MU_LAW;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
766 *speed = 8000;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
767 *tracks = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
768 *parsesndfile = parsesundecaudio;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
769 return(fmtSunAudio); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
770 if (!memcmp(format+12,"\001\000\000",4)) *fmt = AFMT_MU_LAW;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
771 else if (!memcmp(format+12,"\002\000\000",4)) *fmt = AFMT_S8;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
772 else if (!memcmp(format+12,"\003\000\000",4)) *fmt = AFMT_S16_LE;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
773 else return(fmtIllegal);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
774 /* Keep compatibility with Linux 68k, etc. by not relying on byte-sex */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
775 *speed = format[16]+256*(format[17]+256*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
776 (format[18]+256*format[19]));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
777 *tracks = format[20];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
778 *parsesndfile = parsesundecaudio;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
779 return(fmtSunAudio); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
780 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
781 *fmt = AFMT_U8;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
782 *speed = 8000;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
783 *tracks = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
784 *parsesndfile = parseraw;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
785 return(fmtRaw); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
786 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
787
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
788 /* Initialize the soundcard and mixer device with the parameters that we
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
789 found in the header of the sound file. If the soundcard is not capable of
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
790 natively supporting the required parameters, then try to set up conversion
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
791 routines.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
792 The difficulty with setting up the sound card is that the parameters are
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
793 not fully orthogonal; changing one of them might affect some of the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
794 others, too. Thus we do quite a lot of double checking; actually most of
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
795 this is not needed right now, but it will come in handy, if the kernel's
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
796 sounddriver ever changes or if third-party sounddrivers are used. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
797 static int audio_init(int mixx_fd, int auddio_fd, int fmt, int speed,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
798 int tracks, int *volume,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
799 size_t (**sndcnv) (void **, size_t *sz, void **))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
800 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
801 int i,the_speed,the_stereo,the_fmt;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
802
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
803 *sndcnv = sndcnvnop;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
804
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
805 if (ioctl(auddio_fd,SNDCTL_DSP_SYNC,NULL) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
806 perror("SNDCTL_DSP_SYNC");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
807 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
808
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
809 /* Initialize sound hardware with prefered parameters */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
810
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
811 /* The PCSP driver does not support reading of the sampling rate via the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
812 SOUND_PCM_READ_RATE ioctl; determine "the_speed" here */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
813 the_speed = speed; ioctl(audio_fd,SNDCTL_DSP_SPEED,&the_speed);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
814 /* The PCSP driver does not support reading of the mono/stereo flag, thus
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
815 we assume, that failure to change this mode means we are in mono mode */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
816 if (((i = (the_stereo = tracks)-1),ioctl(audio_fd,SNDCTL_DSP_STEREO,&i)) < 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
817 the_stereo = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
818 the_fmt = fmt; ioctl(audio_fd,SNDCTL_DSP_SETFMT,&the_fmt);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
819
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
820 /* If the sound hardware cannot support 16 bit format or requires a
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
821 different byte sex then try to drop to 8 bit format */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
822 the_fmt = AFMT_QUERY; ioctl(audio_fd,SNDCTL_DSP_SETFMT,&the_fmt);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
823 if (fmt != the_fmt) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
824 if (fmt == AFMT_S16_LE || fmt == AFMT_S16_BE) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
825 *sndcnv = fmt == AFMT_S16_BE ? sndcnv2byteBE : sndcnv2byteLE;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
826 if (((i=fmt=AFMT_U8),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
827 fmt != i || ioctl(audio_fd,SNDCTL_DSP_SETFMT,&the_fmt) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
828 fmt != the_fmt) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
829 perror("SNDCTL_DSP_SETFMT");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
830 return(0); } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
831 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
832 perror("SNDCTL_DSP_SETFMT");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
833 return(0); } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
834 else if (fmt == AFMT_S8) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
835 *sndcnv = sndcnv2unsigned;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
836 if (((i=fmt=AFMT_U8),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
837 fmt != i || ioctl(audio_fd,SNDCTL_DSP_SETFMT,&the_fmt) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
838 fmt != the_fmt) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
839 perror("SNDCTRL_DSP_SETFMT");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
840 return(0); } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
841
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
842 /* Try to request stereo playback (if needed); if this cannot be supported
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
843 by the hardware, then install conversion routines for mono playback */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
844
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
845 /* This ioctl will fail if we use the PCSP driver; thus the value of
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
846 "the_stereo" is still unchanged */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
847 ioctl(audio_fd,SOUND_PCM_READ_CHANNELS,&the_stereo);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
848 if (tracks != the_stereo) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
849 if (tracks == 2) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
850 tracks = 1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
851 *sndcnv = *sndcnv == sndcnv2byteLE ? sndcnv2monobyteLE :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
852 *sndcnv == sndcnv2byteBE ? sndcnv2monobyteBE :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
853 *sndcnv == sndcnv2unsigned ? sndcnv2monounsigned :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
854 the_fmt == AFMT_S16_LE ? sndcnv16_2monoLE :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
855 the_fmt == AFMT_S16_BE ? sndcnv16_2monoBE :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
856 the_fmt == AFMT_S8 ? sndcnv8S_2mono :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
857 the_fmt == AFMT_U8 ? sndcnv8U_2mono :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
858 the_fmt == AFMT_MU_LAW ? sndcnvULaw_2mono : NULL;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
859 if (*sndcnv == NULL) { /* this should not happen */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
860 perror("SNDCTL_DSP_STEREO");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
861 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
862 /* Switch to mono mode */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
863 if (((i = 0),ioctl(audio_fd,SNDCTL_DSP_STEREO,&i)) < 0 || i) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
864 perror("SNDCTL_DSP_STEREO");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
865 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
866 /* Now double check that everything is set as expected */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
867 if (((i = AFMT_QUERY),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
868 (i != the_fmt &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
869 (((i=the_fmt),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
870 i != the_fmt ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
871 ((i = AFMT_QUERY),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
872 i != the_fmt)) ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
873 (ioctl(audio_fd,SOUND_PCM_READ_CHANNELS,&i) >= 0 &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
874 i != 1)) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
875 /* There was no way that we could set the soundcard to a meaningful
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
876 mode */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
877 perror("SNDCTL_DSP_SETFMT and SNDCTL_DSP_STEREO");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
878 return(0); } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
879 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
880 /* Somebody set the soundcard to stereo even though we requested
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
881 mono; this should not happen... */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
882 if (((i = the_stereo = tracks),ioctl(audio_fd,SNDCTL_DSP_STEREO,&i))<0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
883 i != the_stereo-1) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
884 perror("SNDCTL_DSP_STEREO");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
885 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
886 if (((i = AFMT_QUERY),ioctl(audio_fd,SNDCTL_DSP_SETFMT,&i)) < 0 ||
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
887 i != the_fmt) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
888 perror("SNDCTL_DSP_SETFMT");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
889 return(0); } } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
890
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
891 /* Fail if deviations from desired sampling frequency are too big */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
892
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
893 /* This ioctl will fail if we use the PCSP driver; thus the value of
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
894 "the_speed" is still unchanged */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
895 ioctl(audio_fd,SOUND_PCM_READ_RATE,&the_speed);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
896 if (speed*14 < the_speed*10 || speed*6 > the_speed*10) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
897 char buffer[256];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
898 sprintf(buffer,"SNDCTL_DSP_SPEED (req: %d, rtn: %d)",speed,the_speed);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
899 perror(buffer);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
900 return(0); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
901
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
902 /* Use the mixer device for setting the playback volume */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
903 if (mixx_fd > 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
904 int vol = *volume & 0xFF;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
905 if (ioctl(mixx_fd,SOUND_MIXER_READ_PCM,volume) < 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
906 *volume = -1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
907 if (vol < 0) vol = 0; else if (vol > 100) vol = 100;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
908 #ifdef NOVOLUMECTRLFORMULAW
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
909 if (fmt == AFMT_MU_LAW)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
910 vol = 100;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
911 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
912 vol |= 256*vol;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
913 /* Do not signal an error, if volume control is unavailable! */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
914 ioctl(mixx_fd,SOUND_MIXER_WRITE_PCM,&vol); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
915
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
916 #if defined(LINUXPLAYSTANDALONE) && 1
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
917 /* Debugging output is displayed only when compiled as stand-alone version */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
918 {int the_volume;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
919 the_fmt = AFMT_QUERY;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
920 ioctl(audio_fd,SNDCTL_DSP_SETFMT,&the_fmt);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
921 ioctl(auddio_fd,SOUND_PCM_READ_CHANNELS,&the_stereo);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
922 ioctl(auddio_fd,SOUND_PCM_READ_RATE,&the_speed);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
923 ioctl(mixx_fd,SOUND_MIXER_READ_PCM,&the_volume);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
924 fprintf(stderr,"%s, %s, %dHz, L:%d/R:%d\n",
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
925 the_fmt == AFMT_MU_LAW ? "AFMT_MU_LAW" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
926 the_fmt == AFMT_A_LAW ? "AFMT_A_LAW" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
927 the_fmt == AFMT_IMA_ADPCM ? "AFMT_IMA_ADPCM" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
928 the_fmt == AFMT_U8 ? "AFMT_U8" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
929 the_fmt == AFMT_S16_LE ? "AFMT_S16_LE" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
930 the_fmt == AFMT_S16_BE ? "AFMT_S16_BE" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
931 the_fmt == AFMT_S8 ? "AFMT_S8" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
932 the_fmt == AFMT_U16_LE ? "AFMT_U16_LE" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
933 the_fmt == AFMT_U16_BE ? "AFMT_U16_BE" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
934 the_fmt == AFMT_MPEG ? "AFMT_MPEG" :
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
935 "AFMT_???",
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
936 the_stereo == 2 ? "stereo" : "mono",
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
937 the_speed,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
938 the_volume / 256, the_volume % 256); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
939 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
940
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
941 return(1);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
942 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
943
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
944 /* XEmacs requires code both for playback of pre-loaded data and for playback
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
945 from a soundfile; we use one function for both cases */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
946 static void linux_play_data_or_file(int fd,unsigned char *data,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
947 int length,int volume)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
948 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
949 size_t (*parsesndfile)(void **dayta,size_t *sz,void **outbuf);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
950 size_t (*sndcnv)(void **dayta,size_t *sz,void **);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
951 fmtType ffmt;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
952 int fmt,speed,tracks;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
953 unsigned char *pptr,*optr,*cptr,*sptr;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
954 int wrtn,rrtn,crtn,prtn;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
955
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
956 /* We need to read at least the header information before we can start
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
957 doing anything */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
958 if (!data || length < HEADERSZ)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
959 if (fd < 0) return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
960 else {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
961 length = read(fd,sndbuf,SNDBUFSZ);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
962 if (length < HEADERSZ)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
963 return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
964 data = sndbuf;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
965 length = SNDBUFSZ; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
966
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
967 ffmt = analyze_format(data,&fmt,&speed,&tracks,&parsesndfile);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
968
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
969 if (ffmt != fmtRaw && ffmt != fmtSunAudio && ffmt != fmtWave) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
970 warn("Unsupported file format (neither RAW, nor Sun/DECAudio, nor WAVE)");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
971 return; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
972
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
973 /* The VoxWare-SDK discourages opening /dev/audio; opening /dev/dsp and
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
974 properly intializing it via ioctl() is prefered */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
975 if ((audio_fd=open((audio_dev="/dev/dsp"),
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
976 (O_WRONLY|O_NDELAY),0)) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
977 perror(audio_dev);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
978 if (mix_fd > 0 && mix_fd != audio_fd) { close(mix_fd); mix_fd = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
979 return; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
980
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
981 /* The VoxWare-SDK discourages direct manipulation of the mixer device as
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
982 this could lead to problems, when multiple sound cards are installed */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
983 mix_fd = audio_fd;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
984
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
985 sighup_handler = signal(SIGHUP,(__sighandler_t)sighandler);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
986 sigint_handler = signal(SIGINT,(__sighandler_t)sighandler);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
987
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
988 if (!audio_init(mix_fd,audio_fd,fmt,speed,tracks,&volume,&sndcnv))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
989 goto END_OF_PLAY;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
990 audio_vol = volume;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
991
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
992 /* Initialize global parser state information to zero */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
993 memset(&parsestate,0,sizeof(parsestate));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
994
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
995 /* Mainloop: read a block of data, parse its contents, perform all
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
996 the neccessary conversions and output it to the sound
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
997 device; repeat until all data has been processed */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
998 rrtn = length;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
999 do {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1000 for (pptr = data; (prtn = parsesndfile((void **)&pptr,&rrtn,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1001 (void **)&optr)) > 0; )
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1002 for (cptr = optr; (crtn = sndcnv((void **)&cptr,&prtn,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1003 (void **)&sptr)) > 0; ) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1004 for (;;) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1005 if ((wrtn = write(audio_fd,sptr,crtn)) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1006 perror("write"); goto END_OF_PLAY; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1007 else if (wrtn) break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1008 else if (ioctl(audio_fd,SNDCTL_DSP_SYNC,NULL) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1009 perror("SNDCTL_DSP_SYNC"); goto END_OF_PLAY; } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1010 if (wrtn != crtn) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1011 char buf[255];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1012 sprintf(buf,"play: crtn = %d, wrtn = %d",crtn,wrtn);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1013 warn(buf);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1014 goto END_OF_PLAY; } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1015 if (fd >= 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1016 if ((rrtn = read(fd,sndbuf,SNDBUFSZ)) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1017 perror("read"); goto END_OF_PLAY; } }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1018 else
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1019 break;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1020 } while (rrtn > 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1021
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1022 /* Verify that we could fully parse the entire soundfile; this is needed
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1023 only for files in WAVE format */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1024 if (ffmt == fmtWave && parsestate.wave.state != wvOutOfBlock &&
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1025 parsestate.wave.state != wvFatal)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1026 warn("Unexpected end of WAVE file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1027
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1028 END_OF_PLAY:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1029 /* Now cleanup all used resources */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1030
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1031 ioctl(audio_fd,SNDCTL_DSP_SYNC,NULL);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1032 ioctl(audio_fd,SNDCTL_DSP_RESET,NULL);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1033
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1034 signal(SIGHUP,sighup_handler);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1035 signal(SIGINT,sigint_handler);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1036
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1037 if (mix_fd > 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1038 if (audio_vol >= 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1039 ioctl(mix_fd,SOUND_MIXER_WRITE_PCM,&audio_vol);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1040 audio_vol = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1041 if (mix_fd != audio_fd)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1042 close(mix_fd);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1043 mix_fd = -1; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1044
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1045 close(audio_fd);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1046 audio_fd = -1;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1047
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1048 return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1049 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1050
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1051 /* Call "linux_play_data_or_file" with the appropriate parameters for
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1052 playing a soundfile */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1053 void play_sound_file (char *sound_file, int volume);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1054 void play_sound_file (char *sound_file, int volume)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1055 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1056 int fd;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1057
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1058 if ((fd=open(sound_file,O_RDONLY,0)) < 0) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1059 perror(sound_file);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1060 return; }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1061 linux_play_data_or_file(fd,NULL,0,volume);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1062 close(fd);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1063 return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1064 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1065
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1066 /* Call "linux_play_data_or_file" with the appropriate parameters for
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1067 playing pre-loaded data */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1068 void play_sound_data (unsigned char *data, int length, int volume);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1069 void play_sound_data (unsigned char *data, int length, int volume)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1070 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1071 linux_play_data_or_file(-1,data,length,volume);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1072 return;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1073 }