From cde1ee8fa147dfd15dc5c5b43093cd8c8a402b74 Mon Sep 17 00:00:00 2001 From: Miller Puckette Date: Wed, 16 Jan 2008 21:54:11 +0000 Subject: 0.41-0 test 11 svn path=/trunk/; revision=9147 --- pd/portmidi/pm_common/pminternal.h | 75 ++-- pd/portmidi/pm_common/pmutil.c | 281 ++++++++++++--- pd/portmidi/pm_common/pmutil.h | 82 ++++- pd/portmidi/pm_common/portmidi.c | 702 ++++++++++++++++++++++++------------- pd/portmidi/pm_common/portmidi.h | 418 +++++++++------------- 5 files changed, 966 insertions(+), 592 deletions(-) (limited to 'pd/portmidi/pm_common') diff --git a/pd/portmidi/pm_common/pminternal.h b/pd/portmidi/pm_common/pminternal.h index 20677308..b8c03232 100644 --- a/pd/portmidi/pm_common/pminternal.h +++ b/pd/portmidi/pm_common/pminternal.h @@ -28,11 +28,11 @@ void pm_free(void *ptr); /* if an error occurs while opening or closing a midi stream, set these: */ extern int pm_hosterror; extern char pm_hosterror_text[PM_HOST_ERROR_MSG_LEN]; - + struct pm_internal_struct; /* these do not use PmInternal because it is not defined yet... */ -typedef PmError (*pm_write_short_fn)(struct pm_internal_struct *midi, +typedef PmError (*pm_write_short_fn)(struct pm_internal_struct *midi, PmEvent *buffer); typedef PmError (*pm_begin_sysex_fn)(struct pm_internal_struct *midi, PmTimestamp timestamp); @@ -42,7 +42,8 @@ typedef PmError (*pm_write_byte_fn)(struct pm_internal_struct *midi, unsigned char byte, PmTimestamp timestamp); typedef PmError (*pm_write_realtime_fn)(struct pm_internal_struct *midi, PmEvent *buffer); -typedef PmError (*pm_write_flush_fn)(struct pm_internal_struct *midi); +typedef PmError (*pm_write_flush_fn)(struct pm_internal_struct *midi, + PmTimestamp timestamp); typedef PmTimestamp (*pm_synchronize_fn)(struct pm_internal_struct *midi); /* pm_open_fn should clean up all memory and close the device if any part of the open fails */ @@ -69,7 +70,7 @@ typedef struct { pm_abort_fn abort; /* abort */ pm_close_fn close; /* close device */ pm_poll_fn poll; /* read pending midi events into portmidi buffer */ - pm_has_host_error_fn has_host_error; /* true when device has had host + pm_has_host_error_fn has_host_error; /* true when device has had host error message */ pm_host_error_fn host_error; /* provide text readable host error message for device (clears and resets) */ @@ -83,7 +84,7 @@ typedef struct { PmDeviceInfo pub; /* some portmidi state also saved in here (for autmatic device closing (see PmDeviceInfo struct) */ void *descriptor; /* ID number passed to win32 multimedia API open */ - void *internalDescriptor; /* points to PmInternal device, allows automatic + void *internalDescriptor; /* points to PmInternal device, allows automatic device closing */ pm_fns_type dictionary; } descriptor_node, *descriptor_type; @@ -97,69 +98,77 @@ typedef unsigned long (*time_get_proc_type)(void *time_info); typedef struct pm_internal_struct { int device_id; /* which device is open (index to descriptors) */ short write_flag; /* MIDI_IN, or MIDI_OUT */ - + PmTimeProcPtr time_proc; /* where to get the time */ void *time_info; /* pass this to get_time() */ - - long buffer_len; /* how big is the buffer */ - PmEvent *buffer; /* storage for: - - midi input + long buffer_len; /* how big is the buffer or queue? */ +#ifdef NEWBUFFER + PmQueue *queue; +#else + PmEvent *buffer; /* storage for: + - midi input - midi output w/latency != 0 */ long head; long tail; - + int overflow; /* set to non-zero if input is dropped */ +#endif long latency; /* time delay in ms between timestamps and actual output */ /* set to zero to get immediate, simple blocking output */ /* if latency is zero, timestamps will be ignored; */ /* if midi input device, this field ignored */ - - int overflow; /* set to non-zero if input is dropped */ - int flush; /* flag to drop incoming sysex data because of overflow */ - int sysex_in_progress; /* use for overflow management */ + + int sysex_in_progress; /* when sysex status is seen, this flag becomes + * true until EOX is seen. When true, new data is appended to the + * stream of outgoing bytes. When overflow occurs, sysex data is + * dropped (until an EOX or non-real-timei status byte is seen) so + * that, if the overflow condition is cleared, we don't start + * sending data from the middle of a sysex message. If a sysex + * message is filtered, sysex_in_progress is false, causing the + * message to be dropped. */ PmMessage sysex_message; /* buffer for 4 bytes of sysex data */ int sysex_message_count; /* how many bytes in sysex_message so far */ long filters; /* flags that filter incoming message classes */ - int channel_mask; /* filter incoming messages based on channel */ + int channel_mask; /* flter incoming messages based on channel */ PmTimestamp last_msg_time; /* timestamp of last message */ PmTimestamp sync_time; /* time of last synchronization */ PmTimestamp now; /* set by PmWrite to current time */ int first_message; /* initially true, used to run first synchronization */ pm_fns_type dictionary; /* implementation functions */ void *descriptor; /* system-dependent state */ - + /* the following are used to expedite sysex data */ + /* on windows, in debug mode, based on some profiling, these optimizations + * cut the time to process sysex bytes from about 7.5 to 0.26 usec/byte, + * but this does not count time in the driver, so I don't know if it is + * important + */ + unsigned char *fill_base; /* addr of ptr to sysex data */ + int *fill_offset_ptr; /* offset of next sysex byte */ + int fill_length; /* how many sysex bytes to write */ } PmInternal; -typedef struct { - long head; - long tail; - long len; - long msg_size; - long overflow; - char *buffer; -} PmQueueRep; /* defined by system specific implementation, e.g. pmwinmm, used by PortMidi */ -void pm_init(void); -void pm_term(void); +void pm_init(void); +void pm_term(void); /* defined by portMidi, used by pmwinmm */ PmError none_write_short(PmInternal *midi, PmEvent *buffer); -PmError none_sysex(PmInternal *midi, PmTimestamp timestamp); -PmError none_write_byte(PmInternal *midi, unsigned char byte, +PmError none_write_byte(PmInternal *midi, unsigned char byte, PmTimestamp timestamp); PmTimestamp none_synchronize(PmInternal *midi); PmError pm_fail_fn(PmInternal *midi); +PmError pm_fail_timestamp_fn(PmInternal *midi, PmTimestamp timestamp); PmError pm_success_fn(PmInternal *midi); PmError pm_add_device(char *interf, char *name, int input, void *descriptor, pm_fns_type dictionary); -void pm_read_byte(PmInternal *midi, unsigned char byte, PmTimestamp timestamp); -void pm_begin_sysex(PmInternal *midi); -void pm_end_sysex(PmInternal *midi); +unsigned int pm_read_bytes(PmInternal *midi, unsigned char *data, int len, + PmTimestamp timestamp); void pm_read_short(PmInternal *midi, PmEvent *event); -#define none_write_flush pm_fail_fn +#define none_write_flush pm_fail_timestamp_fn +#define none_sysex pm_fail_timestamp_fn #define none_poll pm_fail_fn #define success_poll pm_success_fn diff --git a/pd/portmidi/pm_common/pmutil.c b/pd/portmidi/pm_common/pmutil.c index 1178b80b..42f386c1 100644 --- a/pd/portmidi/pm_common/pmutil.c +++ b/pd/portmidi/pm_common/pmutil.c @@ -2,30 +2,73 @@ applications that use PortMidi */ #include "stdlib.h" +#include "assert.h" #include "memory.h" #include "portmidi.h" #include "pmutil.h" #include "pminternal.h" +#ifdef WIN32 +#define bzero(addr, siz) memset(addr, 0, siz) +#endif + +// #define QUEUE_DEBUG 1 +#ifdef QUEUE_DEBUG +#include "stdio.h" +#endif + +/* code is based on 4-byte words -- it should work on a 64-bit machine + as long as a "long" has 4 bytes. This code could be generalized to + be independent of the size of "long" */ + +typedef long int32; + +typedef struct { + long head; + long tail; + long len; + long msg_size; /* number of int32 in a message including extra word */ + long overflow; + long peek_overflow; + int32 *buffer; + int32 *peek; + int peek_flag; +} PmQueueRep; + PmQueue *Pm_QueueCreate(long num_msgs, long bytes_per_msg) { PmQueueRep *queue = (PmQueueRep *) pm_alloc(sizeof(PmQueueRep)); - - /* arg checking */ + int int32s_per_msg = ((bytes_per_msg + sizeof(int32) - 1) & + ~(sizeof(int32) - 1)) / sizeof(int32); + /* arg checking */ if (!queue) - return NULL; - - queue->len = num_msgs * bytes_per_msg; - queue->buffer = pm_alloc(queue->len); + return NULL; + + /* need extra word per message for non-zero encoding */ + queue->len = num_msgs * (int32s_per_msg + 1); + queue->buffer = (int32 *) pm_alloc(queue->len * sizeof(int32)); + bzero(queue->buffer, queue->len * sizeof(int32)); if (!queue->buffer) { pm_free(queue); return NULL; + } else { /* allocate the "peek" buffer */ + queue->peek = (int32 *) pm_alloc(int32s_per_msg * sizeof(int32)); + if (!queue->peek) { + /* free everything allocated so far and return */ + pm_free(queue->buffer); + pm_free(queue); + return NULL; + } } + bzero(queue->buffer, queue->len * sizeof(int32)); queue->head = 0; queue->tail = 0; - queue->msg_size = bytes_per_msg; + /* msg_size is in words */ + queue->msg_size = int32s_per_msg + 1; /* note extra word is counted */ queue->overflow = FALSE; + queue->peek_overflow = FALSE; + queue->peek_flag = FALSE; return queue; } @@ -33,12 +76,13 @@ PmQueue *Pm_QueueCreate(long num_msgs, long bytes_per_msg) PmError Pm_QueueDestroy(PmQueue *q) { PmQueueRep *queue = (PmQueueRep *) q; - - /* arg checking */ - if (!queue || !queue->buffer) - return pmBadPtr; + + /* arg checking */ + if (!queue || !queue->buffer || !queue->peek) + return pmBadPtr; - pm_free(queue->buffer); + pm_free(queue->peek); + pm_free(queue->buffer); pm_free(queue); return pmNoError; } @@ -48,19 +92,87 @@ PmError Pm_Dequeue(PmQueue *q, void *msg) { long head; PmQueueRep *queue = (PmQueueRep *) q; + int i; + int32 *msg_as_int32 = (int32 *) msg; - /* arg checking */ - if(!queue) - return pmBadPtr; + /* arg checking */ + if (!queue) + return pmBadPtr; + /* a previous peek operation encountered an overflow, but the overflow + * has not yet been reported to client, so do it now. No message is + * returned, but on the next call, we will return the peek buffer. + */ + if (queue->peek_overflow) { + queue->peek_overflow = FALSE; + return pmBufferOverflow; + } + if (queue->peek_flag) { +#ifdef QUEUE_DEBUG + printf("Pm_Dequeue returns peek msg:"); + for (i = 0; i < queue->msg_size - 1; i++) { + printf(" %d", queue->peek[i]); + } + printf("\n"); +#endif + memcpy(msg, queue->peek, (queue->msg_size - 1) * sizeof(int32)); + queue->peek_flag = FALSE; + return 1; + } - if (queue->overflow) { - queue->overflow = FALSE; + head = queue->head; + /* if writer overflows, it writes queue->overflow = tail+1 so that + * when the reader gets to that position in the buffer, it can + * return the overflow condition to the reader. The problem is that + * at overflow, things have wrapped around, so tail == head, and the + * reader will detect overflow immediately instead of waiting until + * it reads everything in the buffer, wrapping around again to the + * point where tail == head. So the condition also checks that + * queue->buffer[head] is zero -- if so, then the buffer is now + * empty, and we're at the point in the msg stream where overflow + * occurred. It's time to signal overflow to the reader. If + * queue->buffer[head] is non-zero, there's a message there and we + * should read all the way around the buffer before signalling overflow. + * There is a write-order dependency here, but to fail, the overflow + * field would have to be written while an entire buffer full of + * writes are still pending. I'm assuming out-of-order writes are + * possible, but not that many. + */ + if (queue->overflow == head + 1 && !queue->buffer[head]) { + queue->overflow = 0; /* non-overflow condition */ return pmBufferOverflow; } - head = queue->head; /* make sure this is written after access */ - if (head == queue->tail) return 0; - memcpy(msg, queue->buffer + head, queue->msg_size); + /* test to see if there is data in the queue -- test from back + * to front so if writer is simultaneously writing, we don't + * waste time discovering the write is not finished + */ + for (i = queue->msg_size - 1; i >= 0; i--) { + if (!queue->buffer[head + i]) { + return 0; + } + } +#ifdef QUEUE_DEBUG + printf("Pm_Dequeue:"); + for (i = 0; i < queue->msg_size; i++) { + printf(" %d", queue->buffer[head + i]); + } + printf("\n"); +#endif + memcpy(msg, (char *) &queue->buffer[head + 1], + sizeof(int32) * (queue->msg_size - 1)); + /* fix up zeros */ + i = queue->buffer[head]; + while (i < queue->msg_size) { + int32 j; + i--; /* msg does not have extra word so shift down */ + j = msg_as_int32[i]; + msg_as_int32[i] = 0; + i = j; + } + /* signal that data has been removed by zeroing: */ + bzero((char *) &queue->buffer[head], sizeof(int32) * queue->msg_size); + + /* update head */ head += queue->msg_size; if (head == queue->len) head = 0; queue->head = head; @@ -68,65 +180,132 @@ PmError Pm_Dequeue(PmQueue *q, void *msg) } -/* source should not enqueue data if overflow is set */ -/**/ + +PmError Pm_SetOverflow(PmQueue *q) +{ + PmQueueRep *queue = (PmQueueRep *) q; + long tail; + /* no more enqueue until receiver acknowledges overflow */ + if (queue->overflow) return pmBufferOverflow; + if (!queue) + return pmBadPtr; + tail = queue->tail; + queue->overflow = tail + 1; + return pmBufferOverflow; +} + + PmError Pm_Enqueue(PmQueue *q, void *msg) { PmQueueRep *queue = (PmQueueRep *) q; long tail; + int i; + int32 *src = (int32 *) msg; + int32 *ptr; - /* arg checking */ - if (!queue) - return pmBadPtr; + int32 *dest; - tail = queue->tail; - memcpy(queue->buffer + tail, msg, queue->msg_size); - tail += queue->msg_size; - if (tail == queue->len) tail = 0; - if (tail == queue->head) { - queue->overflow = TRUE; - /* do not update tail, so message is lost */ + int rslt; + /* no more enqueue until receiver acknowledges overflow */ + if (!queue) return pmBadPtr; + if (queue->overflow) return pmBufferOverflow; + rslt = Pm_QueueFull(q); + /* already checked above: if (rslt == pmBadPtr) return rslt; */ + tail = queue->tail; + if (rslt) { + queue->overflow = tail + 1; return pmBufferOverflow; } + + /* queue is has room for message, and overflow flag is cleared */ + ptr = &queue->buffer[tail]; + dest = ptr + 1; + for (i = 1; i < queue->msg_size; i++) { + int32 j = src[i - 1]; + if (!j) { + *ptr = i; + ptr = dest; + } else { + *dest = j; + } + dest++; + } + *ptr = i; +#ifdef QUEUE_DEBUG + printf("Pm_Enqueue:"); + for (i = 0; i < queue->msg_size; i++) { + printf(" %d", queue->buffer[tail + i]); + } + printf("\n"); +#endif + tail += queue->msg_size; + if (tail == queue->len) tail = 0; queue->tail = tail; return pmNoError; } + int Pm_QueueEmpty(PmQueue *q) { PmQueueRep *queue = (PmQueueRep *) q; if (!queue) return TRUE; - return (queue->head == queue->tail); + return (queue->buffer[queue->head] == 0); } + int Pm_QueueFull(PmQueue *q) { PmQueueRep *queue = (PmQueueRep *) q; - long tail; - - /* arg checking */ - if(!queue) - return pmBadPtr; - - tail = queue->tail; - tail += queue->msg_size; - if (tail == queue->len) { - tail = 0; + int tail; + int i; + /* arg checking */ + if (!queue) + return pmBadPtr; + tail = queue->tail; + /* test to see if there is space in the queue */ + for (i = 0; i < queue->msg_size; i++) { + if (queue->buffer[tail + i]) { + return TRUE; + } } - return (tail == queue->head); + return FALSE; } void *Pm_QueuePeek(PmQueue *q) { - long head; PmQueueRep *queue = (PmQueueRep *) q; + PmError rslt; + long temp; - /* arg checking */ - if(!queue) - return NULL; + /* arg checking */ + if (!queue) + return NULL; - head = queue->head; /* make sure this is written after access */ - if (head == queue->tail) return NULL; - return queue->buffer + head; + if (queue->peek_flag) { + return queue->peek; + } + /* this is ugly: if peek_overflow is set, then Pm_Dequeue() + * returns immediately with pmBufferOverflow, but here, we + * want Pm_Dequeue() to really check for data. If data is + * there, we can return it + */ + temp = queue->peek_overflow; + queue->peek_overflow = FALSE; + rslt = Pm_Dequeue(q, queue->peek); + queue->peek_overflow = temp; + + if (rslt == 1) { + queue->peek_flag = TRUE; + return queue->peek; + } else if (rslt == pmBufferOverflow) { + /* when overflow is indicated, the queue is empty and the + * first message that was dropped by Enqueue (signalling + * pmBufferOverflow to its caller) would have been the next + * message in the queue. Pm_QueuePeek will return NULL, but + * remember that an overflow occurred. (see Pm_Dequeue) + */ + queue->peek_overflow = TRUE; + } + return NULL; } diff --git a/pd/portmidi/pm_common/pmutil.h b/pd/portmidi/pm_common/pmutil.h index a7fb3ebf..8e429467 100644 --- a/pd/portmidi/pm_common/pmutil.h +++ b/pd/portmidi/pm_common/pmutil.h @@ -2,6 +2,10 @@ applications that use PortMidi */ +#ifdef __cplusplus +extern "C" { +#endif /* __cplusplus */ + typedef void PmQueue; /* @@ -10,6 +14,33 @@ typedef void PmQueue; the message size as parameters. The queue only accepts fixed sized messages. Returns NULL if memory cannot be allocated. + This queue implementation uses the "light pipe" algorithm which + operates correctly even with multi-processors and out-of-order + memory writes. (see Alexander Dokumentov, "Lock-free Interprocess + Communication," Dr. Dobbs Portal, http://www.ddj.com/, + articleID=189401457, June 15, 2006. This algorithm requires + that messages be translated to a form where no words contain + zeros. Each word becomes its own "data valid" tag. Because of + this translation, we cannot return a pointer to data still in + the queue when the "peek" method is called. Instead, a buffer + is preallocated so that data can be copied there. Pm_QueuePeek() + dequeues a message into this buffer and returns a pointer to + it. A subsequent Pm_Dequeue() will copy from this buffer. + + This implementation does not try to keep reader/writer data in + separate cache lines or prevent thrashing on cache lines. + However, this algorithm differs by doing inserts/removals in + units of messages rather than units of machine words. Some + performance improvement might be obtained by not clearing data + immediately after a read, but instead by waiting for the end + of the cache line, especially if messages are smaller than + cache lines. See the Dokumentov article for explanation. + + The algorithm is extended to handle "overflow" reporting. To report + an overflow, the sender writes the current tail position to a field. + The receiver must acknowlege receipt by zeroing the field. The sender + will not send more until the field is zeroed. + Pm_QueueDestroy() destroys the queue and frees its storage. */ @@ -19,10 +50,11 @@ PmError Pm_QueueDestroy(PmQueue *queue); /* Pm_Dequeue() removes one item from the queue, copying it into msg. Returns 1 if successful, and 0 if the queue is empty. - Returns pmBufferOverflow, clears the overflow flag, and does not - return a data item if the overflow flag is set. (This protocol - ensures that the reader will be notified when data is lost due - to overflow.) + Returns pmBufferOverflow if what would have been the next thing + in the queue was dropped due to overflow. (So when overflow occurs, + the receiver can receive a queue full of messages before getting the + overflow report. This protocol ensures that the reader will be + notified when data is lost due to overflow. */ PmError Pm_Dequeue(PmQueue *queue, void *msg); @@ -40,7 +72,12 @@ PmError Pm_Enqueue(PmQueue *queue, void *msg); Pm_QueueEmpty() returns non-zero if the queue is empty Either condition may change immediately because a parallel - enqueue or dequeue operation could be in progress. + enqueue or dequeue operation could be in progress. Furthermore, + Pm_QueueEmpty() is optimistic: it may say false, when due to + out-of-order writes, the full message has not arrived. Therefore, + Pm_Dequeue() could still return 0 after Pm_QueueEmpty() returns + false. On the other hand, Pm_QueueFull() is pessimistic: if it + returns false, then Pm_Enqueue() is guaranteed to succeed. */ int Pm_QueueFull(PmQueue *queue); int Pm_QueueEmpty(PmQueue *queue); @@ -49,8 +86,39 @@ int Pm_QueueEmpty(PmQueue *queue); /* Pm_QueuePeek() returns a pointer to the item at the head of the queue, or NULL if the queue is empty. The item is not removed from the queue. - If queue is in an overflow state, a valid pointer is returned and the - queue remains in the overflow state. + Pm_QueuePeek() will not indicate when an overflow occurs. If you want + to get and check pmBufferOverflow messages, use the return value of + Pm_QueuePeek() *only* as an indication that you should call + Pm_Dequeue(). At the point where a direct call to Pm_Dequeue() would + return pmBufferOverflow, Pm_QueuePeek() will return NULL but internally + clear the pmBufferOverflow flag, enabling Pm_Enqueue() to resume + enqueuing messages. A subsequent call to Pm_QueuePeek() + will return a pointer to the first message *after* the overflow. + Using this as an indication to call Pm_Dequeue(), the first call + to Pm_Dequeue() will return pmBufferOverflow. The second call will + return success, copying the same message pointed to by the previous + Pm_QueuePeek(). + + When to use Pm_QueuePeek(): (1) when you need to look at the message + data to decide who should be called to receive it. (2) when you need + to know a message is ready but cannot accept the message. + + Note that Pm_QueuePeek() is not a fast check, so if possible, you + might as well just call Pm_Dequeue() and accept the data if it is there. */ void *Pm_QueuePeek(PmQueue *queue); +/* + Pm_SetOverflow() allows the writer (enqueuer) to signal an overflow + condition to the reader (dequeuer). E.g. when transfering data from + the OS to an application, if the OS indicates a buffer overrun, + Pm_SetOverflow() can be used to insure that the reader receives a + pmBufferOverflow result from Pm_Dequeue(). Returns pmBadPtr if queue + is NULL, returns pmBufferOverflow if buffer is already in an overflow + state, returns pmNoError if successfully set overflow state. + */ +PmError Pm_SetOverflow(PmQueue *queue); + +#ifdef __cplusplus +} +#endif /* __cplusplus */ diff --git a/pd/portmidi/pm_common/portmidi.c b/pd/portmidi/pm_common/portmidi.c index e1b962d7..71988e47 100644 --- a/pd/portmidi/pm_common/portmidi.c +++ b/pd/portmidi/pm_common/portmidi.c @@ -2,6 +2,9 @@ #include "string.h" #include "portmidi.h" #include "porttime.h" +#ifdef NEWBUFFER +#include "pmutil.h" +#endif #include "pminternal.h" #include @@ -31,7 +34,7 @@ #define is_empty(midi) ((midi)->tail == (midi)->head) static int pm_initialized = FALSE; -int pm_hosterror = FALSE; +int pm_hosterror; char pm_hosterror_text[PM_HOST_ERROR_MSG_LEN]; #ifdef PM_CHECK_ERRORS @@ -80,24 +83,24 @@ int pm_descriptor_max = 0; int pm_descriptor_index = 0; descriptor_type descriptors = NULL; -/* pm_add_device -- describe interface/device pair to library +/* pm_add_device -- describe interface/device pair to library * - * This is called at intialization time, once for each + * This is called at intialization time, once for each * interface (e.g. DirectSound) and device (e.g. SoundBlaster 1) * The strings are retained but NOT COPIED, so do not destroy them! * * returns pmInvalidDeviceId if device memory is exceeded * otherwise returns pmNoError */ -PmError pm_add_device(char *interf, char *name, int input, +PmError pm_add_device(char *interf, char *name, int input, void *descriptor, pm_fns_type dictionary) { if (pm_descriptor_index >= pm_descriptor_max) { // expand descriptors - descriptor_type new_descriptors = + descriptor_type new_descriptors = pm_alloc(sizeof(descriptor_node) * (pm_descriptor_max + 32)); if (!new_descriptors) return pmInsufficientMemory; if (descriptors) { - memcpy(new_descriptors, descriptors, + memcpy(new_descriptors, descriptors, sizeof(descriptor_node) * pm_descriptor_max); free(descriptors); } @@ -114,14 +117,14 @@ PmError pm_add_device(char *interf, char *name, int input, /* ID number passed to win32 multimedia API open */ descriptors[pm_descriptor_index].descriptor = descriptor; - + /* points to PmInternal, allows automatic device closing */ descriptors[pm_descriptor_index].internalDescriptor = NULL; descriptors[pm_descriptor_index].dictionary = dictionary; - + pm_descriptor_index++; - + return pmNoError; } @@ -132,20 +135,15 @@ portmidi implementation ==================================================================== */ -int Pm_CountDevices( void ) -{ - PmError err = Pm_Initialize(); - if (err) - return pm_errmsg(err); +int Pm_CountDevices( void ) { + Pm_Initialize(); + /* no error checking -- Pm_Initialize() does not fail */ return pm_descriptor_index; } -const PmDeviceInfo* Pm_GetDeviceInfo( PmDeviceID id ) -{ - PmError err = Pm_Initialize(); - if (err) - return NULL; +const PmDeviceInfo* Pm_GetDeviceInfo( PmDeviceID id ) { + Pm_Initialize(); /* no error check needed */ if (id >= 0 && id < pm_descriptor_index) { return &descriptors[id].pub; } @@ -158,35 +156,30 @@ PmError pm_success_fn(PmInternal *midi) { } /* none_write -- returns an error if called */ -PmError none_write_short(PmInternal *midi, PmEvent *buffer) -{ +PmError none_write_short(PmInternal *midi, PmEvent *buffer) { return pmBadPtr; } -/* none_sysex -- placeholder for begin_sysex and end_sysex */ -PmError none_sysex(PmInternal *midi, PmTimestamp timestamp) -{ +/* pm_fail_timestamp_fn -- placeholder for begin_sysex and flush */ +PmError pm_fail_timestamp_fn(PmInternal *midi, PmTimestamp timestamp) { return pmBadPtr; } -PmError none_write_byte(PmInternal *midi, unsigned char byte, - PmTimestamp timestamp) -{ +PmError none_write_byte(PmInternal *midi, unsigned char byte, + PmTimestamp timestamp) { return pmBadPtr; } /* pm_fail_fn -- generic function, returns error if called */ -PmError pm_fail_fn(PmInternal *midi) -{ +PmError pm_fail_fn(PmInternal *midi) { return pmBadPtr; } -static PmError none_open(PmInternal *midi, void *driverInfo) -{ +static PmError none_open(PmInternal *midi, void *driverInfo) { return pmBadPtr; } static void none_get_host_error(PmInternal * midi, char * msg, unsigned int len) { - strcpy(msg,""); + strcpy(msg, ""); } static unsigned int none_has_host_error(PmInternal * midi) { return FALSE; @@ -207,11 +200,11 @@ pm_fns_node pm_none_dictionary = { none_write_flush, none_synchronize, none_open, - none_abort, + none_abort, none_close, none_poll, none_has_host_error, - none_get_host_error + none_get_host_error }; @@ -220,34 +213,38 @@ const char *Pm_GetErrorText( PmError errnum ) { switch(errnum) { - case pmNoError: - msg = ""; + case pmNoError: + msg = ""; break; - case pmHostError: - msg = "PortMidi: `Host error'"; + case pmHostError: + msg = "PortMidi: `Host error'"; break; - case pmInvalidDeviceId: - msg = "PortMidi: `Invalid device ID'"; + case pmInvalidDeviceId: + msg = "PortMidi: `Invalid device ID'"; break; - case pmInsufficientMemory: - msg = "PortMidi: `Insufficient memory'"; + case pmInsufficientMemory: + msg = "PortMidi: `Insufficient memory'"; break; - case pmBufferTooSmall: - msg = "PortMidi: `Buffer too small'"; + case pmBufferTooSmall: + msg = "PortMidi: `Buffer too small'"; break; - case pmBadPtr: - msg = "PortMidi: `Bad pointer'"; + case pmBadPtr: + msg = "PortMidi: `Bad pointer'"; break; - case pmInternalError: - msg = "PortMidi: `Internal PortMidi Error'"; + case pmInternalError: + msg = "PortMidi: `Internal PortMidi Error'"; break; case pmBufferOverflow: msg = "PortMidi: `Buffer overflow'"; break; case pmBadData: msg = "PortMidi: `Invalid MIDI message Data'"; - default: - msg = "PortMidi: `Illegal error number'"; + break; + case pmBufferMaxSize: + msg = "PortMidi: `Buffer cannot be made larger'"; + break; + default: + msg = "PortMidi: `Illegal error number'"; break; } return msg; @@ -260,11 +257,11 @@ const char *Pm_GetErrorText( PmError errnum ) { void Pm_GetHostErrorText(char * msg, unsigned int len) { assert(msg); assert(len > 0); - if (pm_hosterror) { /* we have the string already from open or close */ + if (pm_hosterror) { strncpy(msg, (char *) pm_hosterror_text, len); pm_hosterror = FALSE; pm_hosterror_text[0] = 0; /* clear the message; not necessary, but it - might help with debugging */ + might help with debugging */ msg[len - 1] = 0; /* make sure string is terminated */ } else { msg[0] = 0; /* no string to return */ @@ -273,11 +270,12 @@ void Pm_GetHostErrorText(char * msg, unsigned int len) { int Pm_HasHostError(PortMidiStream * stream) { + if (pm_hosterror) return TRUE; if (stream) { PmInternal * midi = (PmInternal *) stream; pm_hosterror = (*midi->dictionary->has_host_error)(midi); if (pm_hosterror) { - midi->dictionary->host_error(midi, pm_hosterror_text, + midi->dictionary->host_error(midi, pm_hosterror_text, PM_HOST_ERROR_MSG_LEN); /* now error message is global */ return TRUE; @@ -288,8 +286,9 @@ int Pm_HasHostError(PortMidiStream * stream) { PmError Pm_Initialize( void ) { - pm_hosterror_text[0] = 0; /* the null string */ if (!pm_initialized) { + pm_hosterror = FALSE; + pm_hosterror_text[0] = 0; /* the null string */ pm_init(); pm_initialized = TRUE; } @@ -300,6 +299,13 @@ PmError Pm_Initialize( void ) { PmError Pm_Terminate( void ) { if (pm_initialized) { pm_term(); + // if there are no devices, descriptors might still be NULL + if (descriptors != NULL) { + free(descriptors); + descriptors = NULL; + } + pm_descriptor_index = 0; + pm_descriptor_max = 0; pm_initialized = FALSE; } return pmNoError; @@ -308,33 +314,23 @@ PmError Pm_Terminate( void ) { /* Pm_Read -- read up to length longs from source into buffer */ /* - returns number of longs actually read, or error code - - When the reader wants data: - if overflow_flag: - do not get anything - empty the buffer (read_ptr = write_ptr) - clear overflow_flag - return pmBufferOverflow - get data - return number of messages -*/ + * returns number of longs actually read, or error code + */ PmError Pm_Read(PortMidiStream *stream, PmEvent *buffer, long length) { PmInternal *midi = (PmInternal *) stream; int n = 0; +#ifndef NEWBUFFER long head; +#endif PmError err = pmNoError; - + pm_hosterror = FALSE; /* arg checking */ if(midi == NULL) err = pmBadPtr; - else if(Pm_HasHostError(midi)) - err = pmHostError; else if(!descriptors[midi->device_id].pub.opened) err = pmBadPtr; else if(!descriptors[midi->device_id].pub.input) - err = pmBadPtr; - + err = pmBadPtr; /* First poll for data in the buffer... * This either simply checks for data, or attempts first to fill the buffer * with data from the MIDI hardware; this depends on the implementation. @@ -343,9 +339,26 @@ PmError Pm_Read(PortMidiStream *stream, PmEvent *buffer, long length) { else err = (*(midi->dictionary->poll))(midi); if (err != pmNoError) { + if (err == pmHostError) { + midi->dictionary->host_error(midi, pm_hosterror_text, + PM_HOST_ERROR_MSG_LEN); + pm_hosterror = TRUE; + } return pm_errmsg(err); } +#ifdef NEWBUFFER + while (n < length) { + PmError err = Pm_Dequeue(midi->queue, buffer++); + if (err == pmBufferOverflow) { + /* ignore the data we have retreived so far */ + return pm_errmsg(pmBufferOverflow); + } else if (err == 0) { /* empty queue */ + break; + } + n++; + } +#else head = midi->head; while (head != midi->tail && n < length) { PmEvent event = midi->buffer[head++]; @@ -359,6 +372,7 @@ PmError Pm_Read(PortMidiStream *stream, PmEvent *buffer, long length) { midi->overflow = FALSE; return pm_errmsg(pmBufferOverflow); } +#endif return n; } @@ -366,12 +380,14 @@ PmError Pm_Poll( PortMidiStream *stream ) { PmInternal *midi = (PmInternal *) stream; PmError err; +#ifdef NEWBUFFER + PmEvent *event; +#endif + pm_hosterror = FALSE; /* arg checking */ if(midi == NULL) err = pmBadPtr; - else if(Pm_HasHostError(midi)) - err = pmHostError; else if(!descriptors[midi->device_id].pub.opened) err = pmBadPtr; else if(!descriptors[midi->device_id].pub.input) @@ -379,12 +395,41 @@ PmError Pm_Poll( PortMidiStream *stream ) else err = (*(midi->dictionary->poll))(midi); - if (err != pmNoError) + if (err != pmNoError) { + if (err == pmHostError) { + midi->dictionary->host_error(midi, pm_hosterror_text, + PM_HOST_ERROR_MSG_LEN); + pm_hosterror = TRUE; + } return pm_errmsg(err); - else - return midi->head != midi->tail; + } + +#ifdef NEWBUFFER + event = (PmEvent *) Pm_QueuePeek(midi->queue); + return event != NULL; +#else + return midi->head != midi->tail; +#endif +} + + +/* this is called from Pm_Write and Pm_WriteSysEx to issue a + * call to the system-dependent end_sysex function and handle + * the error return + */ +static PmError pm_end_sysex(PmInternal *midi) +{ + PmError err = (*midi->dictionary->end_sysex)(midi, 0); + midi->sysex_in_progress = FALSE; + if (err == pmHostError) { + midi->dictionary->host_error(midi, pm_hosterror_text, + PM_HOST_ERROR_MSG_LEN); + pm_hosterror = TRUE; + } + return err; } + /* to facilitate correct error-handling, Pm_Write, Pm_WriteShort, and Pm_WriteSysEx all operate a state machine that "outputs" calls to write_short, begin_sysex, write_byte, end_sysex, and write_realtime */ @@ -395,21 +440,20 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length) PmError err; int i; int bits; - + + pm_hosterror = FALSE; /* arg checking */ if(midi == NULL) err = pmBadPtr; - else if(Pm_HasHostError(midi)) - err = pmHostError; else if(!descriptors[midi->device_id].pub.opened) err = pmBadPtr; else if(!descriptors[midi->device_id].pub.output) err = pmBadPtr; else err = pmNoError; - + if (err != pmNoError) goto pm_write_error; - + if (midi->latency == 0) { midi->now = 0; } else { @@ -420,7 +464,19 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length) midi->first_message = FALSE; } } - + /* error recovery: when a sysex is detected, we call + * dictionary->begin_sysex() followed by calls to + * dictionary->write_byte() and dictionary->write_realtime() + * until an end-of-sysex is detected, when we call + * dictionary->end_sysex(). After an error occurs, + * Pm_Write() continues to call functions. For example, + * it will continue to call write_byte() even after + * an error sending a sysex message, and end_sysex() will be + * called when an EOX or non-real-time status is found. + * When errors are detected, Pm_Write() returns immediately, + * so it is possible that this will drop data and leave + * sysex messages in a partially transmitted state. + */ for (i = 0; i < length; i++) { unsigned long msg = buffer[i].message; bits = 0; @@ -433,21 +489,21 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length) goto pm_write_error; } midi->sysex_in_progress = TRUE; - if ((err = (*midi->dictionary->begin_sysex)(midi, + if ((err = (*midi->dictionary->begin_sysex)(midi, buffer[i].timestamp)) != pmNoError) goto pm_write_error; if ((err = (*midi->dictionary->write_byte)(midi, MIDI_SYSEX, - buffer[i].timestamp)) != pmNoError) + buffer[i].timestamp)) != pmNoError) goto pm_write_error; bits = 8; /* fall through to continue sysex processing */ - } else if ((msg & MIDI_STATUS_MASK) && + } else if ((msg & MIDI_STATUS_MASK) && (Pm_MessageStatus(msg) != MIDI_EOX)) { /* a non-sysex message */ if (midi->sysex_in_progress) { - /* this should be a non-realtime message */ + /* this should be a realtime message */ if (is_real_time(msg)) { - if ((err = (*midi->dictionary->write_realtime)(midi, + if ((err = (*midi->dictionary->write_realtime)(midi, &(buffer[i]))) != pmNoError) goto pm_write_error; } else { @@ -459,23 +515,34 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length) goto pm_write_error; } } else { /* regular short midi message */ - if ((err = (*midi->dictionary->write_short)(midi, + if ((err = (*midi->dictionary->write_short)(midi, &(buffer[i]))) != pmNoError) goto pm_write_error; continue; } } if (midi->sysex_in_progress) { /* send sysex bytes until EOX */ + /* see if we can accelerate data transfer */ + if (bits == 0 && midi->fill_base && /* 4 bytes to copy */ + (*midi->fill_offset_ptr) + 4 <= midi->fill_length && + (msg & 0x80808080) == 0) { /* all data */ + /* copy 4 bytes from msg to fill_base + fill_offset */ + unsigned char *ptr = midi->fill_base + + *(midi->fill_offset_ptr); + ptr[0] = msg; ptr[1] = msg >> 8; + ptr[2] = msg >> 18; ptr[3] = msg >> 24; + (*midi->fill_offset_ptr) += 4; + continue; + } + /* no acceleration, so do byte-by-byte copying */ while (bits < 32) { unsigned char midi_byte = (unsigned char) (msg >> bits); - if ((err = (*midi->dictionary->write_byte)(midi, midi_byte, + if ((err = (*midi->dictionary->write_byte)(midi, midi_byte, buffer[i].timestamp)) != pmNoError) goto pm_write_error; if (midi_byte == MIDI_EOX) { - midi->sysex_in_progress = FALSE; - if ((err = (*midi->dictionary->end_sysex)(midi, - buffer[i].timestamp)) != pmNoError) - goto pm_write_error; + err = pm_end_sysex(midi); + if (err != pmNoError) goto error_exit; break; /* from while loop */ } bits += 8; @@ -487,8 +554,15 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length) } } /* after all messages are processed, send the data */ - err = (*midi->dictionary->write_flush)(midi); + if (!midi->sysex_in_progress) + err = (*midi->dictionary->write_flush)(midi, 0); pm_write_error: + if (err == pmHostError) { + midi->dictionary->host_error(midi, pm_hosterror_text, + PM_HOST_ERROR_MSG_LEN); + pm_hosterror = TRUE; + } +error_exit: return pm_errmsg(err); } @@ -496,20 +570,22 @@ pm_write_error: PmError Pm_WriteShort( PortMidiStream *stream, long when, long msg) { PmEvent event; - + event.timestamp = when; event.message = msg; return Pm_Write(stream, &event, 1); } -PmError Pm_WriteSysEx(PortMidiStream *stream, PmTimestamp when, +PmError Pm_WriteSysEx(PortMidiStream *stream, PmTimestamp when, unsigned char *msg) { /* allocate buffer space for PM_DEFAULT_SYSEX_BUFFER_SIZE bytes */ /* each PmEvent holds sizeof(PmMessage) bytes of sysex data */ #define BUFLEN (PM_DEFAULT_SYSEX_BUFFER_SIZE / sizeof(PmMessage)) PmEvent buffer[BUFLEN]; + int buffer_size = 1; /* first time, send 1. After that, it's BUFLEN */ + PmInternal *midi = (PmInternal *) stream; /* the next byte in the buffer is represented by an index, bufx, and a shift in bits */ int shift = 0; @@ -521,28 +597,58 @@ PmError Pm_WriteSysEx(PortMidiStream *stream, PmTimestamp when, /* insert next byte into buffer */ buffer[bufx].message |= ((*msg) << shift); shift += 8; + if (*msg++ == MIDI_EOX) break; if (shift == 32) { shift = 0; bufx++; - if (bufx == BUFLEN) { - PmError err = Pm_Write(stream, buffer, BUFLEN); + if (bufx == buffer_size) { + PmError err = Pm_Write(stream, buffer, buffer_size); + /* note: Pm_Write has already called errmsg() */ if (err) return err; /* prepare to fill another buffer */ bufx = 0; + buffer_size = BUFLEN; + /* optimization: maybe we can just copy bytes */ + if (midi->fill_base) { + PmError err; + while (*(midi->fill_offset_ptr) < midi->fill_length) { + midi->fill_base[(*midi->fill_offset_ptr)++] = *msg; + if (*msg++ == MIDI_EOX) { + err = pm_end_sysex(midi); + if (err != pmNoError) return pm_errmsg(err); + goto end_of_sysex; + } + } + /* I thought that I could do a pm_Write here and + * change this if to a loop, avoiding calls in Pm_Write + * to the slower write_byte, but since + * sysex_in_progress is true, this will not flush + * the buffer, and we'll infinite loop: */ + /* err = Pm_Write(stream, buffer, 0); + if (err) return err; */ + /* instead, the way this works is that Pm_Write calls + * write_byte on 4 bytes. The first, since the buffer + * is full, will flush the buffer and allocate a new + * one. This primes the buffer so + * that we can return to the loop above and fill it + * efficiently without a lot of function calls. + */ + buffer_size = 1; /* get another message started */ + } } buffer[bufx].message = 0; buffer[bufx].timestamp = when; - } + } /* keep inserting bytes until you find MIDI_EOX */ - if (*msg++ == MIDI_EOX) break; } - +end_of_sysex: /* we're finished sending full buffers, but there may * be a partial one left. */ if (shift != 0) bufx++; /* add partial message to buffer len */ if (bufx) { /* bufx is number of PmEvents to send from buffer */ - return Pm_Write(stream, buffer, bufx); + PmError err = Pm_Write(stream, buffer, bufx); + if (err) return err; } return pmNoError; } @@ -554,26 +660,25 @@ PmError Pm_OpenInput(PortMidiStream** stream, void *inputDriverInfo, long bufferSize, PmTimeProcPtr time_proc, - void *time_info) -{ + void *time_info) { PmInternal *midi; PmError err = pmNoError; pm_hosterror = FALSE; *stream = NULL; - + /* arg checking */ - if (inputDevice < 0 || inputDevice >= pm_descriptor_index) + if (inputDevice < 0 || inputDevice >= pm_descriptor_index) err = pmInvalidDeviceId; - else if (!descriptors[inputDevice].pub.input) + else if (!descriptors[inputDevice].pub.input) err = pmBadPtr; else if(descriptors[inputDevice].pub.opened) err = pmBadPtr; - - if (err != pmNoError) + + if (err != pmNoError) goto error_return; /* create portMidi internal data */ - midi = (PmInternal *) pm_alloc(sizeof(PmInternal)); + midi = (PmInternal *) pm_alloc(sizeof(PmInternal)); *stream = midi; if (!midi) { err = pmInsufficientMemory; @@ -589,29 +694,36 @@ PmError Pm_OpenInput(PortMidiStream** stream, system-specific midi_out_open() method. */ if (bufferSize <= 0) bufferSize = 256; /* default buffer size */ +#ifdef NEWBUFFER + midi->queue = Pm_QueueCreate(bufferSize, sizeof(PmEvent)); + if (!midi->queue) { +#else else bufferSize++; /* buffer holds N-1 msgs, so increase request by 1 */ - midi->buffer_len = bufferSize; /* portMidi input storage */ - midi->buffer = (PmEvent *) pm_alloc(sizeof(PmEvent) * midi->buffer_len); - if (!midi->buffer) { + midi->buffer = (PmEvent *) pm_alloc(sizeof(PmEvent) * midi->buffer_len); + midi->head = 0; + midi->tail = 0; + midi->overflow = FALSE; + if (!midi->buffer) { +#endif /* free portMidi data */ *stream = NULL; - pm_free(midi); + pm_free(midi); err = pmInsufficientMemory; goto error_return; } - midi->head = 0; - midi->tail = 0; + midi->buffer_len = bufferSize; /* portMidi input storage */ midi->latency = 0; /* not used */ - midi->overflow = FALSE; - midi->flush = FALSE; midi->sysex_in_progress = FALSE; - midi->sysex_message = 0; - midi->sysex_message_count = 0; + midi->sysex_message = 0; + midi->sysex_message_count = 0; midi->filters = PM_FILT_ACTIVE; midi->channel_mask = 0xFFFF; midi->sync_time = 0; midi->first_message = TRUE; midi->dictionary = descriptors[inputDevice].dictionary; + midi->fill_base = NULL; + midi->fill_offset_ptr = NULL; + midi->fill_length = 0; descriptors[inputDevice].internalDescriptor = midi; /* open system dependent input device */ err = (*midi->dictionary->open)(midi, inputDriverInfo); @@ -619,13 +731,21 @@ PmError Pm_OpenInput(PortMidiStream** stream, *stream = NULL; descriptors[inputDevice].internalDescriptor = NULL; /* free portMidi data */ - pm_free(midi->buffer); +#ifdef NEWBUFFER + Pm_QueueDestroy(midi->queue); +#else + pm_free(midi->buffer); +#endif pm_free(midi); } else { /* portMidi input open successful */ descriptors[inputDevice].pub.opened = TRUE; } error_return: + /* note: if there is a pmHostError, it is the responsibility + * of the system-dependent code (*midi->dictionary->open)() + * to set pm_hosterror and pm_hosterror_text + */ return pm_errmsg(err); } @@ -636,26 +756,25 @@ PmError Pm_OpenOutput(PortMidiStream** stream, long bufferSize, PmTimeProcPtr time_proc, void *time_info, - long latency) -{ + long latency ) { PmInternal *midi; PmError err = pmNoError; pm_hosterror = FALSE; *stream = NULL; - + /* arg checking */ if (outputDevice < 0 || outputDevice >= pm_descriptor_index) err = pmInvalidDeviceId; - else if (!descriptors[outputDevice].pub.output) - err = pmBadPtr; + else if (!descriptors[outputDevice].pub.output) + err = pmInvalidDeviceId; else if (descriptors[outputDevice].pub.opened) - err = pmBadPtr; - if (err != pmNoError) + err = pmInvalidDeviceId; + if (err != pmNoError) goto error_return; /* create portMidi internal data */ - midi = (PmInternal *) pm_alloc(sizeof(PmInternal)); - *stream = midi; + midi = (PmInternal *) pm_alloc(sizeof(PmInternal)); + *stream = midi; if (!midi) { err = pmInsufficientMemory; goto error_return; @@ -666,32 +785,36 @@ PmError Pm_OpenOutput(PortMidiStream** stream, /* if latency > 0, we need a time reference. If none is provided, use PortTime library */ if (time_proc == NULL && latency != 0) { - if (!Pt_Started()) + if (!Pt_Started()) Pt_Start(1, 0, 0); /* time_get does not take a parameter, so coerce */ midi->time_proc = (PmTimeProcPtr) Pt_Time; } midi->time_info = time_info; - /* when stream used, this buffer allocated and used by - winmm_out_open; deleted by winmm_out_close */ midi->buffer_len = bufferSize; +#ifdef NEWBUFFER + midi->queue = NULL; /* unused by output */ +#else midi->buffer = NULL; midi->head = 0; /* unused by output */ midi->tail = 0; /* unused by output */ + midi->overflow = FALSE; /* not used */ +#endif /* if latency zero, output immediate (timestamps ignored) */ /* if latency < 0, use 0 but don't return an error */ if (latency < 0) latency = 0; midi->latency = latency; - midi->overflow = FALSE; /* not used */ - midi->flush = FALSE; /* not used */ midi->sysex_in_progress = FALSE; midi->sysex_message = 0; /* unused by output */ midi->sysex_message_count = 0; /* unused by output */ midi->filters = 0; /* not used for output */ - midi->channel_mask = 0xFFFF; /* not used for output */ + midi->channel_mask = 0xFFFF; midi->sync_time = 0; midi->first_message = TRUE; midi->dictionary = descriptors[outputDevice].dictionary; + midi->fill_base = NULL; + midi->fill_offset_ptr = NULL; + midi->fill_length = 0; descriptors[outputDevice].internalDescriptor = midi; /* open system dependent output device */ err = (*midi->dictionary->open)(midi, outputDriverInfo); @@ -699,15 +822,19 @@ PmError Pm_OpenOutput(PortMidiStream** stream, *stream = NULL; descriptors[outputDevice].internalDescriptor = NULL; /* free portMidi data */ - pm_free(midi); + pm_free(midi); } else { /* portMidi input open successful */ descriptors[outputDevice].pub.opened = TRUE; } error_return: + /* note: system-dependent code must set pm_hosterror and + * pm_hosterror_text if a pmHostError occurs + */ return pm_errmsg(err); } + PmError Pm_SetChannelMask(PortMidiStream *stream, int mask) { PmInternal *midi = (PmInternal *) stream; @@ -721,8 +848,8 @@ PmError Pm_SetChannelMask(PortMidiStream *stream, int mask) return pm_errmsg(err); } -PmError Pm_SetFilter(PortMidiStream *stream, long filters) -{ + +PmError Pm_SetFilter(PortMidiStream *stream, long filters) { PmInternal *midi = (PmInternal *) stream; PmError err = pmNoError; @@ -737,11 +864,11 @@ PmError Pm_SetFilter(PortMidiStream *stream, long filters) } -PmError Pm_Close( PortMidiStream *stream ) -{ +PmError Pm_Close( PortMidiStream *stream ) { PmInternal *midi = (PmInternal *) stream; PmError err = pmNoError; + pm_hosterror = FALSE; /* arg checking */ if (midi == NULL) /* midi must point to something */ err = pmBadPtr; @@ -751,8 +878,8 @@ PmError Pm_Close( PortMidiStream *stream ) /* and the device should be in the opened state */ else if (!descriptors[midi->device_id].pub.opened) err = pmBadPtr; - - if (err != pmNoError) + + if (err != pmNoError) goto error_return; /* close the device */ @@ -760,15 +887,21 @@ PmError Pm_Close( PortMidiStream *stream ) /* even if an error occurred, continue with cleanup */ descriptors[midi->device_id].internalDescriptor = NULL; descriptors[midi->device_id].pub.opened = FALSE; - pm_free(midi->buffer); - pm_free(midi); +#ifdef NEWBUFFER + if (midi->queue) Pm_QueueDestroy(midi->queue); +#else + if (midi->buffer) pm_free(midi->buffer); +#endif + pm_free(midi); error_return: + /* system dependent code must set pm_hosterror and + * pm_hosterror_text if a pmHostError occurs. + */ return pm_errmsg(err); } -PmError Pm_Abort( PortMidiStream* stream ) -{ +PmError Pm_Abort( PortMidiStream* stream ) { PmInternal *midi = (PmInternal *) stream; PmError err; /* arg checking */ @@ -780,39 +913,47 @@ PmError Pm_Abort( PortMidiStream* stream ) err = pmBadPtr; else err = (*midi->dictionary->abort)(midi); + + if (err == pmHostError) { + midi->dictionary->host_error(midi, pm_hosterror_text, + PM_HOST_ERROR_MSG_LEN); + pm_hosterror = TRUE; + } return pm_errmsg(err); } +#ifndef NEWBUFFER +/* this is apparently an orphan routine -- I can find no reference to it now -RBD */ + /* in win32 multimedia API (via callbacks) some of these functions used; assume never fail */ long pm_next_time(PmInternal *midi) { /* arg checking */ assert(midi != NULL); - assert(!Pm_HasHostError(midi)); - + return midi->buffer[midi->head].timestamp; } +#endif + + /* pm_channel_filtered returns non-zero if the channel mask is blocking the current channel */ -static int pm_channel_filtered(int status, int mask) -{ - if ((status & 0xF0) == 0xF0) /* 0xF? messages don't have a channel */ - return 0; - return !(Pm_Channel(status & 0x0F) & mask); - /* it'd be easier to return 0 for filtered, 1 for allowed, - but it would different from the other filtering functions - */ +#define pm_channel_filtered(status, mask) \ + ((((status) & 0xF0) != 0xF0) && (!(Pm_Channel((status) & 0x0F) & (mask)))) + -} /* The following two functions will checks to see if a MIDI message matches the filtering criteria. Since the sysex routines only want to filter realtime messages, we need to have separate routines. */ + /* pm_realtime_filtered returns non-zero if the filter will kill the current message. Note that only realtime messages are checked here. */ -static int pm_realtime_filtered(int status, long filters) -{ +#define pm_realtime_filtered(status, filters) \ + ((((status) & 0xF0) == 0xF0) && ((1 << ((status) & 0xF)) & (filters))) + +/* return ((status == MIDI_ACTIVE) && (filters & PM_FILT_ACTIVE)) || ((status == MIDI_CLOCK) && (filters & PM_FILT_CLOCK)) || ((status == MIDI_START) && (filters & PM_FILT_PLAY)) @@ -825,16 +966,18 @@ static int pm_realtime_filtered(int status, long filters) || ((status == MIDI_SONGPOS) && (filters & PM_FILT_SONG_POSITION)) || ((status == MIDI_SONGSEL) && (filters & PM_FILT_SONG_SELECT)) || ((status == MIDI_TUNE) && (filters & PM_FILT_TUNE)); -} +}*/ + + /* pm_status_filtered returns non-zero if a filter will kill the current message, based on status. Note that sysex and real time are not checked. It is up to the subsystem (winmm, core midi, alsa) to filter sysex, as it is handled more easily and efficiently at that level. Realtime message are filtered in pm_realtime_filtered. */ +#define pm_status_filtered(status, filters) ((1 << (16 + ((status) >> 4))) & (filters)) -static int pm_status_filtered(int status, long filters) -{ - status &= 0xF0; /* remove channel information */ + +/* return ((status == MIDI_NOTE_ON) && (filters & PM_FILT_NOTE)) || ((status == MIDI_NOTE_OFF) && (filters & PM_FILT_NOTE)) || ((status == MIDI_CHANNEL_AT) && (filters & PM_FILT_CHANNEL_AFTERTOUCH)) @@ -844,137 +987,198 @@ static int pm_status_filtered(int status, long filters) || ((status == MIDI_PITCHBEND) && (filters & PM_FILT_PITCHBEND)); } +*/ + +void pm_flush_sysex(PmInternal *midi, PmTimestamp timestamp) +{ + PmEvent event; + + /* there may be nothing in the buffer */ + if (midi->sysex_message_count == 0) return; /* nothing to flush */ + + event.message = midi->sysex_message; + event.timestamp = timestamp; + /* copied from pm_read_short, avoids filtering */ + if (Pm_Enqueue(midi->queue, &event) == pmBufferOverflow) { + midi->sysex_in_progress = FALSE; + } + midi->sysex_message_count = 0; + midi->sysex_message = 0; +} -/* pm_read_short and pm_read_byte + +/* pm_read_short and pm_read_bytes are the interface between system-dependent MIDI input handlers and the system-independent PortMIDI code. The input handler MUST obey these rules: 1) all short input messages must be sent to pm_read_short, which enqueues them to a FIFO for the application. - 2) eash sysex byte should be reported by calling pm_read_byte - (which sets midi->sysex_in_progress). After the eox byte, - pm_read_byte will clear sysex_in_progress and midi->flush - (Note that the overflow flag is managed by pm_read_short - and Pm_Read, so the supplier should not read or write it.) + 2) eash buffer of sysex bytes should be reported by calling pm_read_bytes + (which sets midi->sysex_in_progress). After the eox byte, + pm_read_bytes will clear sysex_in_progress */ -/* pm_read_short is the place where all input messages arrive from +/* pm_read_short is the place where all input messages arrive from system-dependent code such as pmwinmm.c. Here, the messages - are entered into the PortMidi input buffer. + are entered into the PortMidi input buffer. */ - - /* Algorithnm: - if overflow or flush, return - ATOMIC: - enqueue data - if buffer overflow, set overflow - if buffer overflow and sysex_in_progress, set flush - */ void pm_read_short(PmInternal *midi, PmEvent *event) -{ +{ +#ifndef NEWBUFFER long tail; +#endif int status; /* arg checking */ assert(midi != NULL); - assert(!Pm_HasHostError(midi)); /* midi filtering is applied here */ status = Pm_MessageStatus(event->message); if (!pm_status_filtered(status, midi->filters) - && !pm_realtime_filtered(status, midi->filters) + && (!is_real_time(status) || + !pm_realtime_filtered(status, midi->filters)) && !pm_channel_filtered(status, midi->channel_mask)) { /* if sysex is in progress and we get a status byte, it had better be a realtime message or the starting SYSEX byte; otherwise, we exit the sysex_in_progress state */ - if (midi->sysex_in_progress && (status & MIDI_STATUS_MASK) && - !is_real_time(status) && status != MIDI_SYSEX ) { + if (midi->sysex_in_progress && (status & MIDI_STATUS_MASK)) { + /* two choices: real-time or not. If it's real-time, then + * this should be delivered as a sysex byte because it is + * embedded in a sysex message + */ + if (is_real_time(status)) { + midi->sysex_message |= + (status << (8 * midi->sysex_message_count++)); + if (midi->sysex_message_count == 4) { + pm_flush_sysex(midi, event->timestamp); + } + } else { /* otherwise, it's not real-time. This interrupts + * a sysex message in progress */ + midi->sysex_in_progress = FALSE; + } +#ifdef NEWBUFFER + } else if (Pm_Enqueue(midi->queue, event) == pmBufferOverflow) { midi->sysex_in_progress = FALSE; - midi->flush = FALSE; } - - /* don't try to do anything more in an overflow state */ - if (midi->overflow || midi->flush) return; - - /* insert the message */ - tail = midi->tail; - midi->buffer[tail++] = *event; - if (tail == midi->buffer_len) tail = 0; - if (tail == midi->head || midi->overflow) { - midi->overflow = TRUE; - if (midi->sysex_in_progress) midi->flush = TRUE; - /* drop the rest of the message, this must be cleared - by caller when EOX is received */ - return; +#else + } else { + /* don't try to do anything more in an overflow state */ + if (midi->overflow) return; + + /* insert the message */ + tail = midi->tail; + midi->buffer[tail++] = *event; + if (tail == midi->buffer_len) tail = 0; + if (tail == midi->head || midi->overflow) { + midi->overflow = TRUE; + midi->sysex_in_progress = FALSE; + /* drop the rest of the message, this must be cleared + by caller when EOX is received */ + return; + } + midi->tail = tail; /* complete the write */ } - midi->tail = tail; /* complete the write */ +#endif } } - -void pm_flush_sysex(PmInternal *midi, PmTimestamp timestamp) +/* pm_read_bytes -- read one (partial) sysex msg from MIDI data */ +/* + * returns how many bytes processed + */ +unsigned int pm_read_bytes(PmInternal *midi, unsigned char *data, + int len, PmTimestamp timestamp) { + unsigned int i = 0; /* index into data */ PmEvent event; - - /* there may be nothing in the buffer */ - if (midi->sysex_message_count == 0) return; /* nothing to flush */ - - event.message = midi->sysex_message; event.timestamp = timestamp; - pm_read_short(midi, &event); - midi->sysex_message_count = 0; - midi->sysex_message = 0; -} - - -void pm_read_byte(PmInternal *midi, unsigned char byte, PmTimestamp timestamp) -{ assert(midi); - assert(!Pm_HasHostError(midi)); - /* here is the logic for controlling sysex_in_progress */ - if (midi->sysex_in_progress) { - if (byte == MIDI_EOX) midi->sysex_in_progress = FALSE; - else if (byte == MIDI_SYSEX) { - /* problem: need to terminate the current sysex and start - a new one - */ - pm_flush_sysex(midi, timestamp); - } - } else if (byte == MIDI_SYSEX) { - midi->sysex_in_progress = TRUE; - } else { - /* error: we're getting data bytes or EOX but we're no sysex is - in progress. Drop the data. (Would it be better to report an - error? Is this a host error or a pmBadData error? Is this - ever possible? - */ -#ifdef DEBUG - printf("PortMidi debug msg: unexpected sysex data or EOX\n"); -#endif - return; + /* note that since buffers may not have multiples of 4 bytes, + * pm_read_bytes may be called in the middle of an outgoing + * 4-byte PortMidi message. sysex_in_progress indicates that + * a sysex has been sent but no eox. + */ + if (len == 0) return 0; /* sanity check */ + if (!midi->sysex_in_progress) { + while (i < len) { /* process all data */ + unsigned char byte = data[i++]; + if (byte == MIDI_SYSEX && + !pm_realtime_filtered(byte, midi->filters)) { + midi->sysex_in_progress = TRUE; + i--; /* back up so code below will get SYSEX byte */ + break; /* continue looping below to process msg */ + } else if (byte == MIDI_EOX) { + midi->sysex_in_progress = FALSE; + return i; /* done with one message */ + } else if (byte & MIDI_STATUS_MASK) { + /* We're getting MIDI but no sysex in progress. + * Either the SYSEX status byte was dropped or + * the message was filtered. Drop the data, but + * send any embedded realtime bytes. + */ + /* assume that this is a real-time message: + * it is an error to pass non-real-time messages + * to pm_read_bytes + */ + event.message = byte; + pm_read_short(midi, &event); + } + } /* all bytes in the buffer are processed */ } - - if (pm_realtime_filtered(byte, midi->filters)) - return; - /* this awkward expression places the bytes in increasingly higher- - order bytes of the long message */ - midi->sysex_message |= (byte << (8 * midi->sysex_message_count++)); - if (midi->sysex_message_count == 4 || !midi->sysex_in_progress) { - pm_flush_sysex(midi, timestamp); - if (!midi->sysex_in_progress) midi->flush = FALSE; + /* Now, isysex_in_progress) { + if (midi->sysex_message_count == 0 && i <= len - 4 && + ((event.message = (((long) data[i]) | + (((long) data[i+1]) << 8) | + (((long) data[i+2]) << 16) | + (((long) data[i+3]) << 24))) & + 0x80808080) == 0) { /* all data, no status */ + if (Pm_Enqueue(midi->queue, &event) == pmBufferOverflow) { + midi->sysex_in_progress = FALSE; + } + i += 4; + } else { + while (i < len) { + /* send one byte at a time */ + unsigned char byte = data[i++]; + if (is_real_time(byte) && + pm_realtime_filtered(byte, midi->filters)) { + continue; /* real-time data is filtered, so omit */ + } + midi->sysex_message |= + (byte << (8 * midi->sysex_message_count++)); + if (byte == MIDI_EOX) { + midi->sysex_in_progress = FALSE; + pm_flush_sysex(midi, event.timestamp); + return i; + } else if (midi->sysex_message_count == 4) { + pm_flush_sysex(midi, event.timestamp); + /* after handling at least one non-data byte + * and reaching a 4-byte message boundary, + * resume trying to send 4 at a time in outer loop + */ + break; + } + } + } } + return i; } +#ifndef NEWBUFFER +/* this code is apparently never called */ int pm_queue_full(PmInternal *midi) { long tail; /* arg checking */ assert(midi != NULL); - assert(!Pm_HasHostError(midi)); - + tail = midi->tail + 1; if (tail == midi->buffer_len) tail = 0; return tail == midi->head; } - +#endif diff --git a/pd/portmidi/pm_common/portmidi.h b/pd/portmidi/pm_common/portmidi.h index 8c9c04c2..8f4879f3 100644 --- a/pd/portmidi/pm_common/portmidi.h +++ b/pd/portmidi/pm_common/portmidi.h @@ -10,7 +10,12 @@ extern "C" { * Latest version available at: http://www.cs.cmu.edu/~music/portmidi/ * * Copyright (c) 1999-2000 Ross Bencina and Phil Burk - * Copyright (c) 2001 Roger B. Dannenberg + * Copyright (c) 2001-2006 Roger B. Dannenberg + * + * Latest version available at: http://www.cs.cmu.edu/~music/portmidi/ + * + * Copyright (c) 1999-2000 Ross Bencina and Phil Burk + * Copyright (c) 2001-2006 Roger B. Dannenberg * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files @@ -23,10 +28,6 @@ extern "C" { * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * - * Any person wishing to distribute modifications to the Software is - * requested to send the modifications to the original developer so that - * they can be incorporated into the canonical version. - * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. @@ -34,114 +35,21 @@ extern "C" { * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + */ + +/* + * The text above constitutes the entire PortMidi license; however, + * the PortMusic community also makes the following non-binding requests: * + * Any person wishing to distribute modifications to the Software is + * requested to send the modifications to the original developer so that + * they can be incorporated into the canonical version. It is also + * requested that these non-binding requests be included along with the + * license above. */ /* CHANGELOG FOR PORTMIDI - * - * 15Nov04 Ben Allison - * - sysex output now uses one buffer/message and reallocates buffer - * - if needed - * - filters expanded for many message types and channels - * - detailed changes are as follows: - * ------------- in pmwinmm.c -------------- - * - new #define symbol: OUTPUT_BYTES_PER_BUFFER - * - change SYSEX_BYTES_PER_BUFFER to 1024 - * - added MIDIHDR_BUFFER_LENGTH(x) to correctly count midihdr buffer length - * - change MIDIHDR_SIZE(x) to (MIDIHDR_BUFFER_LENGTH(x) + sizeof(MIDIHDR)) - * - change allocate_buffer to use new MIDIHDR_BUFFER_LENGTH macro - * - new macros for MIDIHDR_SYSEX_SIZE and MIDIHDR_SYSEX_BUFFER_LENGTH - * - similar to above, but counts appropriately for sysex messages - * - added the following members to midiwinmm_struct for sysex data: - * - LPMIDIHDR *sysex_buffers; ** pool of buffers for sysex data ** - * - int num_sysex_buffers; ** how many sysex buffers ** - * - int next_sysex_buffer; ** index of next sysexbuffer to send ** - * - HANDLE sysex_buffer_signal; ** to wait for free sysex buffer ** - * - duplicated allocate_buffer, alocate_buffers and get_free_output_buffer - * - into equivalent sysex_buffer form - * - changed winmm_in_open to initialize new midiwinmm_struct members and - * - to use the new allocate_sysex_buffer() function instead of - * - allocate_buffer() - * - changed winmm_out_open to initialize new members, create sysex buffer - * - signal, and allocate 2 sysex buffers - * - changed winmm_out_delete to free sysex buffers and shut down the sysex - * - buffer signal - * - create new function resize_sysex_buffer which resizes m->hdr to the - * - passed size, and corrects the midiwinmm_struct accordingly. - * - changed winmm_write_byte to use new resize_sysex_buffer function, - * - if resize fails, write current buffer to output and continue - * - changed winmm_out_callback to use buffer_signal or sysex_buffer_signal - * - depending on which buffer was finished - * ------------- in portmidi.h -------------- - * - added pmBufferMaxSize to PmError to indicate that the buffer would be - * - too large for the underlying API - * - added additional filters - * - added prototype, documentation, and helper macro for Pm_SetChannelMask - * ------------- in portmidi.c -------------- - * - added pm_status_filtered() and pm_realtime_filtered() functions to - * separate filtering logic from buffer logic in pm_read_short - * - added Pm_SetChannelMask function - * - added pm_channel_filtered() function - * ------------- in pminternal.h -------------- - * - added member to PortMidiStream for channel mask - * - * 25May04 RBD - * - removed support for MIDI THRU - * - moved filtering from Pm_Read to pm_enqueue to avoid buffer ovfl - * - extensive work on Mac OS X port, especially sysex and error handling - * - * 18May04 RBD - * - removed side-effects from assert() calls. Now you can disable assert(). - * - no longer check pm_hosterror everywhere, fixing a bug where an open - * failure could cause a write not to work on a previously opened port - * until you call Pm_GetHostErrorText(). - * 16May04 RBD and Chris Roberts - * - Some documentation wordsmithing in portmidi.h - * - Dynamically allocate port descriptor structures - * - Fixed parameter error in midiInPrepareBuffer and midiInAddBuffer. - * - * 09Oct03 RBD - * - Changed Thru handling. Now the client does all the work and the client - * must poll or read to keep thru messages flowing. - * - * 31May03 RBD - * - Fixed various bugs. - * - Added linux ALSA support with help from Clemens Ladisch - * - Added Mac OS X support, implemented by Jon Parise, updated and - * integrated by Andrew Zeldis and Zico Kolter - * - Added latency program to build histogram of system latency using PortTime. - * - * 30Jun02 RBD Extensive rewrite of sysex handling. It works now. - * Extensive reworking of error reporting and error text -- no - * longer use dictionary call to delete data; instead, Pm_Open - * and Pm_Close clean up before returning an error code, and - * error text is saved in a system-independent location. - * Wrote sysex.c to test sysex message handling. - * - * 15Jun02 BCT changes: - * - Added pmHostError text handling. - * - For robustness, check PortMidi stream args not NULL. - * - Re-C-ANSI-fied code (changed many C++ comments to C style) - * - Reorganized code in pmwinmm according to input/output functionality (made - * cleanup handling easier to reason about) - * - Fixed Pm_Write calls (portmidi.h says these should not return length but Pm_Error) - * - Cleaned up memory handling (now system specific data deleted via dictionary - * call in PortMidi, allows client to query host errors). - * - Added explicit asserts to verify various aspects of pmwinmm implementation behaves as - * logic implies it should. Specifically: verified callback routines not reentrant and - * all verified status for all unchecked Win32 MMedia API calls perform successfully - * - Moved portmidi initialization and clean-up routines into DLL to fix Win32 MMedia API - * bug (i.e. if devices not explicitly closed, must reboot to debug application further). - * With this change, clients no longer need explicitly call Pm_Initialize, Pm_Terminate, or - * explicitly Pm_Close open devices when using WinMM version of PortMidi. - * - * 23Jan02 RBD Fixed bug in pmwinmm.c thru handling - * - * 21Jan02 RBD Added tests in Pm_OpenInput() and Pm_OpenOutput() to prevent - * opening an input as output and vice versa. - * Added comments and documentation. - * Implemented Pm_Terminate(). - * + * (see ../CHANGELOG.txt) * * IMPORTANT INFORMATION ABOUT A WIN32 BUG: * @@ -149,15 +57,19 @@ extern "C" { * may crash. PortMidi tries to protect against this by using a DLL to clean up. * * If client exits for example with: - * i) assert - * ii) Ctrl^c, - * then DLL clean-up routine called. However, when client does something + * i) assert + * ii) Ctrl^c, + * then DLL clean-up routine called. However, when client does something * really bad (e.g. assigns value to NULL pointer) then DLL CLEANUP ROUTINE * NEVER RUNS! In this state, if you wait around long enough, you will * probably get the blue screen of death. Can also go into Pview and there will * exist zombie process that you can't kill. * - * NOTES ON HOST ERROR REPORTING: + * You can enable the DLL cleanup routine by defining USE_DLL_FOR_CLEANUP. + * Do not define this preprocessor symbol if you do not want to use this + * feature. + * + * NOTES ON HOST ERROR REPORTING: * * PortMidi errors (of type PmError) are generic, system-independent errors. * When an error does not map to one of the more specific PmErrors, the @@ -165,50 +77,45 @@ extern "C" { * retained a more specific system-dependent error code. The caller can * get more information by calling Pm_HasHostError() to test if there is * a pending host error, and Pm_GetHostErrorText() to get a text string - * describing the error. Host errors are reported on a per-device basis - * because only after you open a device does PortMidi have a place to - * record the host error code. I.e. only - * those routines that receive a (PortMidiStream *) argument check and - * report errors. One exception to this is that Pm_OpenInput() and + * describing the error. Host errors are reported on a per-device basis + * because only after you open a device does PortMidi have a place to + * record the host error code. I.e. only + * those routines that receive a (PortMidiStream *) argument check and + * report errors. One exception to this is that Pm_OpenInput() and * Pm_OpenOutput() can report errors even though when an error occurs, * there is no PortMidiStream* to hold the error. Fortunately, both * of these functions return any error immediately, so we do not really * need per-device error memory. Instead, any host error code is stored - * in a global, pmHostError is returned, and the user can call + * in a global, pmHostError is returned, and the user can call * Pm_GetHostErrorText() to get the error message (and the invalid stream - * parameter will be ignored.) The functions + * parameter will be ignored.) The functions * pm_init and pm_term do not fail or raise * errors. The job of pm_init is to locate all available devices so that * the caller can get information via PmDeviceInfo(). If an error occurs, * the device is simply not listed as available. * * Host errors come in two flavors: - * a) host error + * a) host error * b) host error during callback - * These can occur w/midi input or output devices. (b) can only happen + * These can occur w/midi input or output devices. (b) can only happen * asynchronously (during callback routines), whereas (a) only occurs while - * synchronously running PortMidi and any resulting system dependent calls + * synchronously running PortMidi and any resulting system dependent calls. + * Both (a) and (b) are reported by the next read or write call. You can + * also query for asynchronous errors (b) at any time by calling + * Pm_HasHostError(). * - * Host-error reporting relies on following assumptions: - * 1) PortMidi routines won't allow system dependent routines to be - * called when args are bogus. - * Thus, in pmwinmm.c it is safe to assume: - * - stream ptr valid - * - currently not operating in "has host error" state - * 2) Host-error reporting relies on a staged delivery of error messages. - * When a host error occurs, the error code is saved with the stream. - * The error is reported as a return code from the next operation on - * the stream. This could be immediately if the error is synchronous, - * or delayed if the error is an asynchronous callback problem. In - * any case, when pmHostError is returned, the error is copied to - * a global, pm_hosterror and the error code stored with the stream - * is cleared. If the user chooses to inquire about the error using - * Pm_GetHostErrorText(), the error will be reported as text. If the - * user ignores the error and makes another call on the stream, the - * call will proceed because the error code associated with the stream - * has been cleared. + * NOTES ON COMPILE-TIME SWITCHES * - */ + * DEBUG assumes stdio and a console. Use this if you want automatic, simple + * error reporting, e.g. for prototyping. If you are using MFC or some + * other graphical interface with no console, DEBUG probably should be + * undefined. + * PM_CHECK_ERRORS more-or-less takes over error checking for return values, + * stopping your program and printing error messages when an error + * occurs. This also uses stdio for console text I/O. + * USE_DLL_FOR_CLEANUP is described above. (Windows only.) + * + */ #ifndef FALSE #define FALSE 0 @@ -224,14 +131,19 @@ extern "C" { typedef enum { pmNoError = 0, pmHostError = -10000, - pmInvalidDeviceId, /* out of range or output device when input is requested or vice versa */ + pmInvalidDeviceId, /* out of range or + * output device when input is requested or + * input device when output is requested or + * device is already opened + */ pmInsufficientMemory, pmBufferTooSmall, pmBufferOverflow, pmBadPtr, pmBadData, /* illegal midi data, e.g. missing EOX */ pmInternalError, - pmBufferMaxSize, /* buffer is already as large as it can be */ + pmBufferMaxSize /* buffer is already as large as it can be */ + /* NOTE: If you add a new error type, be sure to update Pm_GetErrorText() */ } PmError; /* @@ -255,32 +167,35 @@ typedef void PortMidiStream; /* Test whether stream has a pending host error. Normally, the client finds - out about errors through returned error codes, but some errors can occur - asynchronously where the client does not - explicitly call a function, and therefore cannot receive an error code. - The client can test for a pending error using Pm_HasHostError(). If true, - the error can be accessed and cleared by calling Pm_GetErrorText(). The - client does not need to call Pm_HasHostError(). Any pending error will be - reported the next time the client performs an explicit function call on - the stream, e.g. an input or output operation. + out about errors through returned error codes, but some errors can occur + asynchronously where the client does not + explicitly call a function, and therefore cannot receive an error code. + The client can test for a pending error using Pm_HasHostError(). If true, + the error can be accessed and cleared by calling Pm_GetErrorText(). + Errors are also cleared by calling other functions that can return + errors, e.g. Pm_OpenInput(), Pm_OpenOutput(), Pm_Read(), Pm_Write(). The + client does not need to call Pm_HasHostError(). Any pending error will be + reported the next time the client performs an explicit function call on + the stream, e.g. an input or output operation. Until the error is cleared, + no new error codes will be obtained, even for a different stream. */ int Pm_HasHostError( PortMidiStream * stream ); /* Translate portmidi error number into human readable message. - These strings are constants (set at compile time) so client has - no need to allocate storage + These strings are constants (set at compile time) so client has + no need to allocate storage */ const char *Pm_GetErrorText( PmError errnum ); /* Translate portmidi host error into human readable message. These strings are computed at run time, so client has to allocate storage. - After this routine executes, the host error is cleared. + After this routine executes, the host error is cleared. */ void Pm_GetHostErrorText(char * msg, unsigned int len); #define HDRLENGTH 50 -#define PM_HOST_ERROR_MSG_LEN 256u /* any host error msg will occupy less +#define PM_HOST_ERROR_MSG_LEN 256u /* any host error msg will occupy less than this number of characters */ /* @@ -292,7 +207,7 @@ void Pm_GetHostErrorText(char * msg, unsigned int len); typedef int PmDeviceID; #define pmNoDevice -1 typedef struct { - int structVersion; + int structVersion; const char *interf; /* underlying MIDI API, e.g. MMSystem or DirectX */ const char *name; /* device name, e.g. USB MidiSport 1x1 */ int input; /* true iff input is available */ @@ -308,12 +223,12 @@ int Pm_CountDevices( void ); Return the default device ID or pmNoDevice if there are no devices. The result can be passed to Pm_OpenMidi(). - + On the PC, the user can specify a default device by setting an environment variable. For example, to use device #1. set PM_RECOMMENDED_OUTPUT_DEVICE=1 - + The user should first determine the available device ID by using the supplied application "testin" or "testout". @@ -327,25 +242,25 @@ int Pm_CountDevices( void ); and HKEY_LOCAL_MACHINE/SOFTWARE/PortMidi/Recommended_Output_Device for a string. The number of the first device with a substring that matches the string exactly is returned. For example, if the string - in the registry is "USB", and device 1 is named + in the registry is "USB", and device 1 is named "In USB MidiSport 1x1", then that will be the default input because it contains the string "USB". In addition to the name, PmDeviceInfo has the member "interf", which is the interface name. (The "interface" is the underlying software - system or API used by PortMidi to access devices. Examples are - MMSystem, DirectX (not implemented), ALSA, OSS (not implemented), etc.) - At present, the only Win32 interface is "MMSystem", the only Linux - interface is "ALSA", and the only Max OS X interface is "CoreMIDI". + system or API used by PortMidi to access devices. Examples are + MMSystem, DirectX (not implemented), ALSA, OSS (not implemented), etc.) + At present, the only Win32 interface is "MMSystem", the only Linux + interface is "ALSA", and the only Max OS X interface is "CoreMIDI". To specify both the interface and the device name in the registry, separate the two with a comma and a space, e.g.: MMSystem, In USB MidiSport 1x1 In this case, the string before the comma must be a substring of - the "interf" string, and the string after the space must be a + the "interf" string, and the string after the space must be a substring of the "name" name string in order to match the device. Note: in the current release, the default is simply the first device - (the input or output device with the lowest PmDeviceID). + (the input or output device with the lowest PmDeviceID). */ PmDeviceID Pm_GetDefaultInputDeviceID( void ); PmDeviceID Pm_GetDefaultOutputDeviceID( void ); @@ -391,26 +306,26 @@ const PmDeviceInfo* Pm_GetDeviceInfo( PmDeviceID id ); outputDriverInfo is never required for correct operation. If not used outputDriverInfo should be NULL. - For input, the buffersize specifies the number of input events to be - buffered waiting to be read using Pm_Read(). For output, buffersize - specifies the number of output events to be buffered waiting for output. + For input, the buffersize specifies the number of input events to be + buffered waiting to be read using Pm_Read(). For output, buffersize + specifies the number of output events to be buffered waiting for output. (In some cases -- see below -- PortMidi does not buffer output at all - and merely passes data to a lower-level API, in which case buffersize - is ignored.) - - latency is the delay in milliseconds applied to timestamps to determine - when the output should actually occur. (If latency is < 0, 0 is assumed.) + and merely passes data to a lower-level API, in which case buffersize + is ignored.) + + latency is the delay in milliseconds applied to timestamps to determine + when the output should actually occur. (If latency is < 0, 0 is assumed.) If latency is zero, timestamps are ignored and all output is delivered immediately. If latency is greater than zero, output is delayed until the message timestamp plus the latency. (NOTE: time is measured relative to the time source indicated by time_proc. Timestamps are absolute, not relative delays or offsets.) In some cases, PortMidi can obtain - better timing than your application by passing timestamps along to the - device driver or hardware. Latency may also help you to synchronize midi - data to audio data by matching midi latency to the audio buffer latency. + better timing than your application by passing timestamps along to the + device driver or hardware. Latency may also help you to synchronize midi + data to audio data by matching midi latency to the audio buffer latency. time_proc is a pointer to a procedure that returns time in milliseconds. It - may be NULL, in which case a default millisecond timebase (PortTime) is + may be NULL, in which case a default millisecond timebase (PortTime) is used. If the application wants to use PortTime, it should start the timer (call Pt_Start) before calling Pm_OpenInput or Pm_OpenOutput. If the application tries to start the timer *after* Pm_OpenInput or Pm_OpenOutput, @@ -419,7 +334,7 @@ const PmDeviceInfo* Pm_GetDeviceInfo( PmDeviceID id ); time_proc result values are appended to incoming MIDI data, and time_proc times are used to schedule outgoing MIDI data (when latency is non-zero). - time_info is a pointer passed to time_proc. + time_info is a pointer passed to time_proc. return value: Upon success Pm_Open() returns PmNoError and places a pointer to a @@ -428,7 +343,7 @@ const PmDeviceInfo* Pm_GetDeviceInfo( PmDeviceID id ); PMError above) and the value of port is invalid. Any stream that is successfully opened should eventually be closed - by calling Pm_Close(). + by calling Pm_Close(). */ PmError Pm_OpenInput( PortMidiStream** stream, @@ -446,7 +361,7 @@ PmError Pm_OpenOutput( PortMidiStream** stream, void *time_info, long latency ); -/* +/* Pm_SetFilter() sets filters on an open input stream to drop selected input types. By default, only active sensing messages are filtered. To prohibit, say, active sensing and sysex messages, call @@ -461,53 +376,52 @@ PmError Pm_OpenOutput( PortMidiStream** stream, */ /* filter active sensing messages (0xFE): */ -#define PM_FILT_ACTIVE 0x1 +#define PM_FILT_ACTIVE (1 << 0x0E) /* filter system exclusive messages (0xF0): */ -#define PM_FILT_SYSEX 0x2 -/* filter clock messages (0xF8 only, does not filter clock start, etc.): */ -#define PM_FILT_CLOCK 0x4 +#define PM_FILT_SYSEX (1 << 0x00) +/* filter clock messages (CLOCK 0xF8, START 0xFA, STOP 0xFC, and CONTINUE 0xFB) */ +#define PM_FILT_CLOCK ((1 << 0x08) | (1 << 0x0A) | (1 << 0x0C) | (1 << 0x0B)) /* filter play messages (start 0xFA, stop 0xFC, continue 0xFB) */ -#define PM_FILT_PLAY 0x8 -/* filter undefined F9 messages (some equipment uses this as a 10ms 'tick') */ -#define PM_FILT_F9 0x10 -#define PM_FILT_TICK PM_FILT_F9 +#define PM_FILT_PLAY (1 << 0x0A) +/* filter tick messages (0xF9) */ +#define PM_FILT_TICK (1 << 0x09) /* filter undefined FD messages */ -#define PM_FILT_FD 0x20 +#define PM_FILT_FD (1 << 0x0D) /* filter undefined real-time messages */ -#define PM_FILT_UNDEFINED (PM_FILT_F9 | PM_FILT_FD) +#define PM_FILT_UNDEFINED PM_FILT_FD /* filter reset messages (0xFF) */ -#define PM_FILT_RESET 0x40 +#define PM_FILT_RESET (1 << 0x0F) /* filter all real-time messages */ -#define PM_FILT_REALTIME (PM_FILT_ACTIVE | PM_FILT_SYSEX | PM_FILT_CLOCK | PM_FILT_PLAY | PM_FILT_UNDEFINED | PM_FILT_RESET) +#define PM_FILT_REALTIME (PM_FILT_ACTIVE | PM_FILT_SYSEX | PM_FILT_CLOCK | \ + PM_FILT_PLAY | PM_FILT_UNDEFINED | PM_FILT_RESET | PM_FILT_TICK) /* filter note-on and note-off (0x90-0x9F and 0x80-0x8F */ -#define PM_FILT_NOTE 0x80 +#define PM_FILT_NOTE ((1 << 0x19) | (1 << 0x18)) /* filter channel aftertouch (most midi controllers use this) (0xD0-0xDF)*/ -#define PM_FILT_CHANNEL_AFTERTOUCH 0x100 -/* per-note aftertouch (Ensoniq holds a patent on generating this on keyboards until June 2006) (0xA0-0xAF) */ -#define PM_FILT_POLY_AFTERTOUCH 0x200 +#define PM_FILT_CHANNEL_AFTERTOUCH (1 << 0x1D) +/* per-note aftertouch (0xA0-0xAF) */ +#define PM_FILT_POLY_AFTERTOUCH (1 << 0x1A) /* filter both channel and poly aftertouch */ #define PM_FILT_AFTERTOUCH (PM_FILT_CHANNEL_AFTERTOUCH | PM_FILT_POLY_AFTERTOUCH) /* Program changes (0xC0-0xCF) */ -#define PM_FILT_PROGRAM 0x400 +#define PM_FILT_PROGRAM (1 << 0x1C) /* Control Changes (CC's) (0xB0-0xBF)*/ -#define PM_FILT_CONTROL 0x800 +#define PM_FILT_CONTROL (1 << 0x1B) /* Pitch Bender (0xE0-0xEF*/ -#define PM_FILT_PITCHBEND 0x1000 +#define PM_FILT_PITCHBEND (1 << 0x1E) /* MIDI Time Code (0xF1)*/ -#define PM_FILT_MTC 0x2000 +#define PM_FILT_MTC (1 << 0x01) /* Song Position (0xF2) */ -#define PM_FILT_SONG_POSITION 0x4000 +#define PM_FILT_SONG_POSITION (1 << 0x02) /* Song Select (0xF3)*/ -#define PM_FILT_SONG_SELECT 0x8000 +#define PM_FILT_SONG_SELECT (1 << 0x03) /* Tuning request (0xF6)*/ -#define PM_FILT_TUNE 0x10000 +#define PM_FILT_TUNE (1 << 0x06) /* All System Common messages (mtc, song position, song select, tune request) */ #define PM_FILT_SYSTEMCOMMON (PM_FILT_MTC | PM_FILT_SONG_POSITION | PM_FILT_SONG_SELECT | PM_FILT_TUNE) PmError Pm_SetFilter( PortMidiStream* stream, long filters ); - /* Pm_SetChannelMask() filters incoming messages based on channel. The mask is a 16-bit bitfield corresponding to appropriate channels @@ -532,11 +446,11 @@ PmError Pm_SetChannelMask(PortMidiStream *stream, int mask); any time. */ PmError Pm_Abort( PortMidiStream* stream ); - + /* Pm_Close() closes a midi stream, flushing any pending buffers. - (PortMidi attempts to close open streams when the application - exits -- this is particularly difficult under Windows.) + (PortMidi attempts to close open streams when the application + exits -- this is particularly difficult under Windows.) */ PmError Pm_Close( PortMidiStream* stream ); @@ -544,7 +458,7 @@ PmError Pm_Close( PortMidiStream* stream ); Pm_Message() encodes a short Midi message into a long word. If data1 and/or data2 are not present, use zero. - Pm_MessageStatus(), Pm_MessageData1(), and + Pm_MessageStatus(), Pm_MessageData1(), and Pm_MessageData2() extract fields from a long-encoded midi message. */ #define Pm_Message(status, data1, data2) \ @@ -560,61 +474,61 @@ PmError Pm_Close( PortMidiStream* stream ); structure carrying 4 bytes of the message, i.e. only the first PmEvent carries the status byte. - Note that MIDI allows nested messages: the so-called "real-time" MIDI - messages can be inserted into the MIDI byte stream at any location, + Note that MIDI allows nested messages: the so-called "real-time" MIDI + messages can be inserted into the MIDI byte stream at any location, including within a sysex message. MIDI real-time messages are one-byte - messages used mainly for timing (see the MIDI spec). PortMidi retains - the order of non-real-time MIDI messages on both input and output, but + messages used mainly for timing (see the MIDI spec). PortMidi retains + the order of non-real-time MIDI messages on both input and output, but it does not specify exactly how real-time messages are processed. This - is particulary problematic for MIDI input, because the input parser - must either prepare to buffer an unlimited number of sysex message - bytes or to buffer an unlimited number of real-time messages that + is particulary problematic for MIDI input, because the input parser + must either prepare to buffer an unlimited number of sysex message + bytes or to buffer an unlimited number of real-time messages that arrive embedded in a long sysex message. To simplify things, the input - parser is allowed to pass real-time MIDI messages embedded within a - sysex message, and it is up to the client to detect, process, and + parser is allowed to pass real-time MIDI messages embedded within a + sysex message, and it is up to the client to detect, process, and remove these messages as they arrive. When receiving sysex messages, the sysex message is terminated by either an EOX status byte (anywhere in the 4 byte messages) or by a non-real-time status byte in the low order byte of the message. - If you get a non-real-time status byte but there was no EOX byte, it + If you get a non-real-time status byte but there was no EOX byte, it means the sysex message was somehow truncated. This is not considered an error; e.g., a missing EOX can result from the user disconnecting a MIDI cable during sysex transmission. - A real-time message can occur within a sysex message. A real-time - message will always occupy a full PmEvent with the status byte in + A real-time message can occur within a sysex message. A real-time + message will always occupy a full PmEvent with the status byte in the low-order byte of the PmEvent message field. (This implies that the byte-order of sysex bytes and real-time message bytes may not be preserved -- for example, if a real-time message arrives after 3 bytes of a sysex message, the real-time message will be delivered first. The first word of the sysex message will be delivered only after the 4th byte arrives, filling the 4-byte PmEvent message field. - + The timestamp field is observed when the output port is opened with a non-zero latency. A timestamp of zero means "use the current time", which in turn means to deliver the message with a delay of latency (the latency parameter used when opening the output port.) - Do not expect PortMidi to sort data according to timestamps -- - messages should be sent in the correct order, and timestamps MUST + Do not expect PortMidi to sort data according to timestamps -- + messages should be sent in the correct order, and timestamps MUST be non-decreasing. - A sysex message will generally fill many PmEvent structures. On + A sysex message will generally fill many PmEvent structures. On output to a PortMidiStream with non-zero latency, the first timestamp - on sysex message data will determine the time to begin sending the - message. PortMidi implementations may ignore timestamps for the - remainder of the sysex message. - - On input, the timestamp ideally denotes the arrival time of the - status byte of the message. The first timestamp on sysex message - data will be valid. Subsequent timestamps may denote - when message bytes were actually received, or they may be simply + on sysex message data will determine the time to begin sending the + message. PortMidi implementations may ignore timestamps for the + remainder of the sysex message. + + On input, the timestamp ideally denotes the arrival time of the + status byte of the message. The first timestamp on sysex message + data will be valid. Subsequent timestamps may denote + when message bytes were actually received, or they may be simply copies of the first timestamp. - Timestamps for nested messages: If a real-time message arrives in - the middle of some other message, it is enqueued immediately with - the timestamp corresponding to its arrival time. The interrupted - non-real-time message or 4-byte packet of sysex data will be enqueued + Timestamps for nested messages: If a real-time message arrives in + the middle of some other message, it is enqueued immediately with + the timestamp corresponding to its arrival time. The interrupted + non-real-time message or 4-byte packet of sysex data will be enqueued later. The timestamp of interrupted data will be equal to that of the interrupting real-time message to insure that timestamps are non-decreasing. @@ -627,19 +541,19 @@ typedef struct { /* Pm_Read() retrieves midi data into a buffer, and returns the number - of events read. Result is a non-negative number unless an error occurs, + of events read. Result is a non-negative number unless an error occurs, in which case a PmError value will be returned. Buffer Overflow - The problem: if an input overflow occurs, data will be lost, ultimately - because there is no flow control all the way back to the data source. - When data is lost, the receiver should be notified and some sort of - graceful recovery should take place, e.g. you shouldn't resume receiving + The problem: if an input overflow occurs, data will be lost, ultimately + because there is no flow control all the way back to the data source. + When data is lost, the receiver should be notified and some sort of + graceful recovery should take place, e.g. you shouldn't resume receiving in the middle of a long sysex message. - With a lock-free fifo, which is pretty much what we're stuck with to - enable portability to the Mac, it's tricky for the producer and consumer + With a lock-free fifo, which is pretty much what we're stuck with to + enable portability to the Mac, it's tricky for the producer and consumer to synchronously reset the buffer and resume normal operation. Solution: the buffer managed by PortMidi will be flushed when an overflow @@ -652,20 +566,20 @@ typedef struct { PmError Pm_Read( PortMidiStream *stream, PmEvent *buffer, long length ); /* - Pm_Poll() tests whether input is available, + Pm_Poll() tests whether input is available, returning TRUE, FALSE, or an error value. */ PmError Pm_Poll( PortMidiStream *stream); -/* +/* Pm_Write() writes midi data from a buffer. This may contain: - - short messages - or + - short messages + or - sysex messages that are converted into a sequence of PmEvent structures, e.g. sending data from a file or forwarding them from midi input. - Use Pm_WriteSysEx() to write a sysex message stored as a contiguous + Use Pm_WriteSysEx() to write a sysex message stored as a contiguous array of bytes. Sysex data may contain embedded real-time messages. @@ -674,9 +588,9 @@ PmError Pm_Write( PortMidiStream *stream, PmEvent *buffer, long length ); /* Pm_WriteShort() writes a timestamped non-system-exclusive midi message. - Messages are delivered in order as received, and timestamps must be - non-decreasing. (But timestamps are ignored if the stream was opened - with latency = 0.) + Messages are delivered in order as received, and timestamps must be + non-decreasing. (But timestamps are ignored if the stream was opened + with latency = 0.) */ PmError Pm_WriteShort( PortMidiStream *stream, PmTimestamp when, long msg); -- cgit v1.2.1