Rev 2 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
1 | pmbaty | 1 | /* |
2 | * src/syssnd.c |
||
3 | * |
||
4 | * Copyright (C) 1998-2002 BigOrno (bigorno@bigorno.net). All rights reserved. |
||
5 | * |
||
6 | * The use and distribution terms for this software are contained in the file |
||
7 | * named README, which can be found in the root of this distribution. By |
||
8 | * using this software in any fashion, you are agreeing to be bound by the |
||
9 | * terms of this license. |
||
10 | * |
||
11 | * You must not remove this notice, or any other, from this software. |
||
12 | */ |
||
13 | |||
14 | #include <SDL.h> |
||
15 | #include <stdlib.h> |
||
16 | #include <memory.h> |
||
17 | |||
18 | #include "system.h" |
||
19 | #include "game.h" |
||
20 | #include "syssnd.h" |
||
21 | |||
22 | |||
23 | #define ADJVOL(S) (((S) *sndVol) / SDL_MIX_MAXVOLUME) |
||
24 | |||
25 | |||
26 | static U8 isAudioActive = FALSE; |
||
5 | pmbaty | 27 | static channel_t channels[SYSSND_MIXCHANNELS]; |
1 | pmbaty | 28 | |
29 | static U8 sndVol = SDL_MIX_MAXVOLUME; /* internal volume */ |
||
30 | static U8 sndUVol = SYSSND_MAXVOL; /* user-selected volume */ |
||
31 | static U8 sndMute = FALSE; /* mute flag */ |
||
32 | |||
33 | |||
34 | /* |
||
35 | * prototypes |
||
36 | */ |
||
5 | pmbaty | 37 | static void end_channel (channel_t *); |
1 | pmbaty | 38 | |
39 | |||
40 | /* |
||
41 | * Callback -- this is also where all sound mixing is done |
||
42 | * |
||
43 | * Note: it may not be that much a good idea to do all the mixing here ; it |
||
44 | * may be more efficient to mix samples every frame, or maybe everytime a |
||
45 | * new sound is sent to be played. I don't know. |
||
46 | */ |
||
47 | void syssnd_callback (void *userdata, U8 *stream, int len) |
||
48 | { |
||
5 | pmbaty | 49 | channel_t *channel; |
50 | U8 channel_index; |
||
51 | S16 final_sample; |
||
1 | pmbaty | 52 | U32 i; |
53 | |||
5 | pmbaty | 54 | // |
55 | for (i = 0; i < (U32)len; i += 2) |
||
56 | { |
||
57 | final_sample = 0; |
||
1 | pmbaty | 58 | |
5 | pmbaty | 59 | for (channel_index = 0; channel_index < SYSSND_MIXCHANNELS; channel_index++) |
1 | pmbaty | 60 | { |
5 | pmbaty | 61 | channel = &channels[channel_index]; // quick access to channel |
62 | |||
63 | if (channel->loop == 0) |
||
64 | continue; |
||
65 | |||
66 | // channel is active |
||
67 | if (channel->len > 0) |
||
1 | pmbaty | 68 | { |
5 | pmbaty | 69 | // not ending |
70 | final_sample += ADJVOL (*((U16 *) channel->buf)); |
||
71 | channel->buf += 2; |
||
72 | channel->len -= 2; |
||
73 | } |
||
74 | else |
||
75 | { |
||
76 | // ending |
||
77 | if (channel->loop > 0) |
||
78 | channel->loop--; |
||
79 | |||
80 | if (channel->loop) |
||
1 | pmbaty | 81 | { |
5 | pmbaty | 82 | // just loop |
83 | channel->buf = channel->snd->buf; |
||
84 | channel->len = channel->snd->len; |
||
85 | |||
86 | final_sample += ADJVOL (*((U16 *) channel->buf)); |
||
87 | channel->buf += 2; |
||
88 | channel->len -= 2; |
||
1 | pmbaty | 89 | } |
90 | else |
||
5 | pmbaty | 91 | end_channel (channel); // end for real |
1 | pmbaty | 92 | } |
93 | } |
||
94 | |||
5 | pmbaty | 95 | *((S16 *) &stream[i]) = (sndMute ? 0 : final_sample); |
1 | pmbaty | 96 | } |
97 | } |
||
98 | |||
99 | |||
5 | pmbaty | 100 | static void end_channel (channel_t *channel) |
1 | pmbaty | 101 | { |
5 | pmbaty | 102 | channel->loop = 0; |
103 | if (channel->snd->dispose) |
||
104 | syssnd_free (channel->snd); |
||
105 | channel->snd = NULL; |
||
1 | pmbaty | 106 | } |
107 | |||
108 | |||
109 | void syssnd_init (void) |
||
110 | { |
||
5 | pmbaty | 111 | SDL_AudioSpec desired; |
1 | pmbaty | 112 | U16 c; |
113 | |||
114 | if (SDL_InitSubSystem (SDL_INIT_AUDIO) < 0) |
||
115 | return; |
||
116 | |||
117 | desired.freq = SYSSND_FREQ; |
||
5 | pmbaty | 118 | desired.format = AUDIO_S16; |
1 | pmbaty | 119 | desired.channels = SYSSND_CHANNELS; |
120 | desired.samples = SYSSND_MIXSAMPLES; |
||
121 | desired.callback = syssnd_callback; |
||
122 | desired.userdata = NULL; |
||
123 | |||
5 | pmbaty | 124 | if (SDL_OpenAudio (&desired, NULL) < 0) |
1 | pmbaty | 125 | return; |
126 | |||
127 | if (sysarg_args_vol != 0) |
||
128 | { |
||
129 | sndUVol = sysarg_args_vol; |
||
130 | sndVol = SDL_MIX_MAXVOLUME * sndUVol / SYSSND_MAXVOL; |
||
131 | } |
||
132 | |||
133 | for (c = 0; c < SYSSND_MIXCHANNELS; c++) |
||
5 | pmbaty | 134 | channels[c].loop = 0; /* deactivate */ |
1 | pmbaty | 135 | |
136 | isAudioActive = TRUE; |
||
137 | SDL_PauseAudio (0); |
||
138 | } |
||
139 | |||
140 | |||
141 | /* |
||
142 | * Shutdown |
||
143 | */ |
||
144 | void syssnd_shutdown (void) |
||
145 | { |
||
146 | if (!isAudioActive) |
||
147 | return; |
||
148 | |||
149 | SDL_CloseAudio (); |
||
150 | isAudioActive = FALSE; |
||
151 | } |
||
152 | |||
153 | |||
154 | /* |
||
155 | * Toggle mute |
||
156 | * |
||
157 | * When muted, sounds are still managed but not sent to the dsp, hence |
||
158 | * it is possible to un-mute at any time. |
||
159 | */ |
||
160 | void syssnd_toggleMute (void) |
||
161 | { |
||
162 | sndMute = !sndMute; |
||
163 | } |
||
164 | |||
165 | |||
166 | void syssnd_vol (S8 d) |
||
167 | { |
||
168 | if ((d < 0 && sndUVol > 0) || (d > 0 && sndUVol < SYSSND_MAXVOL)) |
||
169 | { |
||
170 | sndUVol += d; |
||
171 | sndVol = SDL_MIX_MAXVOLUME * sndUVol / SYSSND_MAXVOL; |
||
172 | } |
||
173 | } |
||
174 | |||
175 | |||
176 | /* |
||
177 | * Play a sound |
||
178 | * |
||
179 | * loop: number of times the sound should be played, -1 to loop forever |
||
180 | * returns: channel number, or -1 if none was available |
||
181 | * |
||
182 | * NOTE if sound is already playing, simply reset it (i.e. can not have |
||
183 | * twice the same sound playing -- tends to become noisy when too many |
||
184 | * bad guys die at the same time). |
||
185 | */ |
||
186 | S8 syssnd_play (sound_t *sound, S8 loop) |
||
187 | { |
||
5 | pmbaty | 188 | S8 channel_index; |
1 | pmbaty | 189 | |
5 | pmbaty | 190 | if ((sound == NULL) || !isAudioActive) |
191 | return (-1); |
||
1 | pmbaty | 192 | |
5 | pmbaty | 193 | for (channel_index = 0; channel_index < SYSSND_MIXCHANNELS; channel_index++) |
194 | if ((channels[channel_index].snd == sound) || (channels[channel_index].loop == 0)) |
||
195 | break; // look for a free channel slot |
||
196 | if (channel_index == SYSSND_MIXCHANNELS) |
||
197 | return (-1); |
||
1 | pmbaty | 198 | |
5 | pmbaty | 199 | channels[channel_index].loop = loop; |
200 | channels[channel_index].snd = sound; |
||
201 | channels[channel_index].buf = sound->buf; |
||
202 | channels[channel_index].len = sound->len; |
||
1 | pmbaty | 203 | |
5 | pmbaty | 204 | return (channel_index); |
1 | pmbaty | 205 | } |
206 | |||
207 | |||
208 | /* |
||
209 | * Pause |
||
210 | * |
||
211 | * pause: TRUE or FALSE |
||
212 | * clear: TRUE to cleanup all sounds and make sure we start from scratch |
||
213 | */ |
||
5 | pmbaty | 214 | void syssnd_pause (U8 want_pause, U8 want_clear) |
1 | pmbaty | 215 | { |
5 | pmbaty | 216 | U8 channel_index; |
1 | pmbaty | 217 | |
218 | if (!isAudioActive) |
||
219 | return; |
||
220 | |||
5 | pmbaty | 221 | if (want_clear) |
222 | for (channel_index = 0; channel_index < SYSSND_MIXCHANNELS; channel_index++) |
||
223 | channels[channel_index].loop = 0; |
||
1 | pmbaty | 224 | |
5 | pmbaty | 225 | SDL_PauseAudio (want_pause ? 1 : 0); |
1 | pmbaty | 226 | } |
227 | |||
228 | |||
229 | /* |
||
230 | * Stop a channel |
||
231 | */ |
||
232 | void syssnd_stopchan (S8 chan) |
||
233 | { |
||
234 | if (chan < 0 || chan > SYSSND_MIXCHANNELS) |
||
235 | return; |
||
236 | |||
5 | pmbaty | 237 | if (channels[chan].snd) |
238 | end_channel (&channels[chan]); |
||
1 | pmbaty | 239 | } |
240 | |||
241 | |||
242 | /* |
||
243 | * Stop a sound |
||
244 | */ |
||
245 | void syssnd_stopsound (sound_t *sound) |
||
246 | { |
||
247 | U8 i; |
||
248 | |||
249 | if (!sound) |
||
250 | return; |
||
251 | |||
252 | for (i = 0; i < SYSSND_MIXCHANNELS; i++) |
||
5 | pmbaty | 253 | if (channels[i].snd == sound) |
254 | end_channel (&channels[i]); |
||
1 | pmbaty | 255 | } |
256 | |||
257 | |||
258 | /* |
||
259 | * See if a sound is playing |
||
260 | */ |
||
261 | int syssnd_isplaying (sound_t *sound) |
||
262 | { |
||
5 | pmbaty | 263 | U8 i; |
1 | pmbaty | 264 | |
265 | for (i = 0; i < SYSSND_MIXCHANNELS; i++) |
||
5 | pmbaty | 266 | if (channels[i].snd == sound) |
267 | return (1); |
||
1 | pmbaty | 268 | |
5 | pmbaty | 269 | return (0); |
1 | pmbaty | 270 | } |
271 | |||
272 | |||
273 | /* |
||
274 | * Stops all channels. |
||
275 | */ |
||
276 | void syssnd_stopall (void) |
||
277 | { |
||
278 | U8 i; |
||
279 | |||
280 | for (i = 0; i < SYSSND_MIXCHANNELS; i++) |
||
5 | pmbaty | 281 | if (channels[i].snd) |
282 | end_channel (&channels[i]); |
||
1 | pmbaty | 283 | } |
284 | |||
285 | |||
286 | /* |
||
287 | * Load a sound. |
||
288 | */ |
||
289 | sound_t *syssnd_load (char *name) |
||
290 | { |
||
2 | pmbaty | 291 | char *soundfile_fullpathname; |
1 | pmbaty | 292 | sound_t *s; |
293 | SDL_AudioSpec audiospec; |
||
294 | |||
2 | pmbaty | 295 | /* alloc space for sound file's full pathname and build it */ |
296 | soundfile_fullpathname = (char *) malloc (1024); |
||
297 | sprintf_s (soundfile_fullpathname, 1024, "%s/%s", sys_getbasepath (), name); |
||
298 | |||
1 | pmbaty | 299 | /* alloc sound */ |
300 | s = malloc (sizeof (sound_t)); |
||
301 | |||
302 | /* read */ |
||
303 | /* second param == 1 -> close source once read */ |
||
5 | pmbaty | 304 | if (!SDL_LoadWAV_RW (SDL_RWFromFile (soundfile_fullpathname, "rb"), 1, &audiospec, &(s->buf), &(s->len))) |
1 | pmbaty | 305 | { |
306 | free (s); |
||
307 | return NULL; |
||
308 | } |
||
5 | pmbaty | 309 | free (soundfile_fullpathname); |
1 | pmbaty | 310 | |
311 | s->dispose = FALSE; |
||
312 | |||
313 | return s; |
||
314 | } |
||
315 | |||
316 | |||
317 | /* |
||
318 | * |
||
319 | */ |
||
320 | void syssnd_free (sound_t *s) |
||
321 | { |
||
322 | if (!s) |
||
323 | return; |
||
324 | |||
325 | if (s->buf) |
||
326 | SDL_FreeWAV (s->buf); |
||
327 | |||
328 | s->buf = NULL; |
||
329 | s->len = 0; |
||
330 | } |