Mercurial > pidgin.yaz
comparison src/sound.c @ 4430:801830dec409
[gaim-migrate @ 4705]
robot101 gave me a bad patch
committer: Tailor Script <tailor@pidgin.im>
author | Luke Schierer <lschiere@pidgin.im> |
---|---|
date | Sun, 26 Jan 2003 20:43:25 +0000 |
parents | bf770f11132b |
children | 3196d9044a45 |
comparison
equal
deleted
inserted
replaced
4429:bf770f11132b | 4430:801830dec409 |
---|---|
38 | 38 |
39 #include <fcntl.h> | 39 #include <fcntl.h> |
40 #include <sys/types.h> | 40 #include <sys/types.h> |
41 #include <sys/stat.h> | 41 #include <sys/stat.h> |
42 | 42 |
43 #ifdef USE_AO | 43 #ifdef ESD_SOUND |
44 #include <ao/ao.h> | 44 #include <esd.h> |
45 #include <audiofile.h> | 45 #endif |
46 #endif /* USE_AO */ | 46 |
47 #ifdef ARTSC_SOUND | |
48 #include <artsc.h> | |
49 #endif | |
50 | |
51 #ifdef NAS_SOUND | |
52 #include <audio/audiolib.h> | |
53 #endif | |
47 | 54 |
48 #include "gaim.h" | 55 #include "gaim.h" |
49 | 56 |
50 #ifdef _WIN32 | 57 #ifdef _WIN32 |
51 #include "win32dep.h" | 58 #include "win32dep.h" |
52 #endif | 59 #endif |
53 | |
54 #ifdef USE_AO | |
55 static gboolean ao_initialized = FALSE; | |
56 #endif /* USE_AO */ | |
57 | 60 |
58 gboolean mute_sounds = 0; | 61 gboolean mute_sounds = 0; |
59 | 62 |
60 /* description, option bit, default sound file * | 63 /* description, option bit, default sound file * |
61 * set the option bit to 0 to have it not display in prefs * | 64 * set the option bit to 0 to have it not display in prefs * |
74 /* this isn't a terminator, it's the buddy pounce default sound event ;-) */ | 77 /* this isn't a terminator, it's the buddy pounce default sound event ;-) */ |
75 {NULL, 0, "redalert.wav"}, | 78 {NULL, 0, "redalert.wav"}, |
76 {N_("Someone says your name in chat"), OPT_SOUND_CHAT_NICK, "redalert.wav"} | 79 {N_("Someone says your name in chat"), OPT_SOUND_CHAT_NICK, "redalert.wav"} |
77 }; | 80 }; |
78 | 81 |
82 #ifndef _WIN32 | |
83 static int check_dev(char *dev) | |
84 { | |
85 struct stat stat_buf; | |
86 uid_t user = getuid(); | |
87 gid_t group = getgid(), other_groups[32]; | |
88 int i, numgroups; | |
89 | |
90 if ((numgroups = getgroups(32, other_groups)) == -1) | |
91 return 0; | |
92 if (stat(dev, &stat_buf)) | |
93 return 0; | |
94 if (user == stat_buf.st_uid && stat_buf.st_mode & S_IWUSR) | |
95 return 1; | |
96 if (stat_buf.st_mode & S_IWGRP) { | |
97 if (group == stat_buf.st_gid) | |
98 return 1; | |
99 for (i = 0; i < numgroups; i++) | |
100 if (other_groups[i] == stat_buf.st_gid) | |
101 return 1; | |
102 } | |
103 if (stat_buf.st_mode & S_IWOTH) | |
104 return 1; | |
105 return 0; | |
106 } | |
107 | |
108 static void play_audio_file(char *file) | |
109 { | |
110 /* here we can assume that we can write to /dev/audio */ | |
111 char *buf; | |
112 struct stat info; | |
113 int fd = open(file, O_RDONLY); | |
114 if (fd <= 0) { | |
115 return; | |
116 } | |
117 fstat(fd, &info); | |
118 if (info.st_size < 24) | |
119 return; | |
120 buf = malloc(info.st_size + 1); | |
121 read(fd, buf, 24); | |
122 read(fd, buf, info.st_size - 24); | |
123 close(fd); | |
124 | |
125 fd = open("/dev/audio", O_WRONLY | O_EXCL | O_NDELAY); | |
126 if (fd < 0) { | |
127 free(buf); | |
128 return; | |
129 } | |
130 write(fd, buf, info.st_size - 24); | |
131 free(buf); | |
132 close(fd); | |
133 } | |
134 | |
135 static int can_play_audio() | |
136 { | |
137 return check_dev("/dev/audio"); | |
138 } | |
139 | |
140 #ifdef ESD_SOUND | |
141 | |
142 int esd_fd; | |
143 | |
144 static int can_play_esd() | |
145 { | |
146 esd_format_t format = ESD_BITS16 | ESD_STREAM | ESD_PLAY | ESD_MONO; | |
147 | |
148 esd_fd = esd_play_stream(format, 8012, NULL, "gaim"); | |
149 | |
150 if (esd_fd < 0) { | |
151 return 0; | |
152 } | |
153 | |
154 return 1; | |
155 } | |
156 | |
157 #endif /* ESD_SOUND */ | |
158 | |
159 #ifdef ARTSC_SOUND | |
160 | |
161 /* | |
162 ** This routine converts from ulaw to 16 bit linear. | |
163 ** | |
164 ** Craig Reese: IDA/Supercomputing Research Center | |
165 ** 29 September 1989 | |
166 ** | |
167 ** References: | |
168 ** 1) CCITT Recommendation G.711 (very difficult to follow) | |
169 ** 2) MIL-STD-188-113,"Interoperability and Performance Standards | |
170 ** for Analog-to_Digital Conversion Techniques," | |
171 ** 17 February 1987 | |
172 ** | |
173 ** Input: 8 bit ulaw sample | |
174 ** Output: signed 16 bit linear sample | |
175 ** Z-note -- this is from libaudiofile. Thanks guys! | |
176 */ | |
177 | |
178 static int _af_ulaw2linear(unsigned char ulawbyte) | |
179 { | |
180 static int exp_lut[8] = { 0, 132, 396, 924, 1980, 4092, 8316, 16764 }; | |
181 int sign, exponent, mantissa, sample; | |
182 | |
183 ulawbyte = ~ulawbyte; | |
184 sign = (ulawbyte & 0x80); | |
185 exponent = (ulawbyte >> 4) & 0x07; | |
186 mantissa = ulawbyte & 0x0F; | |
187 sample = exp_lut[exponent] + (mantissa << (exponent + 3)); | |
188 if (sign != 0) | |
189 sample = -sample; | |
190 | |
191 return (sample); | |
192 } | |
193 | |
194 static int play_artsc(unsigned char *data, int size) | |
195 { | |
196 arts_stream_t stream; | |
197 guint16 *lineardata; | |
198 int result = 1; | |
199 int error; | |
200 int i; | |
201 | |
202 lineardata = g_malloc(size * 2); | |
203 | |
204 for (i = 0; i < size; i++) { | |
205 lineardata[i] = _af_ulaw2linear(data[i]); | |
206 } | |
207 | |
208 stream = arts_play_stream(8012, 16, 1, "gaim"); | |
209 | |
210 error = arts_write(stream, lineardata, size); | |
211 if (error < 0) { | |
212 result = 0; | |
213 } | |
214 | |
215 arts_close_stream(stream); | |
216 | |
217 g_free(lineardata); | |
218 | |
219 arts_free(); | |
220 | |
221 return result; | |
222 } | |
223 | |
224 static int can_play_artsc() | |
225 { | |
226 int error; | |
227 | |
228 error = arts_init(); | |
229 if (error < 0) | |
230 return 0; | |
231 | |
232 return 1; | |
233 } | |
234 | |
235 static int artsc_play_file(char *file) | |
236 { | |
237 struct stat stat_buf; | |
238 unsigned char *buf = NULL; | |
239 int result = 0; | |
240 int fd = -1; | |
241 | |
242 if (!can_play_artsc()) | |
243 return 0; | |
244 | |
245 fd = open(file, O_RDONLY); | |
246 if (fd < 0) | |
247 return 0; | |
248 | |
249 if (fstat(fd, &stat_buf)) { | |
250 close(fd); | |
251 return 0; | |
252 } | |
253 | |
254 if (!stat_buf.st_size) { | |
255 close(fd); | |
256 return 0; | |
257 } | |
258 | |
259 buf = g_malloc(stat_buf.st_size); | |
260 if (!buf) { | |
261 close(fd); | |
262 return 0; | |
263 } | |
264 | |
265 if (read(fd, buf, stat_buf.st_size) < 0) { | |
266 g_free(buf); | |
267 close(fd); | |
268 return 0; | |
269 } | |
270 | |
271 result = play_artsc(buf, stat_buf.st_size); | |
272 | |
273 g_free(buf); | |
274 close(fd); | |
275 return result; | |
276 } | |
277 | |
278 #endif /* ARTSC_SOUND */ | |
279 | |
280 #ifdef NAS_SOUND | |
281 | |
282 char nas_server[] = "localhost"; | |
283 AuServer *nas_serv = NULL; | |
284 | |
285 static AuBool NasEventHandler(AuServer * aud, AuEvent * ev, AuEventHandlerRec * handler) | |
286 { | |
287 AuElementNotifyEvent *event = (AuElementNotifyEvent *) ev; | |
288 | |
289 if (ev->type == AuEventTypeElementNotify) { | |
290 switch (event->kind) { | |
291 case AuElementNotifyKindState: | |
292 switch (event->cur_state) { | |
293 case AuStateStop: | |
294 _exit(0); | |
295 } | |
296 break; | |
297 } | |
298 } | |
299 return AuTrue; | |
300 } | |
301 | |
302 | |
303 static int play_nas(unsigned char *data, int size) | |
304 { | |
305 AuDeviceID device = AuNone; | |
306 AuFlowID flow; | |
307 AuElement elements[3]; | |
308 int i, n, w; | |
309 | |
310 /* look for an output device */ | |
311 for (i = 0; i < AuServerNumDevices(nas_serv); i++) { | |
312 if ((AuDeviceKind(AuServerDevice(nas_serv, i)) == | |
313 AuComponentKindPhysicalOutput) && | |
314 AuDeviceNumTracks(AuServerDevice(nas_serv, i)) == 1) { | |
315 device = AuDeviceIdentifier(AuServerDevice(nas_serv, i)); | |
316 break; | |
317 } | |
318 } | |
319 | |
320 if (device == AuNone) | |
321 return 0; | |
322 | |
323 if (!(flow = AuCreateFlow(nas_serv, NULL))) | |
324 return 0; | |
325 | |
326 | |
327 AuMakeElementImportClient(&elements[0], 8012, AuFormatULAW8, 1, AuTrue, size, size / 2, 0, NULL); | |
328 AuMakeElementExportDevice(&elements[1], 0, device, 8012, AuUnlimitedSamples, 0, NULL); | |
329 AuSetElements(nas_serv, flow, AuTrue, 2, elements, NULL); | |
330 | |
331 AuStartFlow(nas_serv, flow, NULL); | |
332 | |
333 AuWriteElement(nas_serv, flow, 0, size, data, AuTrue, NULL); | |
334 | |
335 AuRegisterEventHandler(nas_serv, AuEventHandlerIDMask, 0, flow, NasEventHandler, NULL); | |
336 | |
337 while (1) { | |
338 AuHandleEvents(nas_serv); | |
339 } | |
340 | |
341 return 1; | |
342 } | |
343 | |
344 static int can_play_nas() | |
345 { | |
346 if ((nas_serv = AuOpenServer(NULL, 0, NULL, 0, NULL, NULL))) | |
347 return 1; | |
348 return 0; | |
349 } | |
350 | |
351 static int play_nas_file(char *file) | |
352 { | |
353 struct stat stat_buf; | |
354 char *buf; | |
355 int ret; | |
356 int fd = open(file, O_RDONLY); | |
357 if (fd <= 0) | |
358 return 0; | |
359 | |
360 if (!can_play_nas()) | |
361 return 0; | |
362 | |
363 if (stat(file, &stat_buf)) | |
364 return 0; | |
365 | |
366 if (!stat_buf.st_size) | |
367 return 0; | |
368 | |
369 buf = malloc(stat_buf.st_size); | |
370 read(fd, buf, stat_buf.st_size); | |
371 ret = play_nas(buf, stat_buf.st_size); | |
372 free(buf); | |
373 return ret; | |
374 } | |
375 | |
376 #endif /* NAS_SOUND */ | |
377 | |
378 #endif /* !_WIN32 */ | |
379 | |
79 void play_file(char *filename) | 380 void play_file(char *filename) |
80 { | 381 { |
81 #ifndef _WIN32 | 382 #ifndef _WIN32 |
82 #ifdef USE_AO | 383 int pid; |
83 int ao_driver; | 384 #endif |
84 #endif /* USE_AO */ | |
85 pid_t pid; | |
86 #endif | |
87 | |
88 if (awaymessage && !(sound_options & OPT_SOUND_WHEN_AWAY)) | 385 if (awaymessage && !(sound_options & OPT_SOUND_WHEN_AWAY)) |
89 return; /* check here in case a buddy pounce plays a file while away */ | 386 return; /* check here in case a buddy pounce plays a file while away */ |
90 | 387 |
91 if (sound_options & OPT_SOUND_BEEP) { | 388 if (sound_options & OPT_SOUND_BEEP) { |
92 gdk_beep(); | 389 gdk_beep(); |
93 return; | 390 return; |
94 } | 391 } |
95 | 392 |
393 else if (sound_options & OPT_SOUND_NORMAL) { | |
394 debug_printf("attempting to play audio file with internal method -- this is unlikely to work\n"); | |
395 } | |
96 #ifndef _WIN32 | 396 #ifndef _WIN32 |
97 if (!g_file_test(filename, G_FILE_TEST_EXISTS)) { | |
98 gchar *tmp = g_strdup_printf(_("Unable to play sound because the chosen filename (%s) does not exist."), filename); | |
99 do_error_dialog(tmp, NULL, GAIM_ERROR); | |
100 g_free(tmp); | |
101 return; | |
102 } | |
103 | |
104 if (sound_options & OPT_SOUND_CMD) { | |
105 gchar *command = NULL; | |
106 GError *error = NULL; | |
107 | |
108 if (!sound_cmd[0]) { | |
109 do_error_dialog(_("Unable to play sound because the 'Command' sound method has been chosen, but no command has been set."), NULL, GAIM_ERROR); | |
110 return; | |
111 } | |
112 | |
113 command = g_strdup_printf(sound_cmd, filename); | |
114 | |
115 if (g_spawn_command_line_async(command, &error) == FALSE) { | |
116 gchar *tmp = g_strdup_printf(_("Unable to play sound because the configured sound command could not be launched: %s"), error->message); | |
117 do_error_dialog(tmp, NULL, GAIM_ERROR); | |
118 g_free(tmp); | |
119 g_error_free(error); | |
120 } | |
121 | |
122 g_free(command); | |
123 return; | |
124 } | |
125 | |
126 #ifdef USE_AO | |
127 if (!ao_initialized) { | |
128 ao_initialize(); | |
129 } | |
130 | |
131 ao_driver = ao_default_driver_id(); | |
132 | |
133 if (ao_driver == -1) { | |
134 do_error_dialog(_("Unable to play sound because no suitable driver could be found."), NULL, GAIM_ERROR); | |
135 return; | |
136 } | |
137 | |
138 pid = fork(); | 397 pid = fork(); |
139 | 398 |
140 if (pid < 0) | 399 if (pid < 0) |
141 return; | 400 return; |
142 else if (pid == 0) { | 401 else if (pid == 0) { |
143 AFfilehandle file = afOpenFile(filename, "rb", NULL); | 402 alarm(30); |
144 if(file) { | 403 |
145 ao_device *device; | 404 if ((sound_options & OPT_SOUND_CMD) && sound_cmd[0]) { |
146 ao_sample_format format; | 405 char *args[4]; |
147 | 406 char command[4096]; |
148 int in_fmt; | 407 |
149 int bytes_per_frame; | 408 g_snprintf(command, sizeof(command), sound_cmd, filename); |
150 | 409 |
151 format.rate = afGetRate(file, AF_DEFAULT_TRACK); | 410 args[0] = "sh"; |
152 format.channels = afGetChannels(file, AF_DEFAULT_TRACK); | 411 args[1] = "-c"; |
153 afGetSampleFormat(file, AF_DEFAULT_TRACK, &in_fmt, | 412 args[2] = command; |
154 &format.bits); | 413 args[3] = NULL; |
155 | 414 execvp(args[0], args); |
156 bytes_per_frame = format.bits * format.channels / 8; | 415 _exit(0); |
157 | 416 } |
158 device = ao_open_live(ao_driver, &format, NULL); | 417 #ifdef ESD_SOUND |
159 | 418 else if (sound_options & OPT_SOUND_ESD) { |
160 if (device) { | 419 if (esd_play_file(NULL, filename, 1)) |
161 int frames_read; | 420 _exit(0); |
162 char buf[4096]; | 421 } |
163 int buf_frames = sizeof(buf) / bytes_per_frame; | 422 #endif |
164 | 423 |
165 while((frames_read = afReadFrames(file, AF_DEFAULT_TRACK, | 424 #ifdef ARTSC_SOUND |
166 buf, buf_frames))) { | 425 else if (sound_options & OPT_SOUND_ARTSC) { |
167 if(!ao_play(device, buf, frames_read * bytes_per_frame)) | 426 if (artsc_play_file(filename)) |
168 break; | 427 _exit(0); |
169 } | 428 } |
170 ao_close(device); | 429 #endif |
171 } | 430 |
172 | 431 #ifdef NAS_SOUND |
173 ao_shutdown(); | 432 else if (sound_options & OPT_SOUND_NAS) { |
174 afCloseFile(file); | 433 if (play_nas_file(filename)) |
175 } | 434 _exit(0); |
435 } | |
436 #endif | |
437 | |
438 else if ((sound_options & OPT_SOUND_NORMAL) && | |
439 can_play_audio()) { | |
440 play_audio_file(filename); | |
441 _exit(0); | |
442 } | |
443 | |
176 _exit(0); | 444 _exit(0); |
177 } | 445 } |
178 #else /* USE_AO */ | |
179 gdk_beep(); | |
180 #endif /* USE_AO */ | |
181 | |
182 #else /* _WIN32 */ | 446 #else /* _WIN32 */ |
183 debug_printf("Playing %s\n", filename); | 447 debug_printf("Playing %s\n", filename); |
184 if (!PlaySound(filename, 0, SND_ASYNC | SND_FILENAME)) | 448 if (!PlaySound(filename, 0, SND_ASYNC | SND_FILENAME)) |
185 debug_printf("Error playing sound."); | 449 debug_printf("Error playing sound."); |
186 #endif | 450 #endif |
187 } | 451 } |
188 | 452 |
189 void sound_quit() { | |
190 #ifdef USE_AO | |
191 if (ao_initialized) { | |
192 ao_shutdown(); | |
193 } | |
194 #endif | |
195 } | |
196 | |
197 extern int logins_not_muted; | 453 extern int logins_not_muted; |
198 | 454 |
199 void play_sound(int sound) | 455 void play_sound(int sound) |
200 { | 456 { |
201 if (mute_sounds) | 457 if (mute_sounds) |
202 return; | 458 return; |
203 | 459 |
204 if ((sound == SND_BUDDY_ARRIVE) && !logins_not_muted) | 460 if ((sound == SND_BUDDY_ARRIVE) && !logins_not_muted) |
205 return; | 461 return; |
206 | 462 |
207 if (sound >= NUM_SOUNDS) { | 463 if (sound >= NUM_SOUNDS) { |
208 debug_printf("got request for unknown sound: %d\n", sound); | 464 debug_printf("got request for unknown sound: %d\n", sound); |