/*
This file is part of Ingen.
Copyright 2007-2016 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen 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 Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#ifndef INGEN_ENGINE_JACKAUDIODRIVER_HPP
#define INGEN_ENGINE_JACKAUDIODRIVER_HPP
#include "Driver.hpp"
#include "EnginePort.hpp"
#include "types.hpp"
#include
#include
#include
#include
#include
#include
#include // IWYU pragma: keep
#include
#include
#include
#include
#include
#include
#include
namespace raul {
class Path;
} // namespace raul
namespace ingen {
class Atom;
namespace server {
class Buffer;
class DuplexPort;
class Engine;
class FrameTimer;
class RunContext;
/** The Jack Driver.
*
* The process callback here drives the entire audio thread by "pulling"
* events from queues, processing them, running the graphs, and passing
* events along to the PostProcessor.
*
* \ingroup engine
*/
class JackDriver : public Driver
{
public:
explicit JackDriver(Engine& engine);
~JackDriver() override;
bool attach(const std::string& server_name,
const std::string& client_name,
void* jack_client);
bool activate() override;
void deactivate() override;
bool dynamic_ports() const override { return true; }
EnginePort* create_port(DuplexPort* graph_port) override;
EnginePort* get_port(const raul::Path& path) override;
void rename_port(const raul::Path& old_path, const raul::Path& new_path) override;
void port_property(const raul::Path& path, const URI& uri, const Atom& value) override;
void add_port(RunContext& ctx, EnginePort* port) override;
void remove_port(RunContext& ctx, EnginePort* port) override;
void register_port(EnginePort& port) override;
void unregister_port(EnginePort& port) override;
/** Transport state for this frame.
* Intended to only be called from the audio thread. */
const jack_position_t* position() { return &_position; }
jack_transport_state_t transport_state() { return _transport_state; }
void append_time_events(RunContext& ctx, Buffer& buffer) override;
int real_time_priority() override {
return jack_client_real_time_priority(_client);
}
jack_client_t* jack_client() const { return _client; }
SampleCount block_length() const override { return _block_length; }
uint32_t seq_size() const override { return _seq_size; }
SampleCount sample_rate() const override { return _sample_rate; }
SampleCount frame_time() const override;
class PortRegistrationFailedException : public std::exception
{};
private:
friend class JackPort;
static void thread_init_cb(void* jack_driver);
// Static JACK callbacks which call the non-static callbacks (methods)
static void shutdown_cb(void* const jack_driver) {
static_cast(jack_driver)->_shutdown_cb();
}
static int process_cb(jack_nframes_t nframes, void* const jack_driver) {
return static_cast(jack_driver)->_process_cb(nframes);
}
static int block_length_cb(jack_nframes_t nframes, void* const jack_driver) {
return static_cast(jack_driver)->_block_length_cb(nframes);
}
// Internal methods for processing
void pre_process_port(RunContext& ctx, EnginePort* port);
void post_process_port(RunContext& ctx, EnginePort* port) const;
void port_property_internal(const jack_port_t* jport,
const URI& uri,
const Atom& value);
// Non static callbacks (methods)
void _shutdown_cb();
int _process_cb(jack_nframes_t nframes);
int _block_length_cb(jack_nframes_t nframes);
protected:
using Ports = boost::intrusive::slist>;
using AudioBufPtr = std::unique_ptr>;
Engine& _engine;
Ports _ports;
AudioBufPtr _fallback_buffer;
LV2_Atom_Forge _forge;
raul::Semaphore _sem{0};
std::unique_ptr _timer;
std::atomic _flag{false};
jack_client_t* _client{nullptr};
jack_nframes_t _block_length{0};
uint32_t _seq_size{0};
jack_nframes_t _sample_rate{0};
uint32_t _midi_event_type;
bool _is_activated{false};
jack_position_t _position{};
jack_transport_state_t _transport_state{};
double _old_bpm{120.0};
jack_nframes_t _old_frame{0};
bool _old_rolling{false};
};
} // namespace server
} // namespace ingen
#endif // INGEN_ENGINE_JACKAUDIODRIVER_HPP