1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
|
/*
* BRLTTY - A background process providing access to the console screen (when in
* text mode) for a blind person using a refreshable braille display.
*
* Copyright (C) 1995-2014 by The BRLTTY Developers.
*
* BRLTTY comes with ABSOLUTELY NO WARRANTY.
*
* This is free software, placed under the terms of the
* GNU General Public License, as published by the Free Software
* Foundation; either version 2 of the License, or (at your option) any
* later version. Please see the file LICENSE-GPL for details.
*
* Web Page: http://mielke.cc/brltty/
*
* This software is maintained by Dave Mielke <dave@mielke.cc>.
*/
#include "prologue.h"
#include <string.h>
#include "log.h"
#include "parameters.h"
#include "async_alarm.h"
#include "program.h"
#include "tune.h"
#include "notes.h"
static const NoteMethods *noteMethods = NULL;
static NoteDevice *noteDevice = NULL;
static AsyncHandle tuneDeviceCloseTimer = NULL;
static int openErrorLevel = LOG_ERR;
void
suppressTuneDeviceOpenErrors (void) {
openErrorLevel = LOG_DEBUG;
}
void
closeTuneDevice (void) {
if (tuneDeviceCloseTimer) {
asyncCancelRequest(tuneDeviceCloseTimer);
tuneDeviceCloseTimer = NULL;
}
if (noteDevice) {
noteMethods->destruct(noteDevice);
noteDevice = NULL;
}
}
ASYNC_ALARM_CALLBACK(handleTuneDeviceCloseTimeout) {
if (tuneDeviceCloseTimer) {
asyncDiscardHandle(tuneDeviceCloseTimer);
tuneDeviceCloseTimer = NULL;
}
closeTuneDevice();
}
static int tuneInitialized = 0;
static void
exitTunes (void *data) {
closeTuneDevice();
tuneInitialized = 0;
}
static int
openTuneDevice (void) {
const int timeout = TUNE_DEVICE_CLOSE_DELAY;
if (noteDevice) {
asyncResetAlarmIn(tuneDeviceCloseTimer, timeout);
} else if ((noteDevice = noteMethods->construct(openErrorLevel)) != NULL) {
if (asyncSetAlarmIn(&tuneDeviceCloseTimer, timeout, handleTuneDeviceCloseTimeout, NULL)) {
if (!tuneInitialized) {
tuneInitialized = 1;
onProgramExit("tunes", exitTunes, NULL);
}
}
} else {
return 0;
}
return 1;
}
int
setTuneDevice (TuneDevice device) {
const NoteMethods *methods;
switch (device) {
default:
methods = NULL;
break;
#ifdef HAVE_BEEP_SUPPORT
case tdBeeper:
methods = &beepNoteMethods;
break;
#endif /* HAVE_BEEP_SUPPORT */
#ifdef HAVE_PCM_SUPPORT
case tdPcm:
methods = &pcmNoteMethods;
break;
#endif /* HAVE_PCM_SUPPORT */
#ifdef HAVE_MIDI_SUPPORT
case tdMidi:
methods = &midiNoteMethods;
break;
#endif /* HAVE_MIDI_SUPPORT */
#ifdef HAVE_FM_SUPPORT
case tdFm:
methods = &fmNoteMethods;
break;
#endif /* HAVE_FM_SUPPORT */
}
if (!methods) return 0;
if (methods != noteMethods) {
closeTuneDevice();
noteMethods = methods;
}
return 1;
}
int
playTune (const TuneElement *tune) {
while (tune->duration) {
if (!openTuneDevice()) return 0;
if (!noteMethods->play(noteDevice, tune->note, tune->duration)) return 0;
tune += 1;
}
return noteMethods->flush(noteDevice);
}
|