|
|
|
/*
|
|
|
|
|
|
|
|
Copyright (C) 2001 Carsten Kroll
|
|
|
|
ckroll@pinnaclesys.com
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This library 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
|
|
|
|
Library General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* only compile 'sgi' AudioIO class if compiled under IRIX
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_IRIX
|
|
|
|
|
|
|
|
#include <dmedia/audio.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_SELECT_H
|
|
|
|
#include <sys/select.h> // Needed on some systems.
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <iostream>
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "debug.h"
|
|
|
|
#include "audioio.h"
|
|
|
|
|
|
|
|
namespace Arts {
|
|
|
|
|
|
|
|
class AudioIOSGI : public AudioIO {
|
|
|
|
protected:
|
|
|
|
int requestedFragmentSize;
|
|
|
|
int requestedFragmentCount;
|
|
|
|
ALport audio_port,audio_port1;
|
|
|
|
ALconfig audioconfig;
|
|
|
|
int framesz;
|
|
|
|
|
|
|
|
public:
|
|
|
|
AudioIOSGI();
|
|
|
|
|
|
|
|
void setParam(AudioParam param, int& value);
|
|
|
|
int getParam(AudioParam param);
|
|
|
|
|
|
|
|
bool open();
|
|
|
|
void close();
|
|
|
|
int read(void *buffer, int size);
|
|
|
|
int write(void *buffer, int size);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
REGISTER_AUDIO_IO(AudioIOSGI,"sgi","SGI dmedia Audio I/O");
|
|
|
|
};
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace Arts;
|
|
|
|
|
|
|
|
AudioIOSGI::AudioIOSGI()
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* default parameters
|
|
|
|
*/
|
|
|
|
param(samplingRate) = 44100;
|
|
|
|
paramStr(deviceName) = "audioio";
|
|
|
|
requestedFragmentSize = param(fragmentSize) = 4096;
|
|
|
|
requestedFragmentCount = param(fragmentCount) = 10;
|
|
|
|
audio_port=0;
|
|
|
|
audio_port1=0;
|
|
|
|
param(format)=17;
|
|
|
|
param(channels) = 2;
|
|
|
|
param(direction) = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AudioIOSGI::open()
|
|
|
|
{
|
|
|
|
string& _error = paramStr(lastError);
|
|
|
|
string& _deviceName = paramStr(deviceName);
|
|
|
|
int& _channels = param(channels);
|
|
|
|
int& _fragmentSize = param(fragmentSize);
|
|
|
|
int& _fragmentCount = param(fragmentCount);
|
|
|
|
int& _samplingRate = param(samplingRate);
|
|
|
|
int& _format = param(format);
|
|
|
|
int& _direction = param(direction);
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if(_direction != 3 && _direction != 2){
|
|
|
|
_error = "invalid direction";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
framesz=((_format & ~1) >> 3) * _channels;
|
|
|
|
|
|
|
|
audioconfig = alNewConfig();
|
|
|
|
alSetSampFmt(audioconfig,AL_SAMPFMT_TWOSCOMP);
|
|
|
|
|
|
|
|
alSetWidth(audioconfig, _format==8 ? AL_SAMPLE_8 : _format==16 || _format==17 ? AL_SAMPLE_16 : AL_SAMPLE_24);
|
|
|
|
alSetQueueSize(audioconfig,(requestedFragmentSize * requestedFragmentCount) / framesz);
|
|
|
|
alSetChannels(audioconfig,_channels);
|
|
|
|
|
|
|
|
audio_port = alOpenPort("out","w",audioconfig);
|
|
|
|
|
|
|
|
|
|
|
|
if (audio_port == (ALport) 0 ) {
|
|
|
|
err = oserror();
|
|
|
|
if (err == AL_BAD_NO_PORTS) {
|
|
|
|
_error = "System is out of audio ports";
|
|
|
|
} else if (err == AL_BAD_DEVICE_ACCESS) {
|
|
|
|
_error = "Couldn't access audio device";
|
|
|
|
} else if (err == AL_BAD_OUT_OF_MEM) {
|
|
|
|
_error = "Out of memory";
|
|
|
|
}
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_direction == 3){
|
|
|
|
audio_port1 = alOpenPort("in","r",audioconfig);
|
|
|
|
if (audio_port1 == (ALport) 0 ) {
|
|
|
|
err = oserror();
|
|
|
|
if (err == AL_BAD_NO_PORTS) {
|
|
|
|
_error = "System is out of audio ports";
|
|
|
|
} else if (err == AL_BAD_DEVICE_ACCESS) {
|
|
|
|
_error = "Couldn't access audio device";
|
|
|
|
} else if (err == AL_BAD_OUT_OF_MEM) {
|
|
|
|
_error = "Out of memory";
|
|
|
|
}
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Attempt to set a crystal-based sample-rate on the
|
|
|
|
* given device.
|
|
|
|
*/
|
|
|
|
ALpv x[2];
|
|
|
|
x[0].param = AL_MASTER_CLOCK;
|
|
|
|
x[0].value.i = AL_CRYSTAL_MCLK_TYPE;
|
|
|
|
x[1].param = AL_RATE;
|
|
|
|
x[1].value.ll = alDoubleToFixed(double(_samplingRate));
|
|
|
|
if (alSetParams(alGetResource(audio_port),x, 2)<0) {
|
|
|
|
_error="setparams failed: ";
|
|
|
|
_error+=alGetErrorString(oserror());
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_direction == 3)
|
|
|
|
if (alSetParams(alGetResource(audio_port1),x, 2)<0) {
|
|
|
|
_error="setparams failed: ";
|
|
|
|
_error+=alGetErrorString(oserror());
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (x[1].sizeOut < 0) {
|
|
|
|
_error="rate was invalid";
|
|
|
|
close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
alSetFillPoint(audio_port,(alGetQueueSize(audioconfig)*5)/10) ;//50 %
|
|
|
|
if (_direction == 3)
|
|
|
|
alSetFillPoint(audio_port1,(alGetQueueSize(audioconfig)*4)/10) ;//40 %
|
|
|
|
/*
|
|
|
|
* set the fragment settings to what the user requested
|
|
|
|
*/
|
|
|
|
|
|
|
|
_fragmentSize = requestedFragmentSize;
|
|
|
|
_fragmentCount = requestedFragmentCount;
|
|
|
|
|
|
|
|
|
|
|
|
artsdebug("buffering: %d fragments with %d bytes "
|
|
|
|
"(audio latency is %1.1f ms)", _fragmentCount, _fragmentSize,
|
|
|
|
(float)(_fragmentSize*_fragmentCount) /
|
|
|
|
(float)(2.0 * _samplingRate * _channels)*1000.0);
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AudioIOSGI::close()
|
|
|
|
{
|
|
|
|
alFreeConfig(audioconfig);
|
|
|
|
alClosePort(audio_port);
|
|
|
|
audio_port=0;
|
|
|
|
if (param(direction) == 3) {
|
|
|
|
alClosePort(audio_port1);
|
|
|
|
audio_port1=0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AudioIOSGI::setParam(AudioParam p, int& value)
|
|
|
|
{
|
|
|
|
switch(p)
|
|
|
|
{
|
|
|
|
case fragmentSize:
|
|
|
|
param(p) = requestedFragmentSize = value;
|
|
|
|
break;
|
|
|
|
case fragmentCount:
|
|
|
|
param(p) = requestedFragmentCount = value;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
param(p) = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int AudioIOSGI::getParam(AudioParam p)
|
|
|
|
{
|
|
|
|
int frames;
|
|
|
|
switch(p)
|
|
|
|
{
|
|
|
|
case canRead:
|
|
|
|
frames=alGetFilled(audio_port);
|
|
|
|
return frames*framesz;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case canWrite:
|
|
|
|
frames=alGetFillable(audio_port);
|
|
|
|
return frames*framesz;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case selectReadFD:
|
|
|
|
return (param(direction) & directionRead)?
|
|
|
|
alGetFD(audio_port1):-1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case selectWriteFD:
|
|
|
|
return (param(direction) & directionWrite)?
|
|
|
|
alGetFD(audio_port):-1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return param(p);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int AudioIOSGI::read(void *buffer, int size)
|
|
|
|
{
|
|
|
|
arts_assert(audio_port1 != 0);
|
|
|
|
::alReadFrames(audio_port1,buffer,size/framesz);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
int AudioIOSGI::write(void *buffer, int size)
|
|
|
|
{
|
|
|
|
arts_assert(audio_port != 0);
|
|
|
|
::alWriteFrames(audio_port,buffer,size/framesz);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|