#include namespace ruby { #include "ao.hpp" class pAudioAO { public: AudioAO &self; int driver_id; ao_sample_format driver_format; ao_device *audio_device; struct { unsigned frequency; } settings; bool cap(Audio::Setting setting) { if(setting == Audio::Frequency) return true; return false; } uintptr_t get(Audio::Setting setting) { if(setting == Audio::Frequency) return settings.frequency; return false; } bool set(Audio::Setting setting, uintptr_t param) { if(setting == Audio::Frequency) { settings.frequency = param; if(audio_device) { term(); init(); } return true; } return false; } void sample(uint16_t l_sample, uint16_t r_sample) { uint32_t samp = (l_sample << 0) + (r_sample << 16); ao_play(audio_device, (char*)&samp, 4); //This may need to be byte swapped for Big Endian } bool init() { driver_id = ao_default_driver_id(); //ao_driver_id((const char*)driver) if(driver_id < 0) return false; driver_format.bits = 16; driver_format.channels = 2; driver_format.rate = settings.frequency; driver_format.byte_format = AO_FMT_LITTLE; ao_option *options = 0; ao_info *di = ao_driver_info(driver_id); if(!di) return false; if(!strcmp(di->short_name, "alsa")) { ao_append_option(&options, "buffer_time", "100000"); //100ms latency (default was 500ms) } audio_device = ao_open_live(driver_id, &driver_format, options); if(!audio_device) return false; return true; } void term() { if(audio_device) { ao_close(audio_device); audio_device = 0; } } pAudioAO(AudioAO &self_) : self(self_) { audio_device = 0; ao_initialize(); settings.frequency = 22050; } ~pAudioAO() { term(); //ao_shutdown(); //FIXME: this is causing a segfault for some reason when called ... } }; bool AudioAO::cap(Setting setting) { return p.cap(setting); } uintptr_t AudioAO::get(Setting setting) { return p.get(setting); } bool AudioAO::set(Setting setting, uintptr_t param) { return p.set(setting, param); } void AudioAO::sample(uint16_t l_sample, uint16_t r_sample) { p.sample(l_sample, r_sample); } bool AudioAO::init() { return p.init(); } void AudioAO::term() { p.term(); } AudioAO::AudioAO() : p(*new pAudioAO(*this)) {} AudioAO::~AudioAO() { delete &p; } } //namespace ruby