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
23 #include <sys/ioctl.h>
24 #include <sys/types.h>
37 #include <sys/audioio.h>
40 static const ALCchar solaris_device
[] = "Solaris Default";
42 static const char *solaris_driver
= "/dev/audio";
54 static ALuint
SolarisProc(ALvoid
*ptr
)
56 ALCdevice
*Device
= (ALCdevice
*)ptr
;
57 solaris_data
*data
= (solaris_data
*)Device
->ExtraData
;
63 frameSize
= FrameSizeFromDevFmt(Device
->FmtChans
, Device
->FmtType
);
65 while(!data
->killNow
&& Device
->Connected
)
67 ALint len
= data
->data_size
;
68 ALubyte
*WritePtr
= data
->mix_data
;
70 aluMixData(Device
, WritePtr
, len
/frameSize
);
71 while(len
> 0 && !data
->killNow
)
73 wrote
= write(data
->fd
, WritePtr
, len
);
76 if(errno
!= EAGAIN
&& errno
!= EWOULDBLOCK
&& errno
!= EINTR
)
78 ERR("write failed: %s\n", strerror(errno
));
79 aluHandleDisconnect(Device
);
96 static ALCenum
solaris_open_playback(ALCdevice
*device
, const ALCchar
*deviceName
)
101 deviceName
= solaris_device
;
102 else if(strcmp(deviceName
, solaris_device
) != 0)
103 return ALC_INVALID_VALUE
;
105 data
= (solaris_data
*)calloc(1, sizeof(solaris_data
));
108 data
->fd
= open(solaris_driver
, O_WRONLY
);
112 ERR("Could not open %s: %s\n", solaris_driver
, strerror(errno
));
113 return ALC_INVALID_VALUE
;
116 device
->DeviceName
= strdup(deviceName
);
117 device
->ExtraData
= data
;
121 static void solaris_close_playback(ALCdevice
*device
)
123 solaris_data
*data
= (solaris_data
*)device
->ExtraData
;
127 device
->ExtraData
= NULL
;
130 static ALCboolean
solaris_reset_playback(ALCdevice
*device
)
132 solaris_data
*data
= (solaris_data
*)device
->ExtraData
;
137 AUDIO_INITINFO(&info
);
139 info
.play
.sample_rate
= device
->Frequency
;
141 if(device
->FmtChans
!= DevFmtMono
)
142 device
->FmtChans
= DevFmtStereo
;
143 numChannels
= ChannelsFromDevFmt(device
->FmtChans
);
144 info
.play
.channels
= numChannels
;
146 switch(device
->FmtType
)
149 info
.play
.precision
= 8;
150 info
.play
.encoding
= AUDIO_ENCODING_LINEAR
;
153 info
.play
.precision
= 8;
154 info
.play
.encoding
= AUDIO_ENCODING_LINEAR8
;
160 device
->FmtType
= DevFmtShort
;
163 info
.play
.precision
= 16;
164 info
.play
.encoding
= AUDIO_ENCODING_LINEAR
;
168 frameSize
= numChannels
* BytesFromDevFmt(device
->FmtType
);
169 info
.play
.buffer_size
= device
->UpdateSize
*device
->NumUpdates
* frameSize
;
171 if(ioctl(data
->fd
, AUDIO_SETINFO
, &info
) < 0)
173 ERR("ioctl failed: %s\n", strerror(errno
));
177 if(ChannelsFromDevFmt(device
->FmtChans
) != info
.play
.channels
)
179 ERR("Could not set %d channels, got %d instead\n", ChannelsFromDevFmt(device
->FmtChans
), info
.play
.channels
);
183 if(!((info
.play
.precision
== 8 && info
.play
.encoding
== AUDIO_ENCODING_LINEAR8
&& device
->FmtType
== DevFmtUByte
) ||
184 (info
.play
.precision
== 8 && info
.play
.encoding
== AUDIO_ENCODING_LINEAR
&& device
->FmtType
== DevFmtByte
) ||
185 (info
.play
.precision
== 16 && info
.play
.encoding
== AUDIO_ENCODING_LINEAR
&& device
->FmtType
== DevFmtShort
) ||
186 (info
.play
.precision
== 32 && info
.play
.encoding
== AUDIO_ENCODING_LINEAR
&& device
->FmtType
== DevFmtInt
)))
188 ERR("Could not set %s samples, got %d (0x%x)\n", DevFmtTypeString(device
->FmtType
),
189 info
.play
.precision
, info
.play
.encoding
);
193 device
->Frequency
= info
.play
.sample_rate
;
194 device
->UpdateSize
= (info
.play
.buffer_size
/device
->NumUpdates
) + 1;
196 SetDefaultChannelOrder(device
);
201 static ALCboolean
solaris_start_playback(ALCdevice
*device
)
203 solaris_data
*data
= (solaris_data
*)device
->ExtraData
;
205 data
->data_size
= device
->UpdateSize
* FrameSizeFromDevFmt(device
->FmtChans
, device
->FmtType
);
206 data
->mix_data
= calloc(1, data
->data_size
);
208 data
->thread
= StartThread(SolarisProc
, device
);
209 if(data
->thread
== NULL
)
211 free(data
->mix_data
);
212 data
->mix_data
= NULL
;
219 static void solaris_stop_playback(ALCdevice
*device
)
221 solaris_data
*data
= (solaris_data
*)device
->ExtraData
;
227 StopThread(data
->thread
);
231 if(ioctl(data
->fd
, AUDIO_DRAIN
) < 0)
232 ERR("Error draining device: %s\n", strerror(errno
));
234 free(data
->mix_data
);
235 data
->mix_data
= NULL
;
239 static const BackendFuncs solaris_funcs
= {
240 solaris_open_playback
,
241 solaris_close_playback
,
242 solaris_reset_playback
,
243 solaris_start_playback
,
244 solaris_stop_playback
,
251 ALCdevice_LockDefault
,
252 ALCdevice_UnlockDefault
,
253 ALCdevice_GetLatencyDefault
256 ALCboolean
alc_solaris_init(BackendFuncs
*func_list
)
258 ConfigValueStr("solaris", "device", &solaris_driver
);
260 *func_list
= solaris_funcs
;
264 void alc_solaris_deinit(void)
268 void alc_solaris_probe(enum DevProbe type
)
272 case ALL_DEVICE_PROBE
:
276 if(stat(solaris_driver
, &buf
) == 0)
278 AppendAllDevicesList(solaris_device
);
282 case CAPTURE_DEVICE_PROBE
: