X-Git-Url: https://main.carlh.net/gitweb/?a=blobdiff_plain;ds=inline;f=libs%2Fmidi%2B%2B2%2Fmidi%2B%2B%2Fport.h;h=4e63d41141653f8e976e1acbf679f6e54eb06869;hb=4a71e6f28fed69ce8f82e98430180fdbf2da1684;hp=74ed206932728b47f1f6f0e015d03f38c5d48b16;hpb=449aab3c465bbbf66d221fac3d7ea559f1720357;p=ardour.git diff --git a/libs/midi++2/midi++/port.h b/libs/midi++2/midi++/port.h index 74ed206932..4e63d41141 100644 --- a/libs/midi++2/midi++/port.h +++ b/libs/midi++2/midi++/port.h @@ -1,5 +1,5 @@ /* - Copyright (C) 1998-99 Paul Barton-Davis + Copyright (C) 1998-2010 Paul Barton-Davis 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 @@ -16,56 +16,46 @@ */ -#ifndef __libmidi_port_h__ -#define __libmidi_port_h__ +#ifndef __libmidi_port_base_h__ +#define __libmidi_port_base_h__ #include #include -#include -#include +#include "pbd/xml++.h" +#include "pbd/crossthread.h" +#include "pbd/signals.h" +#include "pbd/ringbuffer.h" -#include -#include +#include "midi++/types.h" +#include "midi++/parser.h" namespace MIDI { class Channel; class PortRequest; -class Port : public sigc::trackable { +class Port { public: - enum Type { - Unknown, - JACK_Midi, - ALSA_RawMidi, - ALSA_Sequencer, - CoreMidi_MidiPort, - Null, - FIFO + enum Flags { + IsInput = JackPortIsInput, + IsOutput = JackPortIsOutput, }; - - + + Port (std::string const &, Flags); Port (const XMLNode&); virtual ~Port (); virtual XMLNode& get_state () const; virtual void set_state (const XMLNode&); - // FIXME: make Manager a friend of port so these can be hidden? - - /* Only for use by MidiManager. Don't ever call this. */ - virtual void cycle_start(nframes_t nframes); - /* Only for use by MidiManager. Don't ever call this. */ - virtual void cycle_end(); - /** Write a message to port. * @param msg Raw MIDI message to send * @param msglen Size of @a msg * @param timestamp Time stamp in frames of this message (relative to cycle start) * @return number of bytes successfully written */ - virtual int write (byte *msg, size_t msglen, timestamp_t timestamp) = 0; + virtual int write (const byte *msg, size_t msglen, timestamp_t timestamp) = 0; /** Read raw bytes from a port. * @param buf memory to store read data in @@ -74,8 +64,14 @@ class Port : public sigc::trackable { */ virtual int read (byte *buf, size_t bufsize) = 0; - void parse (); - + /** block until the output FIFO used by non-process threads + * is empty, checking every @a check_interval_usecs usecs + * for current status. Not to be called by a thread that + * executes any part of a JACK process callback (will + * simply return immediately in that situation). + */ + virtual void drain (int /* check_interval_usecs */) {} + /** Write a message to port. * @return true on success. * FIXME: describe semantics here @@ -84,100 +80,58 @@ class Port : public sigc::trackable { return !(write (msg, len, timestamp) == (int) len); } - int three_byte_msg (byte a, byte b, byte c, timestamp_t timestamp) { - byte msg[3]; - - msg[0] = a; - msg[1] = b; - msg[2] = c; + virtual void parse (framecnt_t timestamp) = 0; - return !(write (msg, 3, timestamp) == 3); - } - bool clock (timestamp_t timestamp); - - /* slowdown i/o to a loop of single byte emissions - interspersed with a busy loop of 10000 * this value. - - This may be ignored by a particular instance - of this virtual class. See FD_MidiPort for an - example of where it used. - */ - - void set_slowdown (size_t n) { slowdown = n; } /* select(2)/poll(2)-based I/O */ /** Get the file descriptor for port. * @return File descriptor, or -1 if not selectable. */ - virtual int selectable() const = 0; + virtual int selectable () const = 0; - static void gtk_read_callback (void *ptr, int fd, int cond); - static void write_callback (byte *msg, unsigned int len, void *); - Channel *channel (channel_t chn) { return _channel[chn&0x7F]; } - Parser *input() { return input_parser; } - Parser *output() { return output_parser; } - - void iostat (int *written, int *read, - const size_t **in_counts, - const size_t **out_counts) { - - *written = bytes_written; - *read = bytes_read; - if (input_parser) { - *in_counts = input_parser->message_counts(); - } else { - *in_counts = 0; - } - if (output_parser) { - *out_counts = output_parser->message_counts(); - } else { - *out_counts = 0; - } + Parser* parser () { + return _parser; } - const char *device () const { return _devname.c_str(); } const char *name () const { return _tagname.c_str(); } - Type type () const { return _type; } - int mode () const { return _mode; } bool ok () const { return _ok; } + virtual bool centrally_parsed() const; + void set_centrally_parsed (bool yn) { _centrally_parsed = yn; } + + bool receives_input () const { + return _flags == IsInput; + } + + bool sends_output () const { + return _flags == IsOutput; + } + struct Descriptor { std::string tag; - std::string device; - int mode; - Port::Type type; + Flags flags; Descriptor (const XMLNode&); XMLNode& get_state(); }; + static std::string state_node_name; + protected: - bool _ok; - bool _currently_in_cycle; - nframes_t _nframes_this_cycle; - Type _type; - std::string _devname; - std::string _tagname; - int _mode; - size_t _number; - Channel *_channel[16]; - sigc::connection thru_connection; - unsigned int bytes_written; - unsigned int bytes_read; - Parser *input_parser; - Parser *output_parser; - size_t slowdown; - - virtual std::string get_typestring () const = 0; - - private: - static size_t nports; + bool _ok; + std::string _tagname; + Channel* _channel[16]; + Parser* _parser; + Flags _flags; + bool _centrally_parsed; + + void init (std::string const &, Flags); }; struct PortSet { @@ -187,8 +141,8 @@ struct PortSet { std::list ports; }; -std::ostream & operator << ( std::ostream & os, const Port & port ); +std::ostream & operator << (std::ostream& os, const Port& port); } // namespace MIDI -#endif // __libmidi_port_h__ +#endif // __libmidi_port_base_h__