/* Copyright (C) 2006 Paul Davis Author: David Robillard 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifndef __ardour_smf_source_h__ #define __ardour_smf_source_h__ #include #include #include "evoral/SMF.hpp" #include "ardour/midi_source.h" #include "ardour/file_source.h" namespace Evoral { template class Event; } namespace ARDOUR { template class MidiRingBuffer; /** Standard Midi File (Type 0) Source */ class SMFSource : public MidiSource, public FileSource, public Evoral::SMF { public: /** Constructor for new internal-to-session files */ SMFSource (Session& session, const std::string& path, Source::Flag flags); /** Constructor for existing external-to-session files */ SMFSource (Session& session, const std::string& path); /** Constructor for existing in-session files */ SMFSource (Session& session, const XMLNode&, bool must_exist = false); virtual ~SMFSource (); /** Rename the file on disk referenced by this source to \param newname * * This method exists only for MIDI file sources, not for audio, which * can never be renamed. It exists for MIDI so that we can get * consistent and sane region/source numbering when regions are added * manually (which never happens with audio). */ int rename (const std::string& name); bool safe_file_extension (const std::string& path) const { return safe_midi_file_extension(path); } void append_event_unlocked_beats (const Evoral::Event& ev); void append_event_unlocked_frames (const Evoral::Event& ev, framepos_t source_start); void mark_streaming_midi_write_started (NoteMode mode); void mark_streaming_write_completed (); void mark_midi_streaming_write_completed (Evoral::Sequence::StuckNoteOption, Evoral::MusicalTime when = 0); XMLNode& get_state (); int set_state (const XMLNode&, int version); void load_model (bool lock=true, bool force_reload=false); void destroy_model (); void flush_midi (); void ensure_disk_file (); static bool safe_midi_file_extension (const std::string& path); void prevent_deletion (); protected: void set_path (const std::string& newpath); private: int open_for_write (); framecnt_t read_unlocked (Evoral::EventSink& dst, framepos_t position, framepos_t start, framecnt_t cnt, MidiStateTracker* tracker) const; framecnt_t write_unlocked (MidiRingBuffer& src, framepos_t position, framecnt_t cnt); double _last_ev_time_beats; framepos_t _last_ev_time_frames; /** end time (start + duration) of last call to read_unlocked */ mutable framepos_t _smf_last_read_end; /** time (in SMF ticks, 1 tick per _ppqn) of the last event read by read_unlocked */ mutable framepos_t _smf_last_read_time; }; }; /* namespace ARDOUR */ #endif /* __ardour_smf_source_h__ */