static std::string s_instance_name;
size_t AlsaAudioBackend::_max_buffer_size = 8192;
std::vector<std::string> AlsaAudioBackend::_midi_options;
+std::vector<AudioBackend::DeviceStatus> AlsaAudioBackend::_audio_device_status;
+std::vector<AudioBackend::DeviceStatus> AlsaAudioBackend::_midi_device_status;
AlsaAudioBackend::AlsaAudioBackend (AudioEngine& e, AudioBackendInfo& info)
: AudioBackend (e, info)
std::vector<AudioBackend::DeviceStatus>
AlsaAudioBackend::enumerate_devices () const
{
- std::vector<AudioBackend::DeviceStatus> s;
+ _audio_device_status.clear();
std::map<std::string, std::string> devices;
get_alsa_audio_device_names(devices);
for (std::map<std::string, std::string>::const_iterator i = devices.begin (); i != devices.end(); ++i) {
- s.push_back (DeviceStatus (i->first, true));
+ if (_audio_device == "") _audio_device = i->first;
+ _audio_device_status.push_back (DeviceStatus (i->first, true));
}
- return s;
+ return _audio_device_status;
}
void
_reservation_succeeded = false;
std::string request_device_exe;
- if (!PBD::find_file_in_search_path (
+ if (!PBD::find_file (
PBD::Searchpath(Glib::build_filename(ARDOUR::ardour_dll_directory(), "ardouralsautil")
+ G_SEARCHPATH_SEPARATOR_S + ARDOUR::ardour_dll_directory()),
"ardour-request-device", request_device_exe))
std::vector<AudioBackend::DeviceStatus>
AlsaAudioBackend::enumerate_midi_devices () const
{
- std::vector<AudioBackend::DeviceStatus> s;
+ _midi_device_status.clear();
std::map<std::string, std::string> devices;
if (_midi_driver_option == _("ALSA raw devices")) {
}
else if (_midi_driver_option == _("ALSA sequencer")) {
get_alsa_sequencer_names (devices);
- } else {
- return s;
}
for (std::map<std::string, std::string>::const_iterator i = devices.begin (); i != devices.end(); ++i) {
- s.push_back (DeviceStatus (i->first, true));
+ _midi_device_status.push_back (DeviceStatus (i->first, true));
}
- return s;
+ return _midi_device_status;
}
int
delete m;
}
- unregister_system_ports();
+ unregister_ports();
delete _pcmi; _pcmi = 0;
release_device();
bool
AlsaAudioBackend::in_process_thread ()
{
+ if (pthread_equal (_main_thread, pthread_self()) != 0) {
+ return true;
+ }
+
for (std::vector<pthread_t>::const_iterator i = _threads.begin (); i != _threads.end (); ++i)
{
if (pthread_equal (*i, pthread_self ()) != 0) {
void
AlsaAudioBackend::unregister_port (PortEngine::PortHandle port_handle)
{
- if (!valid_port (port_handle)) {
- PBD::error << _("AlsaBackend::unregister_port: Invalid Port.") << endmsg;
+ if (!_run) {
+ return;
}
AlsaPort* port = static_cast<AlsaPort*>(port_handle);
std::vector<AlsaPort*>::iterator i = std::find (_ports.begin (), _ports.end (), static_cast<AlsaPort*>(port_handle));
}
void
-AlsaAudioBackend::unregister_system_ports()
+AlsaAudioBackend::unregister_ports (bool system_only)
{
size_t i = 0;
_system_inputs.clear();
_system_midi_out.clear();
while (i < _ports.size ()) {
AlsaPort* port = _ports[i];
- if (port->is_physical () && port->is_terminal ()) {
+ if (! system_only || (port->is_physical () && port->is_terminal ())) {
port->disconnect_all ();
+ delete port;
_ports.erase (_ports.begin() + i);
} else {
++i;
_processed_samples = 0;
uint64_t clock1, clock2;
- clock1 = g_get_monotonic_time();
_pcmi->pcm_start ();
int no_proc_errors = 0;
const int bailout = 2 * _samplerate / _samples_per_period;
}
_pcmi->capt_done (_samples_per_period);
- /* de-queue midi*/
+ /* de-queue incoming midi*/
i = 0;
for (std::vector<AlsaPort*>::const_iterator it = _system_midi_in.begin (); it != _system_midi_in.end (); ++it, ++i) {
assert (_rmidi_in.size() > i);
return 0;
}
- i = 0;
- for (std::vector<AlsaPort*>::iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it, ++i) {
+ for (std::vector<AlsaPort*>::iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it) {
static_cast<AlsaMidiPort*>(*it)->next_period();
}
- /* queue midi */
+ /* queue outgoing midi */
i = 0;
for (std::vector<AlsaPort*>::const_iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it, ++i) {
assert (_rmidi_out.size() > i);
}
} else {
// Freewheelin'
+
+ // zero audio input buffers
for (std::vector<AlsaPort*>::const_iterator it = _system_inputs.begin (); it != _system_inputs.end (); ++it) {
memset ((*it)->get_buffer (_samples_per_period), 0, _samples_per_period * sizeof (Sample));
}
- for (std::vector<AlsaPort*>::const_iterator it = _system_midi_in.begin (); it != _system_midi_in.end (); ++it) {
+
+ clock1 = g_get_monotonic_time();
+ uint32_t i = 0;
+ for (std::vector<AlsaPort*>::const_iterator it = _system_midi_in.begin (); it != _system_midi_in.end (); ++it, ++i) {
static_cast<AlsaMidiBuffer*>((*it)->get_buffer(0))->clear ();
+ AlsaMidiIn *rm = _rmidi_in.at(i);
+ void *bptr = (*it)->get_buffer(0);
+ midi_clear(bptr); // zero midi buffer
+
+ // TODO add an API call for this.
+ pframes_t time;
+ uint8_t data[64]; // match MaxAlsaEventSize in alsa_rawmidi.cc
+ size_t size = sizeof(data);
+ while (rm->recv_event (time, data, size)) {
+ ; // discard midi-data from HW.
+ }
+ rm->sync_time (clock1);
}
if (engine.process_callback (_samples_per_period)) {
_pcmi->pcm_stop ();
+ _active = false;
return 0;
}
+
+ // drop all outgoing MIDI messages
+ for (std::vector<AlsaPort*>::const_iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it) {
+ void *bptr = (*it)->get_buffer(0);
+ midi_clear(bptr);
+ }
+
_dsp_load = 1.0;
Glib::usleep (100); // don't hog cpu
}
static int instantiate (const std::string& arg1, const std::string& /* arg2 */);
static int deinstantiate ();
static bool already_configured ();
+static bool available ();
static ARDOUR::AudioBackendInfo _descriptor = {
"ALSA",
deinstantiate,
backend_factory,
already_configured,
+ available
};
static boost::shared_ptr<AudioBackend>
return false;
}
+static bool
+available ()
+{
+ return true;
+}
+
extern "C" ARDOURBACKEND_API ARDOUR::AudioBackendInfo* descriptor ()
{
return &_descriptor;