chiark / gitweb /
Install disorderd under launchd in Mac OS X.
[disorder] / driver / disorder.c
CommitLineData
460b9539 1
2/*
3 * This file is part of DisOrder.
4 * Copyright (C) 2005 Richard Kettlewell
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19 * USA
20 */
21
22#include <config.h>
23
24#include <string.h>
25#include <stdlib.h>
26#include <errno.h>
27#include <unistd.h>
28#include <poll.h>
29#include <ao/ao.h>
30#include <ao/plugin.h>
31
32/* extra declarations to help out lazy <ao/plugin.h> */
33int ao_plugin_test(void);
34ao_info *ao_plugin_driver_info(void);
35char *ao_plugin_file_extension(void);
36
37/* private data structure for this driver */
38struct internal {
39 int fd; /* output file descriptor */
40 int exit_on_error; /* exit on write error */
41};
42
43/* like write() but never returns EINTR/EAGAIN or short */
44static int do_write(int fd, const void *ptr, size_t n) {
45 size_t written = 0;
46 int ret;
47 struct pollfd ufd;
48
49 memset(&ufd, 0, sizeof ufd);
50 ufd.fd = fd;
51 ufd.events = POLLOUT;
52 while(written < n) {
53 ret = write(fd, (const char *)ptr + written, n - written);
54 if(ret < 0) {
55 switch(errno) {
56 case EINTR: break;
57 case EAGAIN:
58 /* Someone sneakily gave us a nonblocking file descriptor, wait until
59 * we can write again */
60 ret = poll(&ufd, 1, -1);
61 if(ret < 0 && errno != EINTR) return -1;
62 break;
63 default:
64 return -1;
65 }
66 } else
67 written += ret;
68 }
69 return written;
70}
71
72/* return 1 if this driver can be opened */
73int ao_plugin_test(void) {
74 return 1;
75}
76
77/* return info about this driver */
78ao_info *ao_plugin_driver_info(void) {
79 static const char *options[] = { "fd" };
80 static const ao_info info = {
81 AO_TYPE_LIVE, /* type */
82 (char *)"DisOrder format driver", /* name */
83 (char *)"disorder", /* short_name */
84 (char *)"http://www.greenend.org.uk/rjk/disorder/", /* comment */
85 (char *)"Richard Kettlewell", /* author */
86 AO_FMT_NATIVE, /* preferred_byte_format */
87 0, /* priority */
88 (char **)options, /* options */
89 1, /* option_count */
90 };
91 return (ao_info *)&info;
92}
93
94/* initialize the private data structure */
95int ao_plugin_device_init(ao_device *device) {
96 struct internal *i = malloc(sizeof (struct internal));
97 const char *e;
98
99 if(!i) return 0;
100 memset(i, 0, sizeof *i);
101 if((e = getenv("DISORDER_RAW_FD")))
102 i->fd = atoi(e);
103 else
104 i->fd = 1;
105 device->internal = i;
106 return 1;
107}
108
109/* set an option */
110int ao_plugin_set_option(ao_device *device,
111 const char *key,
112 const char *value) {
113 struct internal *i = device->internal;
114
115 if(!strcmp(key, "fd"))
116 i->fd = atoi(value);
117 else if(!strcmp(key, "fragile"))
118 i->exit_on_error = atoi(value);
119 /* unknown options are required to be ignored */
120 return 1;
121}
122
123/* open the device */
124int ao_plugin_open(ao_device *device, ao_sample_format *format) {
125 struct internal *i = device->internal;
126
127 /* we would like native-order samples */
128 device->driver_byte_format = AO_FMT_NATIVE;
129 if(do_write(i->fd, format, sizeof *format) < 0) {
130 if(i->exit_on_error) exit(-1);
131 return 0;
132 }
133 return 1;
134}
135
136/* play some samples */
137int ao_plugin_play(ao_device *device, const char *output_samples,
138 uint_32 num_bytes) {
139 struct internal *i = device->internal;
140
141 if(do_write(i->fd, output_samples, num_bytes) < 0) {
142 if(i->exit_on_error) _exit(-1);
143 return 0;
144 }
145 return 1;
146}
147
148/* close the device */
149int ao_plugin_close(ao_device attribute((unused)) *device) {
150 return 1;
151}
152
153/* delete private data structures */
154void ao_plugin_device_clear(ao_device *device) {
155 free(device->internal);
156 device->internal = 0;
157}
158
159/* report preferred filename extension */
160char *ao_plugin_file_extension(void) {
161 return 0;
162}
163
164/*
165Local Variables:
166c-basic-offset:2
167comment-column:40
168End:
169*/