2 * OpenAL cross platform audio library
3 * Copyright (C) 1999-2007 by authors.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
18 * Or go to http://www.gnu.org/copyleft/lgpl.html
33 static const ALCchar sndio_device
[] = "SndIO Default";
36 static ALCboolean
sndio_load(void)
43 struct sio_hdl
*sndHandle
;
53 static ALuint
sndio_proc(ALvoid
*ptr
)
55 ALCdevice
*device
= ptr
;
56 sndio_data
*data
= device
->ExtraData
;
62 frameSize
= FrameSizeFromDevFmt(device
->FmtChans
, device
->FmtType
);
64 while(!data
->killNow
&& device
->Connected
)
66 ALsizei len
= data
->data_size
;
67 ALubyte
*WritePtr
= data
->mix_data
;
69 aluMixData(device
, WritePtr
, len
/frameSize
);
70 while(len
> 0 && !data
->killNow
)
72 wrote
= sio_write(data
->sndHandle
, WritePtr
, len
);
75 ERR("sio_write failed\n");
76 ALCdevice_Lock(device
);
77 aluHandleDisconnect(device
);
78 ALCdevice_Unlock(device
);
92 static ALCenum
sndio_open_playback(ALCdevice
*device
, const ALCchar
*deviceName
)
97 deviceName
= sndio_device
;
98 else if(strcmp(deviceName
, sndio_device
) != 0)
99 return ALC_INVALID_VALUE
;
101 data
= calloc(1, sizeof(*data
));
104 data
->sndHandle
= sio_open(NULL
, SIO_PLAY
, 0);
105 if(data
->sndHandle
== NULL
)
108 ERR("Could not open device\n");
109 return ALC_INVALID_VALUE
;
112 device
->DeviceName
= strdup(deviceName
);
113 device
->ExtraData
= data
;
118 static void sndio_close_playback(ALCdevice
*device
)
120 sndio_data
*data
= device
->ExtraData
;
122 sio_close(data
->sndHandle
);
124 device
->ExtraData
= NULL
;
127 static ALCboolean
sndio_reset_playback(ALCdevice
*device
)
129 sndio_data
*data
= device
->ExtraData
;
134 par
.rate
= device
->Frequency
;
135 par
.pchan
= ((device
->FmtChans
!= DevFmtMono
) ? 2 : 1);
137 switch(device
->FmtType
)
165 par
.le
= SIO_LE_NATIVE
;
167 par
.round
= device
->UpdateSize
;
168 par
.appbufsz
= device
->UpdateSize
* (device
->NumUpdates
-1);
169 if(!par
.appbufsz
) par
.appbufsz
= device
->UpdateSize
;
171 if(!sio_setpar(data
->sndHandle
, &par
) || !sio_getpar(data
->sndHandle
, &par
))
173 ERR("Failed to set device parameters\n");
177 if(par
.bits
!= par
.bps
*8)
179 ERR("Padded samples not supported (%u of %u bits)\n", par
.bits
, par
.bps
*8);
183 device
->Frequency
= par
.rate
;
184 device
->FmtChans
= ((par
.pchan
==1) ? DevFmtMono
: DevFmtStereo
);
186 if(par
.bits
== 8 && par
.sig
== 1)
187 device
->FmtType
= DevFmtByte
;
188 else if(par
.bits
== 8 && par
.sig
== 0)
189 device
->FmtType
= DevFmtUByte
;
190 else if(par
.bits
== 16 && par
.sig
== 1)
191 device
->FmtType
= DevFmtShort
;
192 else if(par
.bits
== 16 && par
.sig
== 0)
193 device
->FmtType
= DevFmtUShort
;
194 else if(par
.bits
== 32 && par
.sig
== 1)
195 device
->FmtType
= DevFmtInt
;
196 else if(par
.bits
== 32 && par
.sig
== 0)
197 device
->FmtType
= DevFmtUInt
;
200 ERR("Unhandled sample format: %s %u-bit\n", (par
.sig
?"signed":"unsigned"), par
.bits
);
204 device
->UpdateSize
= par
.round
;
205 device
->NumUpdates
= (par
.bufsz
/par
.round
) + 1;
207 SetDefaultChannelOrder(device
);
212 static ALCboolean
sndio_start_playback(ALCdevice
*device
)
214 sndio_data
*data
= device
->ExtraData
;
216 if(!sio_start(data
->sndHandle
))
218 ERR("Error starting playback\n");
222 data
->data_size
= device
->UpdateSize
* FrameSizeFromDevFmt(device
->FmtChans
, device
->FmtType
);
223 data
->mix_data
= calloc(1, data
->data_size
);
225 data
->thread
= StartThread(sndio_proc
, device
);
226 if(data
->thread
== NULL
)
228 sio_stop(data
->sndHandle
);
229 free(data
->mix_data
);
230 data
->mix_data
= NULL
;
237 static void sndio_stop_playback(ALCdevice
*device
)
239 sndio_data
*data
= device
->ExtraData
;
245 StopThread(data
->thread
);
249 if(!sio_stop(data
->sndHandle
))
250 ERR("Error stopping device\n");
252 free(data
->mix_data
);
253 data
->mix_data
= NULL
;
257 static const BackendFuncs sndio_funcs
= {
259 sndio_close_playback
,
260 sndio_reset_playback
,
261 sndio_start_playback
,
269 ALCdevice_LockDefault
,
270 ALCdevice_UnlockDefault
,
271 ALCdevice_GetLatencyDefault
274 ALCboolean
alc_sndio_init(BackendFuncs
*func_list
)
278 *func_list
= sndio_funcs
;
282 void alc_sndio_deinit(void)
286 void alc_sndio_probe(enum DevProbe type
)
290 case ALL_DEVICE_PROBE
:
291 AppendAllDevicesList(sndio_device
);
293 case CAPTURE_DEVICE_PROBE
: