648 lines
19 KiB
C++
648 lines
19 KiB
C++
/*
|
|
* Copyright (C) 2006-2016 David Robillard <d@drobilla.net>
|
|
* Copyright (C) 2007-2017 Paul Davis <paul@linuxaudiosystems.com>
|
|
* Copyright (C) 2008 Hans Baier <hansfbaier@googlemail.com>
|
|
* Copyright (C) 2009-2012 Carl Hetherington <carl@carlh.net>
|
|
* Copyright (C) 2012-2015 Tim Mayberry <mojofunk@gmail.com>
|
|
* Copyright (C) 2016-2017 Nick Mainsbridge <mainsbridge@gmail.com>
|
|
* Copyright (C) 2016-2017 Robin Gareus <robin@gareus.org>
|
|
* Copyright (C) 2016 Julien "_FrnchFrgg_" RIVAUD <frnchfrgg@free.fr>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <cmath>
|
|
#include <climits>
|
|
#include <cfloat>
|
|
|
|
#include <set>
|
|
|
|
#include <glibmm/threads.h>
|
|
#include <glibmm/fileutils.h>
|
|
#include <glibmm/miscutils.h>
|
|
|
|
#include "temporal/beats.h"
|
|
|
|
#include "pbd/xml++.h"
|
|
#include "pbd/basename.h"
|
|
#include "pbd/types_convert.h"
|
|
|
|
#include "ardour/automation_control.h"
|
|
#include "ardour/midi_cursor.h"
|
|
#include "ardour/midi_model.h"
|
|
#include "ardour/midi_region.h"
|
|
#include "ardour/midi_ring_buffer.h"
|
|
#include "ardour/midi_source.h"
|
|
#include "ardour/playlist.h"
|
|
#include "ardour/region_factory.h"
|
|
#include "ardour/session.h"
|
|
#include "ardour/source_factory.h"
|
|
#include "ardour/tempo.h"
|
|
#include "ardour/thawlist.h"
|
|
#include "ardour/types.h"
|
|
#include "ardour/evoral_types_convert.h"
|
|
|
|
#include "pbd/i18n.h"
|
|
#include <locale.h>
|
|
|
|
using namespace std;
|
|
using namespace ARDOUR;
|
|
using namespace PBD;
|
|
|
|
namespace PBD {
|
|
DEFINE_ENUM_CONVERT(ARDOUR::Source::Flag);
|
|
}
|
|
|
|
/* Basic MidiRegion constructor (many channels) */
|
|
MidiRegion::MidiRegion (const SourceList& srcs)
|
|
: Region (srcs)
|
|
, _ignore_shift (false)
|
|
{
|
|
/* by default MIDI regions are transparent,
|
|
* this should probably be set depending on use-case,
|
|
* (eg. loop recording, vs copy/edit/paste)
|
|
*/
|
|
override_opaqueness (false);
|
|
|
|
midi_source(0)->ModelChanged.connect_same_thread (_source_connection, boost::bind (&MidiRegion::model_changed, this));
|
|
model_changed ();
|
|
assert(_name.val().find("/") == string::npos);
|
|
assert(_type == DataType::MIDI);
|
|
}
|
|
|
|
MidiRegion::MidiRegion (std::shared_ptr<const MidiRegion> other)
|
|
: Region (other)
|
|
, _ignore_shift (false)
|
|
{
|
|
assert(_name.val().find("/") == string::npos);
|
|
midi_source(0)->ModelChanged.connect_same_thread (_source_connection, boost::bind (&MidiRegion::model_changed, this));
|
|
model_changed ();
|
|
}
|
|
|
|
/** Create a new MidiRegion that is part of an existing one */
|
|
MidiRegion::MidiRegion (std::shared_ptr<const MidiRegion> other, timecnt_t const & offset)
|
|
: Region (other, offset)
|
|
, _ignore_shift (false)
|
|
{
|
|
|
|
assert(_name.val().find("/") == string::npos);
|
|
midi_source(0)->ModelChanged.connect_same_thread (_source_connection, boost::bind (&MidiRegion::model_changed, this));
|
|
model_changed ();
|
|
}
|
|
|
|
MidiRegion::~MidiRegion ()
|
|
{
|
|
}
|
|
|
|
/** Export the MIDI data of the MidiRegion to a new MIDI file (SMF).
|
|
*/
|
|
bool
|
|
MidiRegion::do_export (string const& path) const
|
|
{
|
|
std::shared_ptr<MidiSource> newsrc;
|
|
|
|
/* caller must check for pre-existing file */
|
|
assert (!path.empty());
|
|
assert (!Glib::file_test (path, Glib::FILE_TEST_EXISTS));
|
|
newsrc = std::dynamic_pointer_cast<MidiSource> (SourceFactory::createWritable (DataType::MIDI, _session, path, _session.sample_rate (), false, true));
|
|
|
|
{
|
|
/* Lock our source since we'll be reading from it. write_to() will
|
|
* take a lock on newsrc.
|
|
*/
|
|
Source::ReaderLock lm (midi_source(0)->mutex());
|
|
if (midi_source(0)->export_write_to (lm, newsrc, _start.val().beats(), _start.val().beats() + _length.val().beats())) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/** Create a new MidiRegion that has its own version of some/all of the Source used by another.
|
|
*/
|
|
std::shared_ptr<MidiRegion>
|
|
MidiRegion::clone (string path) const
|
|
{
|
|
std::shared_ptr<MidiSource> newsrc;
|
|
|
|
/* caller must check for pre-existing file */
|
|
assert (!path.empty());
|
|
assert (!Glib::file_test (path, Glib::FILE_TEST_EXISTS));
|
|
newsrc = std::dynamic_pointer_cast<MidiSource>(
|
|
SourceFactory::createWritable(DataType::MIDI, _session, path, _session.sample_rate()));
|
|
return clone (newsrc);
|
|
}
|
|
|
|
std::shared_ptr<MidiRegion>
|
|
MidiRegion::clone (std::shared_ptr<MidiSource> newsrc, ThawList* tl) const
|
|
{
|
|
{
|
|
std::shared_ptr<MidiSource> ms = midi_source(0);
|
|
|
|
/* copy source state (cue markers, captured_for, CC/param interpolation */
|
|
XMLNode& node (ms->get_state());
|
|
/* these must be set, otherwise Source::set_state() returns early */
|
|
node.set_property (X_("id"), newsrc->id());
|
|
node.set_property (X_("name"), newsrc->name());
|
|
node.set_property (X_("flags"), newsrc->flags ());
|
|
node.set_property (X_("take-id"), newsrc->take_id());
|
|
|
|
/* Lock our source since we'll be reading from it. write_to() will
|
|
take a lock on newsrc.
|
|
*/
|
|
|
|
Source::ReaderLock lm (ms->mutex());
|
|
if (ms->write_to (lm, newsrc, Temporal::Beats(), std::numeric_limits<Temporal::Beats>::max())) {
|
|
delete &node;
|
|
return std::shared_ptr<MidiRegion> ();
|
|
}
|
|
|
|
/* compare to SMFSource::set_state */
|
|
newsrc->MidiSource::set_state (node, Stateful::current_state_version);
|
|
newsrc->Source::set_state (node, Stateful::current_state_version);
|
|
delete &node;
|
|
}
|
|
|
|
PropertyList plist (derive_properties ());
|
|
|
|
plist.add (Properties::name, PBD::basename_nosuffix (newsrc->name()));
|
|
plist.add (Properties::whole_file, true);
|
|
plist.add (Properties::external, false);
|
|
plist.add (Properties::import, false);
|
|
plist.add (Properties::layer, 0);
|
|
|
|
std::shared_ptr<MidiRegion> ret (std::dynamic_pointer_cast<MidiRegion> (RegionFactory::create (newsrc, plist, true, tl)));
|
|
|
|
return ret;
|
|
}
|
|
|
|
timecnt_t
|
|
MidiRegion::read_at (Evoral::EventSink<samplepos_t>& out,
|
|
timepos_t const & position,
|
|
timecnt_t const & dur,
|
|
Temporal::Range* loop_range,
|
|
MidiCursor& cursor,
|
|
uint32_t chan_n,
|
|
NoteMode mode,
|
|
MidiNoteTracker* tracker,
|
|
MidiChannelFilter* filter) const
|
|
{
|
|
return _read_at (_sources, out, position, dur, loop_range, cursor, chan_n, mode, tracker, filter);
|
|
}
|
|
|
|
timecnt_t
|
|
MidiRegion::master_read_at (MidiRingBuffer<samplepos_t>& out,
|
|
timepos_t const & position,
|
|
timecnt_t const & dur,
|
|
Temporal::Range* loop_range,
|
|
MidiCursor& cursor,
|
|
uint32_t chan_n,
|
|
NoteMode mode) const
|
|
{
|
|
return _read_at (_master_sources, out, position, dur, loop_range, cursor, chan_n, mode); /* no tracker */
|
|
}
|
|
|
|
timecnt_t
|
|
MidiRegion::_read_at (const SourceList& /*srcs*/,
|
|
Evoral::EventSink<samplepos_t>& dst,
|
|
timepos_t const & pos,
|
|
timecnt_t const & xdur,
|
|
Temporal::Range* loop_range,
|
|
MidiCursor& cursor,
|
|
uint32_t chan_n,
|
|
NoteMode mode,
|
|
MidiNoteTracker* tracker,
|
|
MidiChannelFilter* filter) const
|
|
{
|
|
timecnt_t dur (xdur);
|
|
timecnt_t internal_offset;
|
|
timecnt_t to_read;
|
|
|
|
/* precondition: caller has verified that we cover the desired section */
|
|
|
|
assert(chan_n == 0);
|
|
|
|
if (muted()) {
|
|
return timecnt_t(); /* read nothing */
|
|
}
|
|
|
|
if (pos < position()) {
|
|
/* we are starting the read from before the start of the region */
|
|
internal_offset = timecnt_t (Temporal::BeatTime);;
|
|
dur -= pos.distance (position());
|
|
} else {
|
|
/* we are starting the read from after the start of the region */
|
|
internal_offset = position().distance (pos);
|
|
}
|
|
|
|
if (internal_offset >= _length) {
|
|
return timecnt_t(); /* read nothing */
|
|
}
|
|
|
|
if ((to_read = min (dur, _length - internal_offset)) == 0) {
|
|
return timecnt_t(); /* read nothing */
|
|
}
|
|
|
|
std::shared_ptr<MidiSource> src = midi_source(chan_n);
|
|
|
|
Source::ReaderLock lm (src->mutex());
|
|
|
|
#if 0
|
|
cerr << "MR " << name () << " read @ " << pos << " + " << to_read
|
|
<< " dur was " << dur
|
|
<< " len " << _length
|
|
<< " l-io " << (_length - internal_offset)
|
|
<< " position = " << position()
|
|
<< " _start = " << _start
|
|
<< " intoffset = " << internal_offset
|
|
<< " quarter_note = " << quarter_note()
|
|
<< " start_beat = " << _start_beats
|
|
<< endl;
|
|
#endif
|
|
|
|
/* This call reads events from a source and writes them to `dst' timed in session samples */
|
|
|
|
if (src->midi_read (
|
|
lm, // source lock
|
|
dst, // destination buffer
|
|
position().earlier (_start.val()), // start position of the source on timeline
|
|
_start.val() + internal_offset, // where to start reading in the source
|
|
to_read, // read duration in samples
|
|
loop_range,
|
|
cursor,
|
|
tracker,
|
|
filter,
|
|
_filtered_parameters
|
|
) != to_read) {
|
|
return timecnt_t(); /* "read nothing" */
|
|
}
|
|
|
|
return to_read;
|
|
}
|
|
|
|
int
|
|
MidiRegion::render (Evoral::EventSink<samplepos_t>& dst,
|
|
uint32_t chan_n,
|
|
NoteMode mode,
|
|
MidiChannelFilter* filter) const
|
|
{
|
|
return render_range (dst, chan_n, mode, this->start(), _length, filter);
|
|
}
|
|
|
|
int
|
|
MidiRegion::render_range (Evoral::EventSink<samplepos_t>& dst,
|
|
uint32_t chan_n,
|
|
NoteMode mode,
|
|
timepos_t const & read_start,
|
|
timecnt_t const & read_length,
|
|
MidiChannelFilter* filter) const
|
|
{
|
|
/* precondition: caller has verified that we cover the desired section */
|
|
|
|
assert(chan_n == 0);
|
|
|
|
if (muted()) {
|
|
return 0; /* read nothing */
|
|
}
|
|
|
|
std::shared_ptr<MidiSource> src = midi_source(chan_n);
|
|
|
|
|
|
#if 0
|
|
cerr << "MR " << name () << " render "
|
|
<< " _position = " << position ()
|
|
<< " _start = " << _start
|
|
<< " _length = " << _length
|
|
<< " read_start = " << read_start
|
|
<< " read_length = " << read_length
|
|
<< endl;
|
|
#endif
|
|
|
|
MidiCursor cursor;
|
|
MidiNoteTracker tracker;
|
|
|
|
/* This call reads events from a source and writes them to `dst' timed in session samples */
|
|
|
|
Source::ReaderLock lm (src->mutex());
|
|
|
|
src->midi_read (
|
|
lm, // source lock
|
|
dst, // destination buffer
|
|
source_position(), // start position of the source in session samples
|
|
read_start,
|
|
read_length,
|
|
0,
|
|
cursor,
|
|
&tracker,
|
|
filter,
|
|
_filtered_parameters);
|
|
|
|
/* resolve any notes that were "cut off" by the end of the region. The
|
|
* Note-Off's get inserted at the end of the region
|
|
*/
|
|
|
|
const timepos_t end = source_position() + read_start + read_length;
|
|
tracker.resolve_notes (dst, end.samples());
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
XMLNode&
|
|
MidiRegion::state () const
|
|
{
|
|
return Region::state ();
|
|
}
|
|
|
|
int
|
|
MidiRegion::set_state (const XMLNode& node, int version)
|
|
{
|
|
int ret = Region::set_state (node, version);
|
|
|
|
if (version < 7001) {
|
|
/* previous sessions had uncontionally transparent MIDI regions */
|
|
override_opaqueness (false);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
MidiRegion::recompute_at_end ()
|
|
{
|
|
/* our length has changed
|
|
* so what? stuck notes are dealt with via a note state tracker
|
|
*/
|
|
}
|
|
|
|
void
|
|
MidiRegion::recompute_at_start ()
|
|
{
|
|
/* as above, but the shift was from the front
|
|
* maybe bump currently active note's note-ons up so they sound here?
|
|
* that could be undesireable in certain situations though.. maybe
|
|
* remove the note entirely, including it's note off? something needs to
|
|
* be done to keep the played MIDI sane to avoid messing up voices of
|
|
* polyhonic things etc........
|
|
*/
|
|
}
|
|
|
|
int
|
|
MidiRegion::separate_by_channel (vector< std::shared_ptr<Region> >&) const
|
|
{
|
|
// TODO
|
|
return -1;
|
|
}
|
|
|
|
std::shared_ptr<Evoral::Control>
|
|
MidiRegion::control (const Evoral::Parameter& id, bool create)
|
|
{
|
|
return model()->control(id, create);
|
|
}
|
|
|
|
std::shared_ptr<const Evoral::Control>
|
|
MidiRegion::control (const Evoral::Parameter& id) const
|
|
{
|
|
return model()->control(id);
|
|
}
|
|
|
|
std::shared_ptr<MidiModel>
|
|
MidiRegion::model()
|
|
{
|
|
return midi_source()->model();
|
|
}
|
|
|
|
std::shared_ptr<const MidiModel>
|
|
MidiRegion::model() const
|
|
{
|
|
return midi_source()->model();
|
|
}
|
|
|
|
std::shared_ptr<MidiSource>
|
|
MidiRegion::midi_source (uint32_t n) const
|
|
{
|
|
// Guaranteed to succeed (use a static cast?)
|
|
return std::dynamic_pointer_cast<MidiSource>(source(n));
|
|
}
|
|
|
|
void
|
|
MidiRegion::model_changed ()
|
|
{
|
|
if (!model()) {
|
|
return;
|
|
}
|
|
|
|
/* build list of filtered Parameters, being those whose automation state is not `Play' */
|
|
|
|
_filtered_parameters.clear ();
|
|
|
|
Automatable::Controls const & c = model()->controls();
|
|
|
|
for (Automatable::Controls::const_iterator i = c.begin(); i != c.end(); ++i) {
|
|
std::shared_ptr<AutomationControl> ac = std::dynamic_pointer_cast<AutomationControl> (i->second);
|
|
assert (ac);
|
|
if (ac->alist()->automation_state() != Play) {
|
|
_filtered_parameters.insert (ac->parameter ());
|
|
}
|
|
}
|
|
|
|
/* watch for changes to controls' AutoState */
|
|
midi_source()->AutomationStateChanged.connect_same_thread (
|
|
_model_connection, boost::bind (&MidiRegion::model_automation_state_changed, this, _1)
|
|
);
|
|
|
|
model()->ContentsShifted.connect_same_thread (_model_shift_connection, boost::bind (&MidiRegion::model_shifted, this, _1));
|
|
model()->ContentsChanged.connect_same_thread (_model_changed_connection, boost::bind (&MidiRegion::model_contents_changed, this));
|
|
}
|
|
|
|
void
|
|
MidiRegion::model_contents_changed ()
|
|
{
|
|
send_change (Properties::contents);
|
|
}
|
|
|
|
void
|
|
MidiRegion::model_shifted (timecnt_t distance)
|
|
{
|
|
if (!model()) {
|
|
return;
|
|
}
|
|
|
|
if (!_ignore_shift) {
|
|
PropertyChange what_changed;
|
|
/* _start is a Property, so we cannot call timepos_t methods on
|
|
it directly. ::val() only provides a const, so use
|
|
operator+() rather than operator+=()
|
|
*/
|
|
_start = _start.val() + distance;
|
|
what_changed.add (Properties::start);
|
|
what_changed.add (Properties::contents);
|
|
send_change (what_changed);
|
|
} else {
|
|
_ignore_shift = false;
|
|
}
|
|
}
|
|
|
|
void
|
|
MidiRegion::model_automation_state_changed (Evoral::Parameter const & p)
|
|
{
|
|
/* Update our filtered parameters list after a change to a parameter's AutoState */
|
|
|
|
std::shared_ptr<AutomationControl> ac = model()->automation_control (p);
|
|
if (!ac || ac->alist()->automation_state() == Play) {
|
|
/* It should be "impossible" for ac to be NULL, but if it is, don't
|
|
filter the parameter so events aren't lost. */
|
|
_filtered_parameters.erase (p);
|
|
} else {
|
|
_filtered_parameters.insert (p);
|
|
}
|
|
|
|
/* the source will have an iterator into the model, and that iterator will have been set up
|
|
for a given set of filtered_parameters, so now that we've changed that list we must invalidate
|
|
the iterator.
|
|
*/
|
|
Source::WriterLock lm (midi_source(0)->mutex(), Glib::Threads::TRY_LOCK);
|
|
if (lm.locked()) {
|
|
/* TODO: This is too aggressive, we need more fine-grained invalidation. */
|
|
midi_source(0)->invalidate (lm);
|
|
}
|
|
}
|
|
|
|
/** This is called when a trim drag has resulted in a -ve _start time for this region.
|
|
* Fix it up by adding some empty space to the source.
|
|
*/
|
|
void
|
|
MidiRegion::fix_negative_start ()
|
|
{
|
|
_ignore_shift = true;
|
|
|
|
model()->insert_silence_at_start (-start().beats());
|
|
|
|
_start = timepos_t::zero (start().time_domain());
|
|
}
|
|
|
|
bool
|
|
MidiRegion::set_name (const std::string& str)
|
|
{
|
|
if (_name == str) {
|
|
return true;
|
|
}
|
|
|
|
if (!Session::session_name_is_legal (str).empty ()) {
|
|
return false;
|
|
}
|
|
|
|
return Region::set_name (str);
|
|
}
|
|
|
|
void
|
|
MidiRegion::merge (std::shared_ptr<MidiRegion const> other_region)
|
|
{
|
|
std::shared_ptr<MidiModel const> self = model ();
|
|
std::shared_ptr<MidiModel const> other = other_region->model();
|
|
|
|
Temporal::Beats other_region_start (other_region->start().beats());
|
|
Temporal::Beats other_region_end ((other_region->start() + other_region->length()).beats());
|
|
|
|
Evoral::Sequence<Temporal::Beats>::const_iterator e1 = self->begin();
|
|
Evoral::Sequence<Temporal::Beats>::const_iterator e2 = other->begin();
|
|
|
|
Source::WriterLock sl (midi_source()->mutex ());
|
|
midi_source ()->drop_model (sl);
|
|
std::shared_ptr<MidiModel> new_model (new MidiModel (*midi_source()));
|
|
new_model->start_write ();
|
|
|
|
/* Note: merge() is called with sorted region list so that
|
|
* self->position <= other->position ()
|
|
*/
|
|
|
|
assert (position() <= other_region->position ());
|
|
|
|
while ((e1 != self->end ()) || (e2 != other->end ())) {
|
|
/* map time from other region to this region */
|
|
Temporal::Beats e2t;
|
|
if (e2 != other->end ()) {
|
|
timepos_t abs_time (other_region->source_beats_to_absolute_time (e2->time()));
|
|
e2t = position().distance (abs_time).beats();
|
|
}
|
|
|
|
if (e2 == other->end () || (e1 != self->end () && e1->time() < e2t)) {
|
|
Evoral::Event<Temporal::Beats> ev (*e1, true);
|
|
new_model->append (ev, Evoral::next_event_id());
|
|
++e1;
|
|
continue;
|
|
}
|
|
|
|
if (e1 == self->end () || (e2 != other->end () && e1->time() >= e2t)) {
|
|
if (e2->time() < other_region_start) {
|
|
++e2;
|
|
continue;
|
|
}
|
|
if (e2->time() > other_region_end) {
|
|
/* No more events form other region */
|
|
e2 = other->end ();
|
|
continue;
|
|
}
|
|
|
|
Evoral::Event<Temporal::Beats> ev (*e2, true);
|
|
ev.set_time (e2t);
|
|
new_model->append (ev, Evoral::next_event_id());
|
|
++e2;
|
|
continue;
|
|
}
|
|
|
|
assert (0);
|
|
break;
|
|
}
|
|
|
|
new_model->end_write (Evoral::Sequence<Temporal::Beats>::ResolveStuckNotes, length().beats());
|
|
midi_source()->set_model (sl, new_model);
|
|
|
|
set_length (max (length(), position().distance (other_region->end())));
|
|
}
|
|
|
|
void
|
|
MidiRegion::start_domain_bounce (Temporal::DomainBounceInfo& cmd)
|
|
{
|
|
/* Deal with the region position & length */
|
|
|
|
Region::start_domain_bounce (cmd);
|
|
if (cmd.from != Temporal::BeatTime) {
|
|
return;
|
|
}
|
|
|
|
model()->start_domain_bounce (cmd);
|
|
model()->create_mapping_stash (source_position().beats());
|
|
}
|
|
|
|
void
|
|
MidiRegion::finish_domain_bounce (Temporal::DomainBounceInfo& cmd)
|
|
{
|
|
Region::finish_domain_bounce (cmd);
|
|
|
|
if (cmd.from != Temporal::BeatTime) {
|
|
return;
|
|
}
|
|
|
|
model()->rebuild_from_mapping_stash (source_position().beats());
|
|
model()->finish_domain_bounce (cmd);
|
|
|
|
_model_changed_connection.disconnect ();
|
|
model()->ContentsChanged ();
|
|
model()->ContentsChanged.connect_same_thread (_model_changed_connection, boost::bind (&MidiRegion::model_contents_changed, this));
|
|
}
|