/* * Copyright (C) 2006 Stas Sergeev * * The below copyright strings have to be distributed unchanged together * with this file. This prefix can not be modified or separated. */ /* * This program is free software; you can redistribute it and/or modify * it 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. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "emu.h" #include "dosemu_config.h" #include "init.h" #include "sound/midi.h" #include #include #include #include #include #include static int pipe_fd = -1; #define midopipe_name "pipe_midi" #define midopipe_longname "MIDI Output: named pipe" static int midopipe_init(void *arg) { const char *name = DOSEMU_MIDI_PATH; if (mkfifo(name, 0666) == -1) { S_printf("Unable to mkfifo() '%s'\n", strerror(errno)); return 0; } pipe_fd = RPT_SYSCALL(open(name, O_WRONLY | O_NONBLOCK)); if (pipe_fd == -1) { int err = errno; S_printf("%s: unable to open %s for writing (%s)%s\n", midopipe_name, name, strerror(errno), errno == ENXIO ? ", will continue trying" : ""); if (err == ENXIO) { /* no FIFO readers */ return 1; } else { /* some other problem */ return 0; } } /* open ok */ return 1; } static void midopipe_done(void *arg) { if (pipe_fd == -1) return; close(pipe_fd); pipe_fd = -1; } static void midopipe_write(unsigned char val) { /* Try again to open FIFO on each write in case some readers showed up. */ if (pipe_fd == -1) { pipe_fd = RPT_SYSCALL(open(DOSEMU_MIDI_PATH, O_WRONLY | O_CREAT | O_NONBLOCK, 0666)); if (pipe_fd == -1) { return; } } if (write(pipe_fd, &val, 1) == -1) { error("MIDI: Error writing to %s, resetting: %s\n", midopipe_name, strerror(errno)); close(pipe_fd); pipe_fd = -1; } } static int midopipe_cfg(void *arg) { return pcm_parse_cfg(config.midi_driver, midopipe_name); } static const struct midi_out_plugin midopipe #ifdef __cplusplus { midopipe_name, midopipe_longname, midopipe_cfg, midopipe_init, midopipe_done, 0, midopipe_write, NULL, NULL, ST_ANY, PCM_F_PASSTHRU | PCM_F_EXPLICIT, }; #else = { .name = midopipe_name, .longname = midopipe_longname, .get_cfg = midopipe_cfg, .open = midopipe_init, .close = midopipe_done, .write = midopipe_write, .stype = ST_ANY, .flags = PCM_F_PASSTHRU | PCM_F_EXPLICIT, }; #endif CONSTRUCTOR(static void midopipe_register(void)) { midi_register_output_plugin(&midopipe); }