/* ------------------------------------------------------------------ * Copyright (C) 2011 Martin Storsjo * Copyright (C) 2017 Matthias P. Braendli * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either * express or implied. * See the License for the specific language governing permissions * and limitations under the License. * ------------------------------------------------------------------- */ /*! \file AlsaInput.h * * This input uses libasound to get audio data. */ #pragma once #include "config.h" #if HAVE_ALSA #include <cstdio> #include <string> #include <thread> #include <atomic> #include <alsa/asoundlib.h> #include "SampleQueue.h" #include "common.h" #include "InputInterface.h" /*! Common functionality for the direct alsa input and the * threaded alsa input. The threaded one is used for * drift compensation. */ class AlsaInput : public InputInterface { public: AlsaInput(const std::string& alsa_dev, unsigned int channels, unsigned int rate, SampleQueue<uint8_t>& queue) : m_alsa_dev(alsa_dev), m_channels(channels), m_rate(rate), m_queue(queue) { } AlsaInput(const AlsaInput& other) = delete; AlsaInput& operator=(const AlsaInput& other) = delete; virtual ~AlsaInput(); protected: /* Read from the ALSA device. Returns number of samples, * or -1 in case of error */ ssize_t m_read(uint8_t* buf, snd_pcm_uframes_t length); /* Open the ALSA device and set it up */ void m_init_alsa(void); std::string m_alsa_dev; unsigned int m_channels; unsigned int m_rate; SampleQueue<uint8_t>& m_queue; snd_pcm_t *m_alsa_handle = nullptr; }; class AlsaInputDirect : public AlsaInput { public: AlsaInputDirect(const std::string& alsa_dev, unsigned int channels, unsigned int rate, SampleQueue<uint8_t>& queue) : AlsaInput(alsa_dev, channels, rate, queue) { } virtual void prepare(void) override; virtual bool fault_detected(void) const override { return false; }; virtual bool read_source(size_t num_bytes) override; /*! Read length Bytes from from the alsa device. * length must be a multiple of channels * bytes_per_sample. * * \return the number of bytes read. */ ssize_t read(uint8_t* buf, size_t length); }; class AlsaInputThreaded : public AlsaInput { public: AlsaInputThreaded(const std::string& alsa_dev, unsigned int channels, unsigned int rate, SampleQueue<uint8_t>& queue) : AlsaInput(alsa_dev, channels, rate, queue), m_fault(false), m_running(false) { } virtual ~AlsaInputThreaded(); /*! Start the ALSA thread that fills the queue */ virtual void prepare(void) override; virtual bool fault_detected(void) const override { return m_fault; }; virtual bool read_source(size_t num_bytes) override; private: void process(); std::atomic<bool> m_fault; std::atomic<bool> m_running; std::thread m_thread; }; #endif // HAVE_ALSA