smflite

changeset 0:4264abea8b06

smf-lite initial commit
author John Tsiombikas <nuclear@member.fsf.org>
date Thu, 26 Jan 2012 11:25:11 +0200 (2012-01-26)
parents
children 8e535ca4bb86
files .hgignore Makefile src/config.h src/fake_glib.c src/fake_glib.h src/smf.c src/smf.h src/smf_decode.c src/smf_load.c src/smf_private.h src/smf_save.c src/smf_tempo.c
diffstat 12 files changed, 4534 insertions(+), 0 deletions(-) [+]
line diff
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/.hgignore	Thu Jan 26 11:25:11 2012 +0200
     1.3 @@ -0,0 +1,6 @@
     1.4 +\.d$
     1.5 +\.o$
     1.6 +\.swp$
     1.7 +smflite.a$
     1.8 +^libsmflite.so
     1.9 +^smflite.dylib$
     2.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     2.2 +++ b/Makefile	Thu Jan 26 11:25:11 2012 +0200
     2.3 @@ -0,0 +1,61 @@
     2.4 +PREFIX = /usr/local
     2.5 +
     2.6 +src = $(wildcard src/*.c)
     2.7 +obj = $(src:.c=.o)
     2.8 +dep = $(obj:.o=.d)
     2.9 +
    2.10 +name = smflite
    2.11 +lib_a = lib$(name).a
    2.12 +
    2.13 +ifeq ($(shell uname -s), Darwin)
    2.14 +	lib_so = $(name).dylib
    2.15 +	shared = -dynamiclib
    2.16 +else
    2.17 +	abi = 0
    2.18 +	rev = 0
    2.19 +	devlink = lib$(name).so
    2.20 +	soname = $(devlink).$(abi)
    2.21 +	lib_so = $(soname).$(rev)
    2.22 +	shared = -shared -Wl,-soname=$(soname)
    2.23 +	pic = -fPIC
    2.24 +endif
    2.25 +
    2.26 +CFLAGS = -pedantic -Wall -g $(pic)
    2.27 +
    2.28 +.PHONY: all
    2.29 +all: $(lib_a) $(lib_so)
    2.30 +
    2.31 +$(lib_so): $(obj)
    2.32 +	$(CC) $(shared) -o $@ $(obj) $(LDFLAGS)
    2.33 +
    2.34 +$(lib_a): $(obj)
    2.35 +	$(AR) rcs $@ $(obj)
    2.36 +
    2.37 +-include $(dep)
    2.38 +
    2.39 +%.d: %.c
    2.40 +	@$(CPP) $(CFLAGS) $< -MM -MT $(@:.d=.o) >$@
    2.41 +
    2.42 +.PHONY: clean
    2.43 +clean:
    2.44 +	rm -f $(obj) $(lib_so) $(lib_a)
    2.45 +
    2.46 +.PHONY: install
    2.47 +install: $(lib_so) $(lib_a)
    2.48 +	mkdir -p $(INSTDIR)$(PREFIX)/include $(INSTDIR)$(PREFIX)/lib
    2.49 +	cp src/smf.h $(INSTDIR)$(PREFIX)/include/smf.h
    2.50 +	cp $(lib_a) $(INSTDIR)$(PREFIX)/lib/$(lib_a)
    2.51 +	cp $(lib_so) $(INSTDIR)$(PREFIX)/lib/$(lib_so)
    2.52 +	[ -n "$(soname)" ] && \
    2.53 +		cd $(INSTDIR)$(PREFIX)/lib && \
    2.54 +		ln -s $(lib_so) $(soname) && \
    2.55 +		ln -s $(soname) $(devlink) || true
    2.56 +
    2.57 +.PHONY: uninstall
    2.58 +uninstall:
    2.59 +	rm -f $(INSTDIR)$(PREFIX)/include/smf.h
    2.60 +	rm -f $(INSTDIR)$(PREFIX)/lib/$(lib_a)
    2.61 +	rm -f $(INSTDIR)$(PREFIX)/lib/$(lib_so)
    2.62 +	[ -n "$(soname)" ] && \
    2.63 +		rm -f $(INSTDIR)$(PREFIX)/lib/$(soname) && \
    2.64 +		rm -f $(INSTDIR)$(PREFIX)/lib/$(devlink) || true
     3.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     3.2 +++ b/src/config.h	Thu Jan 26 11:25:11 2012 +0200
     3.3 @@ -0,0 +1,6 @@
     3.4 +#ifndef CONFIG_H_
     3.5 +#define CONFIG_H_
     3.6 +
     3.7 +#define PACKAGE_VERSION	"1.3-lite"
     3.8 +
     3.9 +#endif	/* CONFIG_H_ */
     4.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     4.2 +++ b/src/fake_glib.c	Thu Jan 26 11:25:11 2012 +0200
     4.3 @@ -0,0 +1,126 @@
     4.4 +#include <stdio.h>
     4.5 +#include <stdlib.h>
     4.6 +#include <string.h>
     4.7 +#include <stdarg.h>
     4.8 +#include <assert.h>
     4.9 +#include "fake_glib.h"
    4.10 +
    4.11 +FakeGPtrArray *fg_ptr_array_new(void)
    4.12 +{
    4.13 +	FakeGPtrArray *arr;
    4.14 +
    4.15 +	if(!(arr = malloc(sizeof *arr))) {
    4.16 +		return 0;
    4.17 +	}
    4.18 +	arr->pdata = 0;
    4.19 +	arr->len = 0;
    4.20 +	return arr;
    4.21 +}
    4.22 +
    4.23 +void **fg_ptr_array_free(FakeGPtrArray *arr, int free_seg)
    4.24 +{
    4.25 +	void **res;
    4.26 +
    4.27 +	if(!arr) {
    4.28 +		return 0;
    4.29 +	}
    4.30 +
    4.31 +	if(free_seg) {
    4.32 +		free(arr->pdata);
    4.33 +		res = 0;
    4.34 +	} else {
    4.35 +		res = arr->pdata;
    4.36 +	}
    4.37 +	free(arr);
    4.38 +	return res;
    4.39 +}
    4.40 +
    4.41 +void fg_ptr_array_add(FakeGPtrArray *arr, void *data)
    4.42 +{
    4.43 +	int idx = arr->len++;
    4.44 +
    4.45 +	arr->pdata = realloc(arr->pdata, arr->len * sizeof *arr->pdata);
    4.46 +	assert(arr->pdata);
    4.47 +
    4.48 +	arr->pdata[idx] = data;
    4.49 +}
    4.50 +
    4.51 +int fg_ptr_array_remove(FakeGPtrArray *arr, void *data)
    4.52 +{
    4.53 +	int i;
    4.54 +
    4.55 +	for(i=0; i<arr->len; i++) {
    4.56 +		if(arr->pdata[i] == data) {
    4.57 +			fg_ptr_array_remove_index(arr, i);
    4.58 +			return 1;
    4.59 +		}
    4.60 +	}
    4.61 +
    4.62 +	return 0;
    4.63 +}
    4.64 +
    4.65 +void *fg_ptr_array_remove_index(FakeGPtrArray *arr, unsigned int idx)
    4.66 +{
    4.67 +	void *data = arr->pdata[idx];
    4.68 +	int rest = --arr->len - idx;
    4.69 +	if(rest > 0) {
    4.70 +		memmove(arr->pdata + idx, arr->pdata + idx + 1, rest * sizeof *arr->pdata);
    4.71 +	}
    4.72 +	return data;
    4.73 +}
    4.74 +
    4.75 +void fg_ptr_array_sort(FakeGPtrArray *arr, FakeGCompareFunc cmp)
    4.76 +{
    4.77 +	qsort(arr->pdata, arr->len, sizeof *arr->pdata, cmp);
    4.78 +}
    4.79 +
    4.80 +/* -- logging -- */
    4.81 +
    4.82 +void fg_warning(const char *fmt, ...)
    4.83 +{
    4.84 +	va_list ap;
    4.85 +
    4.86 +	printf("warning: ");
    4.87 +
    4.88 +	va_start(ap, fmt);
    4.89 +	vprintf(fmt, ap);
    4.90 +	va_end(ap);
    4.91 +}
    4.92 +
    4.93 +void fg_critical(const char *fmt, ...)
    4.94 +{
    4.95 +	va_list ap;
    4.96 +
    4.97 +	printf("critical: ");
    4.98 +
    4.99 +	va_start(ap, fmt);
   4.100 +	vfprintf(stderr, fmt, ap);
   4.101 +	va_end(ap);
   4.102 +
   4.103 +	if(getenv("G_DEBUG")) {
   4.104 +		abort();
   4.105 +	}
   4.106 +}
   4.107 +
   4.108 +void fg_error(const char *fmt, ...)
   4.109 +{
   4.110 +	va_list ap;
   4.111 +
   4.112 +	printf("error: ");
   4.113 +
   4.114 +	va_start(ap, fmt);
   4.115 +	vfprintf(stderr, fmt, ap);
   4.116 +	va_end(ap);
   4.117 +	abort();
   4.118 +}
   4.119 +
   4.120 +void fg_debug(const char *fmt, ...)
   4.121 +{
   4.122 +	va_list ap;
   4.123 +
   4.124 +	printf("debug: ");
   4.125 +
   4.126 +	va_start(ap, fmt);
   4.127 +	vprintf(fmt, ap);
   4.128 +	va_end(ap);
   4.129 +}
     5.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     5.2 +++ b/src/fake_glib.h	Thu Jan 26 11:25:11 2012 +0200
     5.3 @@ -0,0 +1,40 @@
     5.4 +#ifndef FAKE_GLIB_H_
     5.5 +#define FAKE_GLIB_H_
     5.6 +
     5.7 +typedef struct FakeGPtrArray {
     5.8 +	void **pdata;
     5.9 +	unsigned int len;
    5.10 +} FakeGPtrArray;
    5.11 +
    5.12 +typedef int (*FakeGCompareFunc)(const void*, const void*);
    5.13 +
    5.14 +#ifndef TRUE
    5.15 +#define TRUE	1
    5.16 +#endif
    5.17 +#ifndef FALSE
    5.18 +#define FALSE	0
    5.19 +#endif
    5.20 +
    5.21 +typedef int gint;
    5.22 +typedef void* gpointer;
    5.23 +typedef const void* gconstpointer;
    5.24 +
    5.25 +#define fg_ptr_array_index(arr, idx) ((arr)->pdata[idx])
    5.26 +
    5.27 +FakeGPtrArray *fg_ptr_array_new(void);
    5.28 +void **fg_ptr_array_free(FakeGPtrArray *arr, int free_seg);
    5.29 +
    5.30 +void fg_ptr_array_add(FakeGPtrArray *arr, void *data);
    5.31 +int fg_ptr_array_remove(FakeGPtrArray *arr, void *data);
    5.32 +void *fg_ptr_array_remove_index(FakeGPtrArray *arr, unsigned int idx);
    5.33 +
    5.34 +void fg_ptr_array_sort(FakeGPtrArray *arr, FakeGCompareFunc cmp);
    5.35 +
    5.36 +/* -- logging -- */
    5.37 +#define fg_message	printf
    5.38 +void fg_warning(const char *fmt, ...);
    5.39 +void fg_critical(const char *fmt, ...);
    5.40 +void fg_error(const char *fmt, ...);
    5.41 +void fg_debug(const char *fmt, ...);
    5.42 +
    5.43 +#endif	/* FAKE_GLIB_H_ */
     6.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     6.2 +++ b/src/smf.c	Thu Jan 26 11:25:11 2012 +0200
     6.3 @@ -0,0 +1,1116 @@
     6.4 +/*-
     6.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
     6.6 + * All rights reserved.
     6.7 + *
     6.8 + * Redistribution and use in source and binary forms, with or without
     6.9 + * modification, are permitted provided that the following conditions
    6.10 + * are met:
    6.11 + * 1. Redistributions of source code must retain the above copyright
    6.12 + *    notice, this list of conditions and the following disclaimer.
    6.13 + * 2. Redistributions in binary form must reproduce the above copyright
    6.14 + *    notice, this list of conditions and the following disclaimer in the
    6.15 + *    documentation and/or other materials provided with the distribution.
    6.16 + *
    6.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
    6.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
    6.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
    6.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
    6.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    6.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
    6.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
    6.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
    6.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
    6.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
    6.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    6.28 + *
    6.29 + */
    6.30 +
    6.31 +/**
    6.32 + * \file
    6.33 + *
    6.34 + * Various functions.
    6.35 + *
    6.36 + */
    6.37 +
    6.38 +/* Reference: http://www.borg.com/~jglatt/tech/midifile.htm */
    6.39 +
    6.40 +#include <stdlib.h>
    6.41 +#include <string.h>
    6.42 +#include <assert.h>
    6.43 +#include <math.h>
    6.44 +#include <errno.h>
    6.45 +#ifdef __MINGW32__
    6.46 +#include <windows.h>
    6.47 +#else /* ! __MINGW32__ */
    6.48 +#include <arpa/inet.h>
    6.49 +#endif /* ! __MINGW32__ */
    6.50 +#include "smf.h"
    6.51 +#include "smf_private.h"
    6.52 +
    6.53 +/**
    6.54 + * Allocates new smf_t structure.
    6.55 + * \return pointer to smf_t or NULL.
    6.56 + */
    6.57 +smf_t *
    6.58 +smf_new(void)
    6.59 +{
    6.60 +	int cantfail;
    6.61 +
    6.62 +	smf_t *smf = malloc(sizeof(smf_t));
    6.63 +	if (smf == NULL) {
    6.64 +		fg_critical("Cannot allocate smf_t structure: %s", strerror(errno));
    6.65 +		return (NULL);
    6.66 +	}
    6.67 +
    6.68 +	memset(smf, 0, sizeof(smf_t));
    6.69 +
    6.70 +	smf->tracks_array = fg_ptr_array_new();
    6.71 +	assert(smf->tracks_array);
    6.72 +
    6.73 +	smf->tempo_array = fg_ptr_array_new();
    6.74 +	assert(smf->tempo_array);
    6.75 +
    6.76 +	cantfail = smf_set_ppqn(smf, 120);
    6.77 +	assert(!cantfail);
    6.78 +
    6.79 +	cantfail = smf_set_format(smf, 0);
    6.80 +	assert(!cantfail);
    6.81 +
    6.82 +	smf_init_tempo(smf);
    6.83 +
    6.84 +	return (smf);
    6.85 +}
    6.86 +
    6.87 +/**
    6.88 + * Frees smf and all it's descendant structures.
    6.89 + */
    6.90 +void
    6.91 +smf_delete(smf_t *smf)
    6.92 +{
    6.93 +	/* Remove all the tracks, from last to first. */
    6.94 +	while (smf->tracks_array->len > 0)
    6.95 +		smf_track_delete(fg_ptr_array_index(smf->tracks_array, smf->tracks_array->len - 1));
    6.96 +
    6.97 +	smf_fini_tempo(smf);
    6.98 +
    6.99 +	assert(smf->tracks_array->len == 0);
   6.100 +	assert(smf->number_of_tracks == 0);
   6.101 +	fg_ptr_array_free(smf->tracks_array, TRUE);
   6.102 +	fg_ptr_array_free(smf->tempo_array, TRUE);
   6.103 +
   6.104 +	memset(smf, 0, sizeof(smf_t));
   6.105 +	free(smf);
   6.106 +}
   6.107 +
   6.108 +/**
   6.109 + * Allocates new smf_track_t structure.
   6.110 + * \return pointer to smf_track_t or NULL.
   6.111 + */
   6.112 +smf_track_t *
   6.113 +smf_track_new(void)
   6.114 +{
   6.115 +	smf_track_t *track = malloc(sizeof(smf_track_t));
   6.116 +	if (track == NULL) {
   6.117 +		fg_critical("Cannot allocate smf_track_t structure: %s", strerror(errno));
   6.118 +		return (NULL);
   6.119 +	}
   6.120 +
   6.121 +	memset(track, 0, sizeof(smf_track_t));
   6.122 +	track->next_event_number = -1;
   6.123 +
   6.124 +	track->events_array = fg_ptr_array_new();
   6.125 +	assert(track->events_array);
   6.126 +
   6.127 +	return (track);
   6.128 +}
   6.129 +
   6.130 +/**
   6.131 + * Detaches track from its smf and frees it.
   6.132 + */
   6.133 +void
   6.134 +smf_track_delete(smf_track_t *track)
   6.135 +{
   6.136 +	assert(track);
   6.137 +	assert(track->events_array);
   6.138 +
   6.139 +	/* Remove all the events, from last to first. */
   6.140 +	while (track->events_array->len > 0)
   6.141 +		smf_event_delete(fg_ptr_array_index(track->events_array, track->events_array->len - 1));
   6.142 +
   6.143 +	if (track->smf)
   6.144 +		smf_track_remove_from_smf(track);
   6.145 +
   6.146 +	assert(track->events_array->len == 0);
   6.147 +	assert(track->number_of_events == 0);
   6.148 +	fg_ptr_array_free(track->events_array, TRUE);
   6.149 +
   6.150 +	memset(track, 0, sizeof(smf_track_t));
   6.151 +	free(track);
   6.152 +}
   6.153 +
   6.154 +
   6.155 +/**
   6.156 + * Appends smf_track_t to smf.
   6.157 + */
   6.158 +void
   6.159 +smf_add_track(smf_t *smf, smf_track_t *track)
   6.160 +{
   6.161 +	int cantfail;
   6.162 +
   6.163 +	assert(track->smf == NULL);
   6.164 +
   6.165 +	track->smf = smf;
   6.166 +	fg_ptr_array_add(smf->tracks_array, track);
   6.167 +
   6.168 +	smf->number_of_tracks++;
   6.169 +	track->track_number = smf->number_of_tracks;
   6.170 +
   6.171 +	if (smf->number_of_tracks > 1) {
   6.172 +		cantfail = smf_set_format(smf, 1);
   6.173 +		assert(!cantfail);
   6.174 +	}
   6.175 +}
   6.176 +
   6.177 +/**
   6.178 + * Detaches track from the smf.
   6.179 + */
   6.180 +void
   6.181 +smf_track_remove_from_smf(smf_track_t *track)
   6.182 +{
   6.183 +	int i, j;
   6.184 +	smf_track_t *tmp;
   6.185 +	smf_event_t *ev;
   6.186 +
   6.187 +	assert(track->smf != NULL);
   6.188 +
   6.189 +	track->smf->number_of_tracks--;
   6.190 +
   6.191 +	assert(track->smf->tracks_array);
   6.192 +	fg_ptr_array_remove(track->smf->tracks_array, track);
   6.193 +
   6.194 +	/* Renumber the rest of the tracks, so they are consecutively numbered. */
   6.195 +	for (i = track->track_number; i <= track->smf->number_of_tracks; i++) {
   6.196 +		tmp = smf_get_track_by_number(track->smf, i);
   6.197 +		tmp->track_number = i;
   6.198 +
   6.199 +		/*
   6.200 +		 * Events have track numbers too.  I guess this wasn't a wise
   6.201 +		 * decision.  ;-/
   6.202 +		 */
   6.203 +		for (j = 1; j <= tmp->number_of_events; j++) {
   6.204 +			ev = smf_track_get_event_by_number(tmp, j);
   6.205 +			ev->track_number = i;
   6.206 +		}
   6.207 +	}
   6.208 +
   6.209 +	track->track_number = -1;
   6.210 +	track->smf = NULL;
   6.211 +}
   6.212 +
   6.213 +/**
   6.214 + * Allocates new smf_event_t structure.  The caller is responsible for allocating
   6.215 + * event->midi_buffer, filling it with MIDI data and setting event->midi_buffer_length properly.
   6.216 + * Note that event->midi_buffer will be freed by smf_event_delete.
   6.217 + * \return pointer to smf_event_t or NULL.
   6.218 + */
   6.219 +smf_event_t *
   6.220 +smf_event_new(void)
   6.221 +{
   6.222 +	smf_event_t *event = malloc(sizeof(smf_event_t));
   6.223 +	if (event == NULL) {
   6.224 +		fg_critical("Cannot allocate smf_event_t structure: %s", strerror(errno));
   6.225 +		return (NULL);
   6.226 +	}
   6.227 +
   6.228 +	memset(event, 0, sizeof(smf_event_t));
   6.229 +
   6.230 +	event->delta_time_pulses = -1;
   6.231 +	event->time_pulses = -1;
   6.232 +	event->time_seconds = -1.0;
   6.233 +	event->track_number = -1;
   6.234 +
   6.235 +	return (event);
   6.236 +}
   6.237 +
   6.238 +/**
   6.239 + * Allocates an smf_event_t structure and fills it with "len" bytes copied
   6.240 + * from "midi_data".
   6.241 + * \param midi_data Pointer to MIDI data.  It sill be copied to the newly allocated event->midi_buffer.
   6.242 + * \param len Length of the buffer.  It must be proper MIDI event length, e.g. 3 for Note On event.
   6.243 + * \return Event containing MIDI data or NULL.
   6.244 + */
   6.245 +smf_event_t *
   6.246 +smf_event_new_from_pointer(void *midi_data, int len)
   6.247 +{
   6.248 +	smf_event_t *event;
   6.249 +
   6.250 +	event = smf_event_new();
   6.251 +	if (event == NULL)
   6.252 +		return (NULL);
   6.253 +
   6.254 +	event->midi_buffer_length = len;
   6.255 +	event->midi_buffer = malloc(event->midi_buffer_length);
   6.256 +	if (event->midi_buffer == NULL) {
   6.257 +		fg_critical("Cannot allocate MIDI buffer structure: %s", strerror(errno));
   6.258 +		smf_event_delete(event);
   6.259 +
   6.260 +		return (NULL);
   6.261 +	}
   6.262 +
   6.263 +	memcpy(event->midi_buffer, midi_data, len);
   6.264 +
   6.265 +	return (event);
   6.266 +}
   6.267 +
   6.268 +/**
   6.269 + * Allocates an smf_event_t structure and fills it with at most three bytes of data.
   6.270 + * For example, if you need to create Note On event, do something like this:
   6.271 + *
   6.272 + * smf_event_new_from_bytes(0x90, 0x3C, 0x7f);
   6.273 + *
   6.274 + * To create event for MIDI message that is shorter than three bytes, do something
   6.275 + * like this:
   6.276 + *
   6.277 + * smf_event_new_from_bytes(0xC0, 0x42, -1);
   6.278 + *
   6.279 + * \param first_byte First byte of MIDI message.  Must be valid status byte.
   6.280 + * \param second_byte Second byte of MIDI message or -1, if message is one byte long.
   6.281 + * \param third_byte Third byte of MIDI message or -1, if message is two bytes long.
   6.282 + * \return Event containing MIDI data or NULL.
   6.283 + */
   6.284 +smf_event_t *
   6.285 +smf_event_new_from_bytes(int first_byte, int second_byte, int third_byte)
   6.286 +{
   6.287 +	int len;
   6.288 +
   6.289 +	smf_event_t *event;
   6.290 +
   6.291 +	event = smf_event_new();
   6.292 +	if (event == NULL)
   6.293 +		return (NULL);
   6.294 +
   6.295 +	if (first_byte < 0) {
   6.296 +		fg_critical("First byte of MIDI message cannot be < 0");
   6.297 +		smf_event_delete(event);
   6.298 +
   6.299 +		return (NULL);
   6.300 +	}
   6.301 +
   6.302 +	if (first_byte > 255) {
   6.303 +		fg_critical("smf_event_new_from_bytes: first byte is %d, which is larger than 255.", first_byte);
   6.304 +		return (NULL);
   6.305 +	}
   6.306 +
   6.307 +	if (!is_status_byte(first_byte)) {
   6.308 +		fg_critical("smf_event_new_from_bytes: first byte is not a valid status byte.");
   6.309 +		return (NULL);
   6.310 +	}
   6.311 +
   6.312 +
   6.313 +	if (second_byte < 0)
   6.314 +		len = 1;
   6.315 +	else if (third_byte < 0)
   6.316 +		len = 2;
   6.317 +	else
   6.318 +		len = 3;
   6.319 +
   6.320 +	if (len > 1) {
   6.321 +		if (second_byte > 255) {
   6.322 +			fg_critical("smf_event_new_from_bytes: second byte is %d, which is larger than 255.", second_byte);
   6.323 +			return (NULL);
   6.324 +		}
   6.325 +
   6.326 +		if (is_status_byte(second_byte)) {
   6.327 +			fg_critical("smf_event_new_from_bytes: second byte cannot be a status byte.");
   6.328 +			return (NULL);
   6.329 +		}
   6.330 +	}
   6.331 +
   6.332 +	if (len > 2) {
   6.333 +		if (third_byte > 255) {
   6.334 +			fg_critical("smf_event_new_from_bytes: third byte is %d, which is larger than 255.", third_byte);
   6.335 +			return (NULL);
   6.336 +		}
   6.337 +
   6.338 +		if (is_status_byte(third_byte)) {
   6.339 +			fg_critical("smf_event_new_from_bytes: third byte cannot be a status byte.");
   6.340 +			return (NULL);
   6.341 +		}
   6.342 +	}
   6.343 +
   6.344 +	event->midi_buffer_length = len;
   6.345 +	event->midi_buffer = malloc(event->midi_buffer_length);
   6.346 +	if (event->midi_buffer == NULL) {
   6.347 +		fg_critical("Cannot allocate MIDI buffer structure: %s", strerror(errno));
   6.348 +		smf_event_delete(event);
   6.349 +
   6.350 +		return (NULL);
   6.351 +	}
   6.352 +
   6.353 +	event->midi_buffer[0] = first_byte;
   6.354 +	if (len > 1)
   6.355 +		event->midi_buffer[1] = second_byte;
   6.356 +	if (len > 2)
   6.357 +		event->midi_buffer[2] = third_byte;
   6.358 +
   6.359 +	return (event);
   6.360 +}
   6.361 +
   6.362 +/**
   6.363 + * Detaches event from its track and frees it.
   6.364 + */
   6.365 +void
   6.366 +smf_event_delete(smf_event_t *event)
   6.367 +{
   6.368 +	if (event->track != NULL)
   6.369 +		smf_event_remove_from_track(event);
   6.370 +
   6.371 +	if (event->midi_buffer != NULL) {
   6.372 +		memset(event->midi_buffer, 0, event->midi_buffer_length);
   6.373 +		free(event->midi_buffer);
   6.374 +	}
   6.375 +
   6.376 +	memset(event, 0, sizeof(smf_event_t));
   6.377 +	free(event);
   6.378 +}
   6.379 +
   6.380 +/**
   6.381 + * Used for sorting track->events_array.
   6.382 + */
   6.383 +static gint
   6.384 +events_array_compare_function(gconstpointer aa, gconstpointer bb)
   6.385 +{
   6.386 +	smf_event_t *a, *b;
   6.387 +
   6.388 +	/* "The comparison function for fg_ptr_array_sort() doesn't take the pointers
   6.389 +	    from the array as arguments, it takes pointers to the pointers in the array." */
   6.390 +	a = (smf_event_t *)*(gpointer *)aa;
   6.391 +	b = (smf_event_t *)*(gpointer *)bb;
   6.392 +
   6.393 +	if (a->time_pulses < b->time_pulses)
   6.394 +		return (-1);
   6.395 +
   6.396 +	if (a->time_pulses > b->time_pulses)
   6.397 +		return (1);
   6.398 +
   6.399 +	/*
   6.400 +	 * We need to preserve original order, otherwise things will break
   6.401 +	 * when there are several events with the same ->time_pulses.
   6.402 +	 * XXX: This is an ugly hack; we should remove sorting altogether.
   6.403 +	 */
   6.404 +
   6.405 +	if (a->event_number < b->event_number)
   6.406 +		return (-1);
   6.407 +
   6.408 +	if (a->event_number > b->event_number)
   6.409 +		return (1);
   6.410 +
   6.411 +	return (0);
   6.412 +}
   6.413 +
   6.414 +/*
   6.415 + * An assumption here is that if there is an EOT event, it will be at the end of the track.
   6.416 + */
   6.417 +static void
   6.418 +remove_eot_if_before_pulses(smf_track_t *track, int pulses)
   6.419 +{
   6.420 +	smf_event_t *event;
   6.421 +
   6.422 +	event = smf_track_get_last_event(track);
   6.423 +
   6.424 +	if (event == NULL)
   6.425 +		return;
   6.426 +
   6.427 +	if (!smf_event_is_eot(event))
   6.428 +		return;
   6.429 +
   6.430 +	if (event->time_pulses > pulses)
   6.431 +		return;
   6.432 +
   6.433 +	smf_event_remove_from_track(event);
   6.434 +}
   6.435 +
   6.436 +/**
   6.437 + * Adds the event to the track and computes ->delta_pulses.  Note that it is faster
   6.438 + * to append events to the end of the track than to insert them in the middle.
   6.439 + * Usually you want to use smf_track_add_event_seconds or smf_track_add_event_pulses
   6.440 + * instead of this one.  Event needs to have ->time_pulses and ->time_seconds already set.
   6.441 + * If you try to add event after an EOT, EOT event will be automatically deleted.
   6.442 + */
   6.443 +void
   6.444 +smf_track_add_event(smf_track_t *track, smf_event_t *event)
   6.445 +{
   6.446 +	int i, last_pulses = 0;
   6.447 +
   6.448 +	assert(track->smf != NULL);
   6.449 +	assert(event->track == NULL);
   6.450 +	assert(event->delta_time_pulses == -1);
   6.451 +	assert(event->time_pulses >= 0);
   6.452 +	assert(event->time_seconds >= 0.0);
   6.453 +
   6.454 +	remove_eot_if_before_pulses(track, event->time_pulses);
   6.455 +
   6.456 +	event->track = track;
   6.457 +	event->track_number = track->track_number;
   6.458 +
   6.459 +	if (track->number_of_events == 0) {
   6.460 +		assert(track->next_event_number == -1);
   6.461 +		track->next_event_number = 1;
   6.462 +	}
   6.463 +
   6.464 +	if (track->number_of_events > 0)
   6.465 +		last_pulses = smf_track_get_last_event(track)->time_pulses;
   6.466 +
   6.467 +	track->number_of_events++;
   6.468 +
   6.469 +	/* Are we just appending element at the end of the track? */
   6.470 +	if (last_pulses <= event->time_pulses) {
   6.471 +		event->delta_time_pulses = event->time_pulses - last_pulses;
   6.472 +		assert(event->delta_time_pulses >= 0);
   6.473 +		fg_ptr_array_add(track->events_array, event);
   6.474 +		event->event_number = track->number_of_events;
   6.475 +
   6.476 +	/* We need to insert in the middle of the track.  XXX: This is slow. */
   6.477 +	} else {
   6.478 +		/* Append, then sort according to ->time_pulses. */
   6.479 +		fg_ptr_array_add(track->events_array, event);
   6.480 +		fg_ptr_array_sort(track->events_array, events_array_compare_function);
   6.481 +
   6.482 +		/* Renumber entries and fix their ->delta_pulses. */
   6.483 +		for (i = 1; i <= track->number_of_events; i++) {
   6.484 +			smf_event_t *tmp = smf_track_get_event_by_number(track, i);
   6.485 +			tmp->event_number = i;
   6.486 +
   6.487 +			if (tmp->delta_time_pulses != -1)
   6.488 +				continue;
   6.489 +
   6.490 +			if (i == 1) {
   6.491 +				tmp->delta_time_pulses = tmp->time_pulses;
   6.492 +			} else {
   6.493 +				tmp->delta_time_pulses = tmp->time_pulses -
   6.494 +					smf_track_get_event_by_number(track, i - 1)->time_pulses;
   6.495 +				assert(tmp->delta_time_pulses >= 0);
   6.496 +			}
   6.497 +		}
   6.498 +
   6.499 +		/* Adjust ->delta_time_pulses of the next event. */
   6.500 +		if (event->event_number < track->number_of_events) {
   6.501 +			smf_event_t *next_event = smf_track_get_event_by_number(track, event->event_number + 1);
   6.502 +			assert(next_event);
   6.503 +			assert(next_event->time_pulses >= event->time_pulses);
   6.504 +			next_event->delta_time_pulses -= event->delta_time_pulses;
   6.505 +			assert(next_event->delta_time_pulses >= 0);
   6.506 +		}
   6.507 +	}
   6.508 +
   6.509 +	if (smf_event_is_tempo_change_or_time_signature(event)) {
   6.510 +		if (smf_event_is_last(event))
   6.511 +			maybe_add_to_tempo_map(event);
   6.512 +		else
   6.513 +			smf_create_tempo_map_and_compute_seconds(event->track->smf);
   6.514 +	}
   6.515 +}
   6.516 +
   6.517 +/**
   6.518 + * Add End Of Track metaevent.  Using it is optional, libsmf will automatically
   6.519 + * add EOT to the tracks during smf_save, with delta_pulses 0.  If you try to add EOT
   6.520 + * in the middle of the track, it will fail and nonzero value will be returned.
   6.521 + * If you try to add EOT after another EOT event, it will be added, but the existing
   6.522 + * EOT event will be removed.
   6.523 + *
   6.524 + * \return 0 if everything went ok, nonzero otherwise.
   6.525 + */
   6.526 +int
   6.527 +smf_track_add_eot_delta_pulses(smf_track_t *track, int delta)
   6.528 +{
   6.529 +	smf_event_t *event;
   6.530 +
   6.531 +	event = smf_event_new_from_bytes(0xFF, 0x2F, 0x00);
   6.532 +	if (event == NULL)
   6.533 +		return (-1);
   6.534 +
   6.535 +	smf_track_add_event_delta_pulses(track, event, delta);
   6.536 +
   6.537 +	return (0);
   6.538 +}
   6.539 +
   6.540 +int
   6.541 +smf_track_add_eot_pulses(smf_track_t *track, int pulses)
   6.542 +{
   6.543 +	smf_event_t *event, *last_event;
   6.544 +
   6.545 +	last_event = smf_track_get_last_event(track);
   6.546 +	if (last_event != NULL) {
   6.547 +		if (last_event->time_pulses > pulses)
   6.548 +			return (-2);
   6.549 +	}
   6.550 +
   6.551 +	event = smf_event_new_from_bytes(0xFF, 0x2F, 0x00);
   6.552 +	if (event == NULL)
   6.553 +		return (-3);
   6.554 +
   6.555 +	smf_track_add_event_pulses(track, event, pulses);
   6.556 +
   6.557 +	return (0);
   6.558 +}
   6.559 +
   6.560 +int
   6.561 +smf_track_add_eot_seconds(smf_track_t *track, double seconds)
   6.562 +{
   6.563 +	smf_event_t *event, *last_event;
   6.564 +
   6.565 +	last_event = smf_track_get_last_event(track);
   6.566 +	if (last_event != NULL) {
   6.567 +		if (last_event->time_seconds > seconds)
   6.568 +			return (-2);
   6.569 +	}
   6.570 +
   6.571 +	event = smf_event_new_from_bytes(0xFF, 0x2F, 0x00);
   6.572 +	if (event == NULL)
   6.573 +		return (-1);
   6.574 +
   6.575 +	smf_track_add_event_seconds(track, event, seconds);
   6.576 +
   6.577 +	return (0);
   6.578 +}
   6.579 +
   6.580 +/**
   6.581 + * Detaches event from its track.
   6.582 + */
   6.583 +void
   6.584 +smf_event_remove_from_track(smf_event_t *event)
   6.585 +{
   6.586 +	int i, was_last;
   6.587 +	smf_event_t *tmp;
   6.588 +	smf_track_t *track;
   6.589 +
   6.590 +	assert(event->track != NULL);
   6.591 +	assert(event->track->smf != NULL);
   6.592 +
   6.593 +	track = event->track;
   6.594 +	was_last = smf_event_is_last(event);
   6.595 +
   6.596 +	/* Adjust ->delta_time_pulses of the next event. */
   6.597 +	if (event->event_number < track->number_of_events) {
   6.598 +		tmp = smf_track_get_event_by_number(track, event->event_number + 1);
   6.599 +		assert(tmp);
   6.600 +		tmp->delta_time_pulses += event->delta_time_pulses;
   6.601 +	}
   6.602 +
   6.603 +	track->number_of_events--;
   6.604 +	fg_ptr_array_remove(track->events_array, event);
   6.605 +
   6.606 +	if (track->number_of_events == 0)
   6.607 +		track->next_event_number = -1;
   6.608 +
   6.609 +	/* Renumber the rest of the events, so they are consecutively numbered. */
   6.610 +	for (i = event->event_number; i <= track->number_of_events; i++) {
   6.611 +		tmp = smf_track_get_event_by_number(track, i);
   6.612 +		tmp->event_number = i;
   6.613 +	}
   6.614 +
   6.615 +	if (smf_event_is_tempo_change_or_time_signature(event)) {
   6.616 +		/* XXX: This will cause problems, when there is more than one Tempo Change event at a given time. */
   6.617 +		if (was_last)
   6.618 +			remove_last_tempo_with_pulses(event->track->smf, event->time_pulses);
   6.619 +		else
   6.620 +			smf_create_tempo_map_and_compute_seconds(track->smf);
   6.621 +	}
   6.622 +
   6.623 +	event->track = NULL;
   6.624 +	event->event_number = -1;
   6.625 +	event->delta_time_pulses = -1;
   6.626 +	event->time_pulses = -1;
   6.627 +	event->time_seconds = -1.0;
   6.628 +}
   6.629 +
   6.630 +/**
   6.631 +  * \return Nonzero if event is Tempo Change or Time Signature metaevent.
   6.632 +  */
   6.633 +int
   6.634 +smf_event_is_tempo_change_or_time_signature(const smf_event_t *event)
   6.635 +{
   6.636 +	if (!smf_event_is_metadata(event))
   6.637 +		return (0);
   6.638 +
   6.639 +	assert(event->midi_buffer_length >= 2);
   6.640 +
   6.641 +	if (event->midi_buffer[1] == 0x51 || event->midi_buffer[1] == 0x58)
   6.642 +		return (1);
   6.643 +
   6.644 +	return (0);
   6.645 +}
   6.646 +
   6.647 +/**
   6.648 +  * Sets "Format" field of MThd header to the specified value.  Note that you
   6.649 +  * don't really need to use this, as libsmf will automatically change format
   6.650 +  * from 0 to 1 when you add the second track.
   6.651 +  * \param smf SMF.
   6.652 +  * \param format 0 for one track per file, 1 for several tracks per file.
   6.653 +  */
   6.654 +int
   6.655 +smf_set_format(smf_t *smf, int format)
   6.656 +{
   6.657 +	assert(format == 0 || format == 1);
   6.658 +
   6.659 +	if (smf->number_of_tracks > 1 && format == 0) {
   6.660 +		fg_critical("There is more than one track, cannot set format to 0.");
   6.661 +		return (-1);
   6.662 +	}
   6.663 +
   6.664 +	smf->format = format;
   6.665 +
   6.666 +	return (0);
   6.667 +}
   6.668 +
   6.669 +/**
   6.670 +  * Sets the PPQN ("Division") field of MThd header.  This is mandatory, you
   6.671 +  * should call it right after smf_new.  Note that changing PPQN will change time_seconds
   6.672 +  * of all the events.
   6.673 +  * \param smf SMF.
   6.674 +  * \param ppqn New PPQN.
   6.675 +  */
   6.676 +int
   6.677 +smf_set_ppqn(smf_t *smf, int ppqn)
   6.678 +{
   6.679 +	assert(ppqn > 0);
   6.680 +
   6.681 +	smf->ppqn = ppqn;
   6.682 +
   6.683 +	return (0);
   6.684 +}
   6.685 +
   6.686 +/**
   6.687 +  * Returns next event from the track given and advances next event counter.
   6.688 +  * Do not depend on End Of Track event being the last event on the track - it
   6.689 +  * is possible that the track will not end with EOT if you haven't added it
   6.690 +  * yet.  EOTs are added automatically during smf_save().
   6.691 +  *
   6.692 +  * \return Event or NULL, if there are no more events left in this track.
   6.693 +  */
   6.694 +smf_event_t *
   6.695 +smf_track_get_next_event(smf_track_t *track)
   6.696 +{
   6.697 +	smf_event_t *event, *next_event;
   6.698 +
   6.699 +	/* End of track? */
   6.700 +	if (track->next_event_number == -1)
   6.701 +		return (NULL);
   6.702 +
   6.703 +	assert(track->next_event_number >= 1);
   6.704 +	assert(track->number_of_events > 0);
   6.705 +
   6.706 +	event = smf_track_get_event_by_number(track, track->next_event_number);
   6.707 +
   6.708 +	assert(event != NULL);
   6.709 +
   6.710 +	/* Is this the last event in the track? */
   6.711 +	if (track->next_event_number < track->number_of_events) {
   6.712 +		next_event = smf_track_get_event_by_number(track, track->next_event_number + 1);
   6.713 +		assert(next_event);
   6.714 +
   6.715 +		track->time_of_next_event = next_event->time_pulses;
   6.716 +		track->next_event_number++;
   6.717 +	} else {
   6.718 +		track->next_event_number = -1;
   6.719 +	}
   6.720 +
   6.721 +	return (event);
   6.722 +}
   6.723 +
   6.724 +/**
   6.725 +  * Returns next event from the track given.  Does not change next event counter,
   6.726 +  * so repeatedly calling this routine will return the same event.
   6.727 +  * \return Event or NULL, if there are no more events left in this track.
   6.728 +  */
   6.729 +static smf_event_t *
   6.730 +smf_peek_next_event_from_track(smf_track_t *track)
   6.731 +{
   6.732 +	smf_event_t *event;
   6.733 +
   6.734 +	/* End of track? */
   6.735 +	if (track->next_event_number == -1)
   6.736 +		return (NULL);
   6.737 +
   6.738 +	assert(track->next_event_number >= 1);
   6.739 +	assert(track->events_array->len != 0);
   6.740 +
   6.741 +	event = smf_track_get_event_by_number(track, track->next_event_number);
   6.742 +
   6.743 +	return (event);
   6.744 +}
   6.745 +
   6.746 +/**
   6.747 + * \return Track with a given number or NULL, if there is no such track.
   6.748 + * Tracks are numbered consecutively starting from one.
   6.749 + */
   6.750 +smf_track_t *
   6.751 +smf_get_track_by_number(const smf_t *smf, int track_number)
   6.752 +{
   6.753 +	smf_track_t *track;
   6.754 +
   6.755 +	assert(track_number >= 1);
   6.756 +
   6.757 +	if (track_number > smf->number_of_tracks)
   6.758 +		return (NULL);
   6.759 +
   6.760 +	track = (smf_track_t *)fg_ptr_array_index(smf->tracks_array, track_number - 1);
   6.761 +
   6.762 +	assert(track);
   6.763 +
   6.764 +	return (track);
   6.765 +}
   6.766 +
   6.767 +/**
   6.768 + * \return Event with a given number or NULL, if there is no such event.
   6.769 + * Events are numbered consecutively starting from one.
   6.770 + */
   6.771 +smf_event_t *
   6.772 +smf_track_get_event_by_number(const smf_track_t *track, int event_number)
   6.773 +{
   6.774 +	smf_event_t *event;
   6.775 +
   6.776 +	assert(event_number >= 1);
   6.777 +
   6.778 +	if (event_number > track->number_of_events)
   6.779 +		return (NULL);
   6.780 +
   6.781 +	event = fg_ptr_array_index(track->events_array, event_number - 1);
   6.782 +
   6.783 +	assert(event);
   6.784 +
   6.785 +	return (event);
   6.786 +}
   6.787 +
   6.788 +/**
   6.789 + * \return Last event on the track or NULL, if track is empty.
   6.790 + */
   6.791 +smf_event_t *
   6.792 +smf_track_get_last_event(const smf_track_t *track)
   6.793 +{
   6.794 +	smf_event_t *event;
   6.795 +
   6.796 +	if (track->number_of_events == 0)
   6.797 +		return (NULL);
   6.798 +
   6.799 +	event = smf_track_get_event_by_number(track, track->number_of_events);
   6.800 +
   6.801 +	return (event);
   6.802 +}
   6.803 +
   6.804 +/**
   6.805 + * Searches for track that contains next event, in time order.  In other words,
   6.806 + * returns the track that contains event that should be played next.
   6.807 + * \return Track with next event or NULL, if there are no events left.
   6.808 + */
   6.809 +smf_track_t *
   6.810 +smf_find_track_with_next_event(smf_t *smf)
   6.811 +{
   6.812 +	int i, min_time = 0;
   6.813 +	smf_track_t *track = NULL, *min_time_track = NULL;
   6.814 +
   6.815 +	/* Find track with event that should be played next. */
   6.816 +	for (i = 1; i <= smf->number_of_tracks; i++) {
   6.817 +		track = smf_get_track_by_number(smf, i);
   6.818 +
   6.819 +		assert(track);
   6.820 +
   6.821 +		/* No more events in this track? */
   6.822 +		if (track->next_event_number == -1)
   6.823 +			continue;
   6.824 +
   6.825 +		if (track->time_of_next_event < min_time || min_time_track == NULL) {
   6.826 +			min_time = track->time_of_next_event;
   6.827 +			min_time_track = track;
   6.828 +		}
   6.829 +	}
   6.830 +
   6.831 +	return (min_time_track);
   6.832 +}
   6.833 +
   6.834 +/**
   6.835 +  * \return Next event, in time order, or NULL, if there are none left.
   6.836 +  */
   6.837 +smf_event_t *
   6.838 +smf_get_next_event(smf_t *smf)
   6.839 +{
   6.840 +	smf_event_t *event;
   6.841 +	smf_track_t *track = smf_find_track_with_next_event(smf);
   6.842 +
   6.843 +	if (track == NULL) {
   6.844 +#if 0
   6.845 +		g_debug("End of the song.");
   6.846 +#endif
   6.847 +
   6.848 +		return (NULL);
   6.849 +	}
   6.850 +
   6.851 +	event = smf_track_get_next_event(track);
   6.852 +
   6.853 +	assert(event != NULL);
   6.854 +
   6.855 +	event->track->smf->last_seek_position = -1.0;
   6.856 +
   6.857 +	return (event);
   6.858 +}
   6.859 +
   6.860 +/**
   6.861 +  * Advance the "next event counter".  This is functionally the same as calling
   6.862 +  * smf_get_next_event and ignoring the return value.
   6.863 +  */
   6.864 +void
   6.865 +smf_skip_next_event(smf_t *smf)
   6.866 +{
   6.867 +	void *notused;
   6.868 +
   6.869 +	notused = smf_get_next_event(smf);
   6.870 +}
   6.871 +
   6.872 +/**
   6.873 +  * \return Next event, in time order, or NULL, if there are none left.  Does
   6.874 +  * not advance position in song.
   6.875 +  */
   6.876 +smf_event_t *
   6.877 +smf_peek_next_event(smf_t *smf)
   6.878 +{
   6.879 +	smf_event_t *event;
   6.880 +	smf_track_t *track = smf_find_track_with_next_event(smf);
   6.881 +
   6.882 +	if (track == NULL) {
   6.883 +#if 0
   6.884 +		g_debug("End of the song.");
   6.885 +#endif
   6.886 +
   6.887 +		return (NULL);
   6.888 +	}
   6.889 +
   6.890 +	event = smf_peek_next_event_from_track(track);
   6.891 +
   6.892 +	assert(event != NULL);
   6.893 +
   6.894 +	return (event);
   6.895 +}
   6.896 +
   6.897 +/**
   6.898 +  * Rewinds the SMF.  What that means is, after calling this routine, smf_get_next_event
   6.899 +  * will return first event in the song.
   6.900 +  */
   6.901 +void
   6.902 +smf_rewind(smf_t *smf)
   6.903 +{
   6.904 +	int i;
   6.905 +	smf_track_t *track = NULL;
   6.906 +	smf_event_t *event;
   6.907 +
   6.908 +	assert(smf);
   6.909 +
   6.910 +	smf->last_seek_position = 0.0;
   6.911 +
   6.912 +	for (i = 1; i <= smf->number_of_tracks; i++) {
   6.913 +		track = smf_get_track_by_number(smf, i);
   6.914 +
   6.915 +		assert(track != NULL);
   6.916 +
   6.917 +		if (track->number_of_events > 0) {
   6.918 +			track->next_event_number = 1;
   6.919 +			event = smf_peek_next_event_from_track(track);
   6.920 +			assert(event);
   6.921 +			track->time_of_next_event = event->time_pulses;
   6.922 +		} else {
   6.923 +			track->next_event_number = -1;
   6.924 +			track->time_of_next_event = 0;
   6.925 +#if 0
   6.926 +			g_warning("Warning: empty track.");
   6.927 +#endif
   6.928 +		}
   6.929 +	}
   6.930 +}
   6.931 +
   6.932 +/**
   6.933 +  * Seeks the SMF to the given event.  After calling this routine, smf_get_next_event
   6.934 +  * will return the event that was the second argument of this call.
   6.935 +  */
   6.936 +int
   6.937 +smf_seek_to_event(smf_t *smf, const smf_event_t *target)
   6.938 +{
   6.939 +	smf_event_t *event;
   6.940 +
   6.941 +	smf_rewind(smf);
   6.942 +
   6.943 +#if 0
   6.944 +	g_debug("Seeking to event %d, track %d.", target->event_number, target->track->track_number);
   6.945 +#endif
   6.946 +
   6.947 +	for (;;) {
   6.948 +		event = smf_peek_next_event(smf);
   6.949 +
   6.950 +		/* There can't be NULL here, unless "target" is not in this smf. */
   6.951 +		assert(event);
   6.952 +
   6.953 +		if (event != target)
   6.954 +			smf_skip_next_event(smf);
   6.955 +		else
   6.956 +			break;
   6.957 +	}
   6.958 +
   6.959 +	smf->last_seek_position = event->time_seconds;
   6.960 +
   6.961 +	return (0);
   6.962 +}
   6.963 +
   6.964 +/**
   6.965 +  * Seeks the SMF to the given position.  For example, after seeking to 1.0 seconds,
   6.966 +  * smf_get_next_event will return first event that happens after the first second of song.
   6.967 +  */
   6.968 +int
   6.969 +smf_seek_to_seconds(smf_t *smf, double seconds)
   6.970 +{
   6.971 +	smf_event_t *event;
   6.972 +
   6.973 +	assert(seconds >= 0.0);
   6.974 +
   6.975 +	if (seconds == smf->last_seek_position) {
   6.976 +#if 0
   6.977 +		g_debug("Avoiding seek to %f seconds.", seconds);
   6.978 +#endif
   6.979 +		return (0);
   6.980 +	}
   6.981 +
   6.982 +	smf_rewind(smf);
   6.983 +
   6.984 +#if 0
   6.985 +	g_debug("Seeking to %f seconds.", seconds);
   6.986 +#endif
   6.987 +
   6.988 +	for (;;) {
   6.989 +		event = smf_peek_next_event(smf);
   6.990 +
   6.991 +		if (event == NULL) {
   6.992 +			fg_critical("Trying to seek past the end of song.");
   6.993 +			return (-1);
   6.994 +		}
   6.995 +
   6.996 +		if (event->time_seconds < seconds)
   6.997 +			smf_skip_next_event(smf);
   6.998 +		else
   6.999 +			break;
  6.1000 +	}
  6.1001 +
  6.1002 +	smf->last_seek_position = seconds;
  6.1003 +
  6.1004 +	return (0);
  6.1005 +}
  6.1006 +
  6.1007 +/**
  6.1008 +  * Seeks the SMF to the given position.  For example, after seeking to 10 pulses,
  6.1009 +  * smf_get_next_event will return first event that happens after the first ten pulses.
  6.1010 +  */
  6.1011 +int
  6.1012 +smf_seek_to_pulses(smf_t *smf, int pulses)
  6.1013 +{
  6.1014 +	smf_event_t *event;
  6.1015 +
  6.1016 +	assert(pulses >= 0);
  6.1017 +
  6.1018 +	smf_rewind(smf);
  6.1019 +
  6.1020 +#if 0
  6.1021 +	g_debug("Seeking to %d pulses.", pulses);
  6.1022 +#endif
  6.1023 +
  6.1024 +	for (;;) {
  6.1025 +		event = smf_peek_next_event(smf);
  6.1026 +
  6.1027 +		if (event == NULL) {
  6.1028 +			fg_critical("Trying to seek past the end of song.");
  6.1029 +			return (-1);
  6.1030 +		}
  6.1031 +
  6.1032 +		if (event->time_pulses < pulses)
  6.1033 +			smf_skip_next_event(smf);
  6.1034 +		else
  6.1035 +			break;
  6.1036 +	}
  6.1037 +
  6.1038 +	smf->last_seek_position = event->time_seconds;
  6.1039 +
  6.1040 +	return (0);
  6.1041 +}
  6.1042 +
  6.1043 +/**
  6.1044 +  * \return Length of SMF, in pulses.
  6.1045 +  */
  6.1046 +int
  6.1047 +smf_get_length_pulses(const smf_t *smf)
  6.1048 +{
  6.1049 +	int pulses = 0, i;
  6.1050 +
  6.1051 +	for (i = 1; i <= smf->number_of_tracks; i++) {
  6.1052 +		smf_track_t *track;
  6.1053 +		smf_event_t *event;
  6.1054 +
  6.1055 +		track = smf_get_track_by_number(smf, i);
  6.1056 +		assert(track);
  6.1057 +
  6.1058 +		event = smf_track_get_last_event(track);
  6.1059 +		/* Empty track? */
  6.1060 +		if (event == NULL)
  6.1061 +			continue;
  6.1062 +
  6.1063 +		if (event->time_pulses > pulses)
  6.1064 +			pulses = event->time_pulses;
  6.1065 +	}
  6.1066 +
  6.1067 +	return (pulses);
  6.1068 +}
  6.1069 +
  6.1070 +/**
  6.1071 +  * \return Length of SMF, in seconds.
  6.1072 +  */
  6.1073 +double
  6.1074 +smf_get_length_seconds(const smf_t *smf)
  6.1075 +{
  6.1076 +	int i;
  6.1077 +	double seconds = 0.0;
  6.1078 +
  6.1079 +	for (i = 1; i <= smf->number_of_tracks; i++) {
  6.1080 +		smf_track_t *track;
  6.1081 +		smf_event_t *event;
  6.1082 +
  6.1083 +		track = smf_get_track_by_number(smf, i);
  6.1084 +		assert(track);
  6.1085 +
  6.1086 +		event = smf_track_get_last_event(track);
  6.1087 +		/* Empty track? */
  6.1088 +		if (event == NULL)
  6.1089 +			continue;
  6.1090 +
  6.1091 +		if (event->time_seconds > seconds)
  6.1092 +			seconds = event->time_seconds;
  6.1093 +	}
  6.1094 +
  6.1095 +	return (seconds);
  6.1096 +}
  6.1097 +
  6.1098 +/**
  6.1099 +  * \return Nonzero, if there are no events in the SMF after this one.
  6.1100 +  * Note that may be more than one "last event", if they occur at the same time.
  6.1101 +  */
  6.1102 +int
  6.1103 +smf_event_is_last(const smf_event_t *event)
  6.1104 +{
  6.1105 +	if (smf_get_length_pulses(event->track->smf) <= event->time_pulses)
  6.1106 +		return (1);
  6.1107 +
  6.1108 +	return (0);
  6.1109 +}
  6.1110 +
  6.1111 +/**
  6.1112 +  * \return Version of libsmf.
  6.1113 +  */
  6.1114 +const char *
  6.1115 +smf_get_version(void)
  6.1116 +{
  6.1117 +	return (SMF_VERSION);
  6.1118 +}
  6.1119 +
     7.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     7.2 +++ b/src/smf.h	Thu Jan 26 11:25:11 2012 +0200
     7.3 @@ -0,0 +1,417 @@
     7.4 +/*-
     7.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
     7.6 + * All rights reserved.
     7.7 + *
     7.8 + * Redistribution and use in source and binary forms, with or without
     7.9 + * modification, are permitted provided that the following conditions
    7.10 + * are met:
    7.11 + * 1. Redistributions of source code must retain the above copyright
    7.12 + *    notice, this list of conditions and the following disclaimer.
    7.13 + * 2. Redistributions in binary form must reproduce the above copyright
    7.14 + *    notice, this list of conditions and the following disclaimer in the
    7.15 + *    documentation and/or other materials provided with the distribution.
    7.16 + *
    7.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
    7.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
    7.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
    7.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
    7.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    7.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
    7.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
    7.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
    7.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
    7.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
    7.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    7.28 + *
    7.29 + */
    7.30 +
    7.31 +/**
    7.32 + * \file
    7.33 + *
    7.34 + * Public interface declaration for libsmf, Standard MIDI File format library.
    7.35 + */
    7.36 +
    7.37 +/**
    7.38 + *
    7.39 + * \mainpage libsmf - general usage instructions
    7.40 + *
    7.41 + * An smf_t structure represents a "song".  Every valid smf contains one or more tracks.
    7.42 + * Tracks contain zero or more events.  Libsmf doesn't care about actual MIDI data, as long
    7.43 + * as it is valid from the MIDI specification point of view - it may be realtime message,
    7.44 + * SysEx, whatever.
    7.45 + * 
    7.46 + * The only field in smf_t, smf_track_t, smf_event_t and smf_tempo_t structures your
    7.47 + * code may modify is event->midi_buffer and event->midi_buffer_length.  Do not modify
    7.48 + * other fields, _ever_.  You may read them, though.  Do not declare static instances
    7.49 + * of these types, i.e. never do something like this:  "smf_t smf;".  Always use
    7.50 + * "smf_t *smf = smf_new();".  The same applies to smf_track_t and smf_event_t.
    7.51 + * 
    7.52 + * Say you want to load a Standard MIDI File (.mid) file and play it back somehow.  This is (roughly)
    7.53 + * how you do this:
    7.54 + * 
    7.55 + * \code
    7.56 + * 	smf_t *smf;
    7.57 + * 	smf_event_t *event;
    7.58 + *
    7.59 + * 	smf = smf_load(file_name);
    7.60 + * 	if (smf == NULL) {
    7.61 + * 		Whoops, something went wrong.
    7.62 + * 		return;
    7.63 + * 	}
    7.64 + * 
    7.65 + * 	while ((event = smf_get_next_event(smf)) != NULL) {
    7.66 + *		if (smf_event_is_metadata(event))
    7.67 + *			continue;
    7.68 + * 
    7.69 + * 		wait until event->time_seconds.
    7.70 + * 		feed_to_midi_output(event->midi_buffer, event->midi_buffer_length);
    7.71 + * 	}
    7.72 + *
    7.73 + *	smf_delete(smf);
    7.74 + *
    7.75 + * \endcode
    7.76 + * 
    7.77 + * Saving works like this:
    7.78 + * 
    7.79 + * \code
    7.80 + *
    7.81 + * 	smf_t *smf;
    7.82 + *	smf_track_t *track;
    7.83 + *	smf_event_t *event;
    7.84 + *
    7.85 + * 	smf = smf_new();
    7.86 + * 	if (smf == NULL) {
    7.87 + * 		Whoops.
    7.88 + * 		return;
    7.89 + * 	}
    7.90 + * 
    7.91 + * 	for (int i = 1; i <= number of tracks; i++) {
    7.92 + * 		track = smf_track_new();
    7.93 + * 		if (track == NULL) {
    7.94 + * 			Whoops.
    7.95 + * 			return;
    7.96 + * 		}
    7.97 + * 
    7.98 + * 		smf_add_track(smf, track);
    7.99 + * 
   7.100 + * 		for (int j = 1; j <= number of events you want to put into this track; j++) {
   7.101 + * 			event = smf_event_new_from_pointer(your MIDI message, message length);
   7.102 + * 			if (event == NULL) {
   7.103 + * 				Whoops.
   7.104 + * 				return;
   7.105 + * 			}
   7.106 + * 
   7.107 + * 			smf_track_add_event_seconds(track, event, seconds since start of the song);
   7.108 + * 		}
   7.109 + * 	}
   7.110 + * 
   7.111 + * 	ret = smf_save(smf, file_name);
   7.112 + * 	if (ret) {
   7.113 + * 		Whoops, saving failed for some reason.
   7.114 + * 		return;
   7.115 + * 	}
   7.116 + *
   7.117 + *	smf_delete(smf);
   7.118 + *
   7.119 + * \endcode
   7.120 + *
   7.121 + * There are two basic ways of getting MIDI data out of smf - sequential or by track/event number.  You may
   7.122 + * mix them if you need to.  First one is used in the example above - seek to the point from which you want
   7.123 + * the playback to start (using smf_seek_to_seconds(), smf_seek_to_pulses() or smf_seek_to_event()) and then
   7.124 + * do smf_get_next_event() in loop, until it returns NULL.  Calling smf_load() causes the smf to be rewound
   7.125 + * to the start of the song.
   7.126 + *
   7.127 + * Getting events by number works like this:
   7.128 + *
   7.129 + * \code
   7.130 + *
   7.131 + * smf_track_t *track = smf_get_track_by_number(smf, track_number);
   7.132 + * smf_event_t *event = smf_track_get_event_by_number(track, event_number);
   7.133 + *
   7.134 + * \endcode
   7.135 + *
   7.136 + * To create new event, use smf_event_new(), smf_event_new_from_pointer() or smf_event_new_from_bytes().
   7.137 + * First one creates an empty event - you need to manually allocate (using malloc(3)) buffer for
   7.138 + * MIDI data, write MIDI data into it, put the address of that buffer into event->midi_buffer,
   7.139 + * and the length of MIDI data into event->midi_buffer_length.  Note that deleting the event
   7.140 + * (using smf_event_delete()) will free the buffer.
   7.141 + *
   7.142 + * Second form does most of this for you: it takes an address of the buffer containing MIDI data,
   7.143 + * allocates storage and copies MIDI data into it.
   7.144 + *
   7.145 + * Third form is useful for manually creating short events, up to three bytes in length, for
   7.146 + * example Note On or Note Off events.  It simply takes three bytes and creates MIDI event containing
   7.147 + * them.  If you need to create MIDI message that takes only two bytes, pass -1 as the third byte.
   7.148 + * For one byte message (System Realtime), pass -1 as second and third byte.
   7.149 + *
   7.150 + * To free an event, use smf_event_delete().
   7.151 + *
   7.152 + * To add event to the track, use smf_track_add_event_delta_pulses(), smf_track_add_event_pulses(),
   7.153 + * or smf_track_add_event_seconds().  The difference between them is in the way you specify the time of
   7.154 + * the event - with the first one, you specify it as an interval, in pulses, from the previous event
   7.155 + * in this track; with the second one, you specify it as pulses from the start of the song, and with the
   7.156 + * last one, you specify it as seconds from the start of the song.  Obviously, the first version can
   7.157 + * only append events at the end of the track.
   7.158 + *
   7.159 + * To remove an event from the track it's attached to, use smf_event_remove_from_track().  You may
   7.160 + * want to free the event (using smf_event_delete()) afterwards.
   7.161 + *
   7.162 + * To create new track, use smf_track_new().  To add track to the smf, use smf_add_track().
   7.163 + * To remove track from its smf, use smf_track_remove_from_smf().  To free the track structure,
   7.164 + * use smf_track_delete().
   7.165 + *
   7.166 + * Note that libsmf keeps things consistent.  If you free (using smf_track_delete()) a track that
   7.167 + * is attached to an smf and contains events, libsmf will detach the events, free them, detach
   7.168 + * the track, free it etc.
   7.169 + *
   7.170 + * Tracks and events are numbered consecutively, starting from one.  If you remove a track or event,
   7.171 + * the rest of tracks/events will get renumbered.  To get the number of a given event in its track, use event->event_number.
   7.172 + * To get the number of track in its smf, use track->track_number.  To get the number of events in the track,
   7.173 + * use track->number_of_events.  To get the number of tracks in the smf, use smf->number_of_tracks.
   7.174 + *
   7.175 + * In SMF File Format, each track has to end with End Of Track metaevent.  If you load SMF file using smf_load(),
   7.176 + * that will be the case.  If you want to create or edit an SMF, you don't need to worry about EOT events;
   7.177 + * libsmf automatically takes care of them for you.  If you try to save an SMF with tracks that do not end
   7.178 + * with EOTs, smf_save() will append them.  If you try to add event that happens after EOT metaevent, libsmf
   7.179 + * will remove the EOT.  If you want to add EOT manually, you can, of course, using smf_track_add_eot_seconds()
   7.180 + * or smf_track_add_eot_pulses().
   7.181 + *
   7.182 + * Each event carries three time values - event->time_seconds, which is seconds since the start of the song,
   7.183 + * event->time_pulses, which is PPQN clocks since the start of the song, and event->delta_pulses, which is PPQN clocks
   7.184 + * since the previous event in that track.  These values are invalid if the event is not attached to the track.
   7.185 + * If event is attached, all three values are valid.  Time of the event is specified when adding the event
   7.186 + * (using smf_track_add_event_seconds(), smf_track_add_event_pulses() or smf_track_add_event_delta_pulses()); the remaining
   7.187 + * two values are computed from that.
   7.188 + *
   7.189 + * Tempo related stuff happens automatically - when you add a metaevent that
   7.190 + * is Tempo Change or Time Signature, libsmf adds that event to the tempo map.  If you remove
   7.191 + * Tempo Change event that is in the middle of the song, the rest of the events will have their
   7.192 + * event->time_seconds recomputed from event->time_pulses before smf_event_remove_from_track() function returns.
   7.193 + * Adding Tempo Change in the middle of the song works in a similar way.
   7.194 + * 	
   7.195 + * MIDI data (event->midi_buffer) is always kept in normalized form - it always begins with status byte
   7.196 + * (no running status), there are no System Realtime events embedded in them etc.  Events like SysExes
   7.197 + * are in "on the wire" form, without embedded length that is used in SMF file format.  Obviously
   7.198 + * libsmf "normalizes" MIDI data during loading and "denormalizes" (adding length to SysExes, escaping
   7.199 + * System Common and System Realtime messages etc) during writing.
   7.200 + *
   7.201 + * Note that you always have to first add the track to smf, and then add events to the track.
   7.202 + * Doing it the other way around will trip asserts.  Also, try to add events at the end of the track and remove
   7.203 + * them from the end of the track, that's much more efficient.
   7.204 + * 
   7.205 + * All the libsmf functions have prefix "smf_".  First argument for routines whose names start with
   7.206 + * "smf_event_" is "smf_event_t *", for routines whose names start with "smf_track_" - "smf_track_t *",
   7.207 + * and for plain "smf_" - "smf_t *".  The only exception are smf_whatever_new routines.
   7.208 + * Library does not use any global variables and is thread-safe,
   7.209 + * as long as you don't try to work on the same SMF (smf_t and its descendant tracks and events) from several
   7.210 + * threads at once without protecting it with mutex.  Library depends on glib and nothing else.  License is
   7.211 + * BSD, two clause, which basically means you can use it freely in your software, both Open Source (including
   7.212 + * GPL) and closed source.
   7.213 + *
   7.214 + */
   7.215 +
   7.216 +#ifndef SMF_H
   7.217 +#define SMF_H
   7.218 +
   7.219 +#ifdef __cplusplus
   7.220 +extern "C" {
   7.221 +#endif
   7.222 +
   7.223 +#include <stdio.h>
   7.224 +
   7.225 +#if defined(__GNUC__) && __GNUC__ >= 4
   7.226 +#define WARN_UNUSED_RESULT __attribute__ ((warn_unused_result))
   7.227 +#else
   7.228 +#define WARN_UNUSED_RESULT
   7.229 +#endif
   7.230 +
   7.231 +struct FakeGPtrArray;
   7.232 +
   7.233 +/** Represents a "song", that is, collection of one or more tracks. */
   7.234 +struct smf_struct {
   7.235 +	int		format;
   7.236 +
   7.237 +	/** These fields are extracted from "division" field of MThd header.  Valid is _either_ ppqn or frames_per_second/resolution. */
   7.238 +	int		ppqn;
   7.239 +	int		frames_per_second;
   7.240 +	int		resolution;
   7.241 +	int		number_of_tracks;
   7.242 +
   7.243 +	/** These are private fields using only by loading and saving routines. */
   7.244 +	FILE		*stream;
   7.245 +	void		*file_buffer;
   7.246 +	int		file_buffer_length;
   7.247 +	int		next_chunk_offset;
   7.248 +	int		expected_number_of_tracks;
   7.249 +
   7.250 +	/** Private, used by smf.c. */
   7.251 +	struct FakeGPtrArray	*tracks_array;
   7.252 +	double		last_seek_position;
   7.253 +
   7.254 +	/** Private, used by smf_tempo.c. */
   7.255 +	/** Array of pointers to smf_tempo_struct. */
   7.256 +	struct FakeGPtrArray	*tempo_array;
   7.257 +};
   7.258 +
   7.259 +typedef struct smf_struct smf_t;
   7.260 +
   7.261 +/** Describes a single tempo or time signature change. */
   7.262 +struct smf_tempo_struct {
   7.263 +	int time_pulses;
   7.264 +	double time_seconds;
   7.265 +	int microseconds_per_quarter_note;
   7.266 +	int numerator;
   7.267 +	int denominator;
   7.268 +	int clocks_per_click;
   7.269 +	int notes_per_note;
   7.270 +};
   7.271 +
   7.272 +typedef struct smf_tempo_struct smf_tempo_t;
   7.273 +
   7.274 +/** Represents a single track. */
   7.275 +struct smf_track_struct {
   7.276 +	smf_t		*smf;
   7.277 +
   7.278 +	int		track_number;
   7.279 +	int		number_of_events;
   7.280 +
   7.281 +	/** These are private fields using only by loading and saving routines. */
   7.282 +	void		*file_buffer;
   7.283 +	int		file_buffer_length;
   7.284 +	int		last_status; /* Used for "running status". */
   7.285 +
   7.286 +	/** Private, used by smf.c. */
   7.287 +	/** Offset into buffer, used in parse_next_event(). */
   7.288 +	int		next_event_offset;
   7.289 +	int		next_event_number;
   7.290 +
   7.291 +	/** Absolute time of next event on events_queue. */
   7.292 +	int		time_of_next_event;
   7.293 +	struct FakeGPtrArray	*events_array;
   7.294 +
   7.295 +	/** API consumer is free to use this for whatever purpose.  NULL in freshly allocated track.
   7.296 +	    Note that tracks might be deallocated not only explicitly, by calling smf_track_delete(),
   7.297 +	    but also implicitly, e.g. when calling smf_delete() with tracks still added to
   7.298 +	    the smf; there is no mechanism for libsmf to notify you about removal of the track. */
   7.299 +	void		*user_pointer;
   7.300 +};
   7.301 +
   7.302 +typedef struct smf_track_struct smf_track_t;
   7.303 +
   7.304 +/** Represents a single MIDI event or metaevent. */
   7.305 +struct smf_event_struct {
   7.306 +	/** Pointer to the track, or NULL if event is not attached. */
   7.307 +	smf_track_t	*track;
   7.308 +
   7.309 +	/** Number of this event in the track.  Events are numbered consecutively, starting from one. */
   7.310 +	int		event_number;
   7.311 +
   7.312 +	/** Note that the time fields are invalid, if event is not attached to a track. */
   7.313 +	/** Time, in pulses, since the previous event on this track. */
   7.314 +	int		delta_time_pulses;
   7.315 +
   7.316 +	/** Time, in pulses, since the start of the song. */
   7.317 +	int		time_pulses;
   7.318 +
   7.319 +	/** Time, in seconds, since the start of the song. */
   7.320 +	double		time_seconds;
   7.321 +
   7.322 +	/** Tracks are numbered consecutively, starting from 1. */
   7.323 +	int		track_number;
   7.324 +
   7.325 +	/** Pointer to the buffer containing MIDI message.  This is freed by smf_event_delete. */
   7.326 +	unsigned char	*midi_buffer;
   7.327 +
   7.328 +	/** Length of the MIDI message in the buffer, in bytes. */
   7.329 +	int		midi_buffer_length; 
   7.330 +
   7.331 +	/** API consumer is free to use this for whatever purpose.  NULL in freshly allocated event.
   7.332 +	    Note that events might be deallocated not only explicitly, by calling smf_event_delete(),
   7.333 +	    but also implicitly, e.g. when calling smf_track_delete() with events still added to
   7.334 +	    the track; there is no mechanism for libsmf to notify you about removal of the event. */
   7.335 +	void		*user_pointer;
   7.336 +};
   7.337 +
   7.338 +typedef struct smf_event_struct smf_event_t;
   7.339 +
   7.340 +/* Routines for manipulating smf_t. */
   7.341 +smf_t *smf_new(void) WARN_UNUSED_RESULT;
   7.342 +void smf_delete(smf_t *smf);
   7.343 +
   7.344 +int smf_set_format(smf_t *smf, int format) WARN_UNUSED_RESULT;
   7.345 +int smf_set_ppqn(smf_t *smf, int format) WARN_UNUSED_RESULT;
   7.346 +
   7.347 +char *smf_decode(const smf_t *smf) WARN_UNUSED_RESULT;
   7.348 +
   7.349 +smf_track_t *smf_get_track_by_number(const smf_t *smf, int track_number) WARN_UNUSED_RESULT;
   7.350 +
   7.351 +smf_event_t *smf_peek_next_event(smf_t *smf) WARN_UNUSED_RESULT;
   7.352 +smf_event_t *smf_get_next_event(smf_t *smf) WARN_UNUSED_RESULT;
   7.353 +void smf_skip_next_event(smf_t *smf);
   7.354 +
   7.355 +void smf_rewind(smf_t *smf);
   7.356 +int smf_seek_to_seconds(smf_t *smf, double seconds) WARN_UNUSED_RESULT;
   7.357 +int smf_seek_to_pulses(smf_t *smf, int pulses) WARN_UNUSED_RESULT;
   7.358 +int smf_seek_to_event(smf_t *smf, const smf_event_t *event) WARN_UNUSED_RESULT;
   7.359 +
   7.360 +int smf_get_length_pulses(const smf_t *smf) WARN_UNUSED_RESULT;
   7.361 +double smf_get_length_seconds(const smf_t *smf) WARN_UNUSED_RESULT;
   7.362 +int smf_event_is_last(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.363 +
   7.364 +void smf_add_track(smf_t *smf, smf_track_t *track);
   7.365 +void smf_track_remove_from_smf(smf_track_t *track);
   7.366 +
   7.367 +/* Routines for manipulating smf_track_t. */
   7.368 +smf_track_t *smf_track_new(void) WARN_UNUSED_RESULT;
   7.369 +void smf_track_delete(smf_track_t *track);
   7.370 +
   7.371 +smf_event_t *smf_track_get_next_event(smf_track_t *track) WARN_UNUSED_RESULT;
   7.372 +smf_event_t *smf_track_get_event_by_number(const smf_track_t *track, int event_number) WARN_UNUSED_RESULT;
   7.373 +smf_event_t *smf_track_get_last_event(const smf_track_t *track) WARN_UNUSED_RESULT;
   7.374 +
   7.375 +void smf_track_add_event_delta_pulses(smf_track_t *track, smf_event_t *event, int pulses);
   7.376 +void smf_track_add_event_pulses(smf_track_t *track, smf_event_t *event, int pulses);
   7.377 +void smf_track_add_event_seconds(smf_track_t *track, smf_event_t *event, double seconds);
   7.378 +int smf_track_add_eot_delta_pulses(smf_track_t *track, int delta) WARN_UNUSED_RESULT;
   7.379 +int smf_track_add_eot_pulses(smf_track_t *track, int pulses) WARN_UNUSED_RESULT;
   7.380 +int smf_track_add_eot_seconds(smf_track_t *track, double seconds) WARN_UNUSED_RESULT;
   7.381 +void smf_event_remove_from_track(smf_event_t *event);
   7.382 +
   7.383 +/* Routines for manipulating smf_event_t. */
   7.384 +smf_event_t *smf_event_new(void) WARN_UNUSED_RESULT;
   7.385 +smf_event_t *smf_event_new_from_pointer(void *midi_data, int len) WARN_UNUSED_RESULT;
   7.386 +smf_event_t *smf_event_new_from_bytes(int first_byte, int second_byte, int third_byte) WARN_UNUSED_RESULT;
   7.387 +smf_event_t *smf_event_new_textual(int type, const char *text);
   7.388 +void smf_event_delete(smf_event_t *event);
   7.389 +
   7.390 +int smf_event_is_valid(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.391 +int smf_event_is_metadata(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.392 +int smf_event_is_system_realtime(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.393 +int smf_event_is_system_common(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.394 +int smf_event_is_sysex(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.395 +int smf_event_is_eot(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.396 +int smf_event_is_textual(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.397 +char *smf_event_decode(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.398 +char *smf_event_extract_text(const smf_event_t *event) WARN_UNUSED_RESULT;
   7.399 +
   7.400 +/* Routines for loading SMF files. */
   7.401 +smf_t *smf_load(const char *file_name) WARN_UNUSED_RESULT;
   7.402 +smf_t *smf_load_from_memory(const void *buffer, const int buffer_length) WARN_UNUSED_RESULT;
   7.403 +
   7.404 +/* Routine for writing SMF files. */
   7.405 +int smf_save(smf_t *smf, const char *file_name) WARN_UNUSED_RESULT;
   7.406 +
   7.407 +/* Routines for manipulating smf_tempo_t. */
   7.408 +smf_tempo_t *smf_get_tempo_by_pulses(const smf_t *smf, int pulses) WARN_UNUSED_RESULT;
   7.409 +smf_tempo_t *smf_get_tempo_by_seconds(const smf_t *smf, double seconds) WARN_UNUSED_RESULT;
   7.410 +smf_tempo_t *smf_get_tempo_by_number(const smf_t *smf, int number) WARN_UNUSED_RESULT;
   7.411 +smf_tempo_t *smf_get_last_tempo(const smf_t *smf) WARN_UNUSED_RESULT;
   7.412 +
   7.413 +const char *smf_get_version(void) WARN_UNUSED_RESULT;
   7.414 +
   7.415 +#ifdef __cplusplus
   7.416 +}
   7.417 +#endif
   7.418 +
   7.419 +#endif /* SMF_H */
   7.420 +
     8.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     8.2 +++ b/src/smf_decode.c	Thu Jan 26 11:25:11 2012 +0200
     8.3 @@ -0,0 +1,638 @@
     8.4 +/*-
     8.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
     8.6 + * All rights reserved.
     8.7 + *
     8.8 + * Redistribution and use in source and binary forms, with or without
     8.9 + * modification, are permitted provided that the following conditions
    8.10 + * are met:
    8.11 + * 1. Redistributions of source code must retain the above copyright
    8.12 + *    notice, this list of conditions and the following disclaimer.
    8.13 + * 2. Redistributions in binary form must reproduce the above copyright
    8.14 + *    notice, this list of conditions and the following disclaimer in the
    8.15 + *    documentation and/or other materials provided with the distribution.
    8.16 + *
    8.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
    8.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
    8.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
    8.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
    8.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    8.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
    8.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
    8.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
    8.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
    8.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
    8.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    8.28 + *
    8.29 + */
    8.30 +
    8.31 +/**
    8.32 + * \file
    8.33 + *
    8.34 + * Event decoding routines.
    8.35 + *
    8.36 + */
    8.37 +
    8.38 +#include <stdlib.h>
    8.39 +#include <string.h>
    8.40 +#include <assert.h>
    8.41 +#include <math.h>
    8.42 +#include <errno.h>
    8.43 +#ifdef __MINGW32__
    8.44 +#include <windows.h>
    8.45 +#else /* ! __MINGW32__ */
    8.46 +#include <arpa/inet.h>
    8.47 +#endif /* ! __MINGW32__ */
    8.48 +#include <stdint.h>
    8.49 +#include "smf.h"
    8.50 +#include "smf_private.h"
    8.51 +
    8.52 +#define BUFFER_SIZE 1024
    8.53 +
    8.54 +/**
    8.55 + * \return Nonzero if event is metaevent.  You should never send metaevents;
    8.56 + * they are not really MIDI messages.  They carry information like track title,
    8.57 + * time signature etc.
    8.58 + */
    8.59 +int
    8.60 +smf_event_is_metadata(const smf_event_t *event)
    8.61 +{
    8.62 +	assert(event->midi_buffer);
    8.63 +	assert(event->midi_buffer_length > 0);
    8.64 +
    8.65 +	if (event->midi_buffer[0] == 0xFF)
    8.66 +		return (1);
    8.67 +
    8.68 +	return (0);
    8.69 +}
    8.70 +
    8.71 +/**
    8.72 + * \return Nonzero if event is System Realtime.
    8.73 + */
    8.74 +int
    8.75 +smf_event_is_system_realtime(const smf_event_t *event)
    8.76 +{
    8.77 +	assert(event->midi_buffer);
    8.78 +	assert(event->midi_buffer_length > 0);
    8.79 +
    8.80 +	if (smf_event_is_metadata(event))
    8.81 +		return (0);
    8.82 +
    8.83 +	if (event->midi_buffer[0] >= 0xF8)
    8.84 +		return (1);
    8.85 +
    8.86 +	return (0);
    8.87 +}
    8.88 +
    8.89 +/**
    8.90 + * \return Nonzero if event is System Common.
    8.91 + */
    8.92 +int
    8.93 +smf_event_is_system_common(const smf_event_t *event)
    8.94 +{
    8.95 +	assert(event->midi_buffer);
    8.96 +	assert(event->midi_buffer_length > 0);
    8.97 +
    8.98 +	if (event->midi_buffer[0] >= 0xF0 && event->midi_buffer[0] <= 0xF7)
    8.99 +		return (1);
   8.100 +
   8.101 +	return (0);
   8.102 +}
   8.103 +/**
   8.104 +  * \return Nonzero if event is SysEx message.
   8.105 +  */
   8.106 +int
   8.107 +smf_event_is_sysex(const smf_event_t *event)
   8.108 +{
   8.109 +	assert(event->midi_buffer);
   8.110 +	assert(event->midi_buffer_length > 0);
   8.111 +
   8.112 +	if (event->midi_buffer[0] == 0xF0)
   8.113 +		return (1);
   8.114 +
   8.115 +	return (0);
   8.116 +}
   8.117 +
   8.118 +static char *
   8.119 +smf_event_decode_textual(const smf_event_t *event, const char *name)
   8.120 +{
   8.121 +	int off = 0;
   8.122 +	char *buf, *extracted;
   8.123 +
   8.124 +	buf = malloc(BUFFER_SIZE);
   8.125 +	if (buf == NULL) {
   8.126 +		fg_critical("smf_event_decode_textual: malloc failed.");
   8.127 +		return (NULL);
   8.128 +	}
   8.129 +
   8.130 +	extracted = smf_event_extract_text(event);
   8.131 +	if (extracted == NULL) {
   8.132 +		free(buf);
   8.133 +		return (NULL);
   8.134 +	}
   8.135 +
   8.136 +	snprintf(buf + off, BUFFER_SIZE - off, "%s: %s", name, extracted);
   8.137 +
   8.138 +	return (buf);
   8.139 +}
   8.140 +
   8.141 +static char *
   8.142 +smf_event_decode_metadata(const smf_event_t *event)
   8.143 +{
   8.144 +	int off = 0, mspqn, flats, isminor;
   8.145 +	char *buf;
   8.146 +
   8.147 +	static const char *const major_keys[] = {"Fb", "Cb", "Gb", "Db", "Ab",
   8.148 +		"Eb", "Bb", "F", "C", "G", "D", "A", "E", "B", "F#", "C#", "G#"};
   8.149 +
   8.150 +	static const char *const minor_keys[] = {"Dbm", "Abm", "Ebm", "Bbm", "Fm",
   8.151 +		"Cm", "Gm", "Dm", "Am", "Em", "Bm", "F#m", "C#m", "G#m", "D#m", "A#m", "E#m"};
   8.152 +
   8.153 +	assert(smf_event_is_metadata(event));
   8.154 +
   8.155 +	switch (event->midi_buffer[1]) {
   8.156 +		case 0x01:
   8.157 +			return (smf_event_decode_textual(event, "Text"));
   8.158 +
   8.159 +		case 0x02:
   8.160 +			return (smf_event_decode_textual(event, "Copyright"));
   8.161 +
   8.162 +		case 0x03:
   8.163 +			return (smf_event_decode_textual(event, "Sequence/Track Name"));
   8.164 +
   8.165 +		case 0x04:
   8.166 +			return (smf_event_decode_textual(event, "Instrument"));
   8.167 +
   8.168 +		case 0x05:
   8.169 +			return (smf_event_decode_textual(event, "Lyric"));
   8.170 +
   8.171 +		case 0x06:
   8.172 +			return (smf_event_decode_textual(event, "Marker"));
   8.173 +
   8.174 +		case 0x07:
   8.175 +			return (smf_event_decode_textual(event, "Cue Point"));
   8.176 +
   8.177 +		case 0x08:
   8.178 +			return (smf_event_decode_textual(event, "Program Name"));
   8.179 +
   8.180 +		case 0x09:
   8.181 +			return (smf_event_decode_textual(event, "Device (Port) Name"));
   8.182 +
   8.183 +		default:
   8.184 +			break;
   8.185 +	}
   8.186 +
   8.187 +	buf = malloc(BUFFER_SIZE);
   8.188 +	if (buf == NULL) {
   8.189 +		fg_critical("smf_event_decode_metadata: malloc failed.");
   8.190 +		return (NULL);
   8.191 +	}
   8.192 +
   8.193 +	switch (event->midi_buffer[1]) {
   8.194 +		case 0x00:
   8.195 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Sequence number");
   8.196 +			break;
   8.197 +
   8.198 +		/* http://music.columbia.edu/pipermail/music-dsp/2004-August/061196.html */
   8.199 +		case 0x20:
   8.200 +			if (event->midi_buffer_length < 4) {
   8.201 +				fg_critical("smf_event_decode_metadata: truncated MIDI message.");
   8.202 +				goto error;
   8.203 +			}
   8.204 +
   8.205 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Channel Prefix: %d", event->midi_buffer[3]);
   8.206 +			break;
   8.207 +
   8.208 +		case 0x21:
   8.209 +			if (event->midi_buffer_length < 4) {
   8.210 +				fg_critical("smf_event_decode_metadata: truncated MIDI message.");
   8.211 +				goto error;
   8.212 +			}
   8.213 +
   8.214 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MIDI Port: %d", event->midi_buffer[3]);
   8.215 +			break;
   8.216 +
   8.217 +		case 0x2F:
   8.218 +			off += snprintf(buf + off, BUFFER_SIZE - off, "End Of Track");
   8.219 +			break;
   8.220 +
   8.221 +		case 0x51:
   8.222 +			if (event->midi_buffer_length < 6) {
   8.223 +				fg_critical("smf_event_decode_metadata: truncated MIDI message.");
   8.224 +				goto error;
   8.225 +			}
   8.226 +
   8.227 +			mspqn = (event->midi_buffer[3] << 16) + (event->midi_buffer[4] << 8) + event->midi_buffer[5];
   8.228 +
   8.229 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Tempo: %d microseconds per quarter note, %.2f BPM",
   8.230 +				mspqn, 60000000.0 / (double)mspqn);
   8.231 +			break;
   8.232 +
   8.233 +		case 0x54:
   8.234 +			off += snprintf(buf + off, BUFFER_SIZE - off, "SMPTE Offset");
   8.235 +			break;
   8.236 +
   8.237 +		case 0x58:
   8.238 +			if (event->midi_buffer_length < 7) {
   8.239 +				fg_critical("smf_event_decode_metadata: truncated MIDI message.");
   8.240 +				goto error;
   8.241 +			}
   8.242 +
   8.243 +			off += snprintf(buf + off, BUFFER_SIZE - off,
   8.244 +				"Time Signature: %d/%d, %d clocks per click, %d notated 32nd notes per quarter note",
   8.245 +				event->midi_buffer[3], (int)pow(2, event->midi_buffer[4]), event->midi_buffer[5],
   8.246 +				event->midi_buffer[6]);
   8.247 +			break;
   8.248 +
   8.249 +		case 0x59:
   8.250 +			if (event->midi_buffer_length < 5) {
   8.251 +				fg_critical("smf_event_decode_metadata: truncated MIDI message.");
   8.252 +				goto error;
   8.253 +			}
   8.254 +
   8.255 +			flats = event->midi_buffer[3];
   8.256 +			isminor = event->midi_buffer[4];
   8.257 +
   8.258 +			if (isminor != 0 && isminor != 1) {
   8.259 +				fg_critical("smf_event_decode_metadata: last byte of the Key Signature event has invalid value %d.", isminor);
   8.260 +				goto error;
   8.261 +			}
   8.262 +
   8.263 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Key Signature: ");
   8.264 +
   8.265 +			if (flats > 8 && flats < 248) {
   8.266 +				off += snprintf(buf + off, BUFFER_SIZE - off, "%d %s, %s key", abs((int8_t)flats),
   8.267 +					flats > 127 ? "flats" : "sharps", isminor ? "minor" : "major");
   8.268 +			} else {
   8.269 +				int i = (flats - 248) & 255;
   8.270 +
   8.271 +				assert(i >= 0 && i < sizeof(minor_keys) / sizeof(*minor_keys));
   8.272 +				assert(i >= 0 && i < sizeof(major_keys) / sizeof(*major_keys));
   8.273 +
   8.274 +				if (isminor)
   8.275 +					off += snprintf(buf + off, BUFFER_SIZE - off, "%s", minor_keys[i]);
   8.276 +				else
   8.277 +					off += snprintf(buf + off, BUFFER_SIZE - off, "%s", major_keys[i]);
   8.278 +			}
   8.279 +
   8.280 +			break;
   8.281 +
   8.282 +		case 0x7F:
   8.283 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Proprietary (aka Sequencer) Event, length %d",
   8.284 +				event->midi_buffer_length);
   8.285 +			break;
   8.286 +
   8.287 +		default:
   8.288 +			goto error;
   8.289 +	}
   8.290 +
   8.291 +	return (buf);
   8.292 +
   8.293 +error:
   8.294 +	free(buf);
   8.295 +
   8.296 +	return (NULL);
   8.297 +}
   8.298 +
   8.299 +static char *
   8.300 +smf_event_decode_system_realtime(const smf_event_t *event)
   8.301 +{
   8.302 +	int off = 0;
   8.303 +	char *buf;
   8.304 +
   8.305 +	assert(smf_event_is_system_realtime(event));
   8.306 +
   8.307 +	if (event->midi_buffer_length != 1) {
   8.308 +		fg_critical("smf_event_decode_system_realtime: event length is not 1.");
   8.309 +		return (NULL);
   8.310 +	}
   8.311 +
   8.312 +	buf = malloc(BUFFER_SIZE);
   8.313 +	if (buf == NULL) {
   8.314 +		fg_critical("smf_event_decode_system_realtime: malloc failed.");
   8.315 +		return (NULL);
   8.316 +	}
   8.317 +
   8.318 +	switch (event->midi_buffer[0]) {
   8.319 +		case 0xF8:
   8.320 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MIDI Clock (realtime)");
   8.321 +			break;
   8.322 +
   8.323 +		case 0xF9:
   8.324 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Tick (realtime)");
   8.325 +			break;
   8.326 +
   8.327 +		case 0xFA:
   8.328 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MIDI Start (realtime)");
   8.329 +			break;
   8.330 +
   8.331 +		case 0xFB:
   8.332 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MIDI Continue (realtime)");
   8.333 +			break;
   8.334 +
   8.335 +		case 0xFC:
   8.336 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MIDI Stop (realtime)");
   8.337 +			break;
   8.338 +
   8.339 +		case 0xFE:
   8.340 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Active Sense (realtime)");
   8.341 +			break;
   8.342 +
   8.343 +		default:
   8.344 +			free(buf);
   8.345 +			return (NULL);
   8.346 +	}
   8.347 +
   8.348 +	return (buf);
   8.349 +}
   8.350 +
   8.351 +static char *
   8.352 +smf_event_decode_sysex(const smf_event_t *event)
   8.353 +{
   8.354 +	int off = 0;
   8.355 +	char *buf, manufacturer, subid, subid2;
   8.356 +
   8.357 +	assert(smf_event_is_sysex(event));
   8.358 +
   8.359 +	if (event->midi_buffer_length < 5) {
   8.360 +		fg_critical("smf_event_decode_sysex: truncated MIDI message.");
   8.361 +		return (NULL);
   8.362 +	}
   8.363 +
   8.364 +	buf = malloc(BUFFER_SIZE);
   8.365 +	if (buf == NULL) {
   8.366 +		fg_critical("smf_event_decode_sysex: malloc failed.");
   8.367 +		return (NULL);
   8.368 +	}
   8.369 +
   8.370 +	manufacturer = event->midi_buffer[1];
   8.371 +
   8.372 +	if (manufacturer == 0x7F) {
   8.373 +		off += snprintf(buf + off, BUFFER_SIZE - off, "SysEx, realtime, channel %d", event->midi_buffer[2]);
   8.374 +	} else if (manufacturer == 0x7E) {
   8.375 +		off += snprintf(buf + off, BUFFER_SIZE - off, "SysEx, non-realtime, channel %d", event->midi_buffer[2]);
   8.376 +	} else {
   8.377 +		off += snprintf(buf + off, BUFFER_SIZE - off, "SysEx, manufacturer 0x%x", manufacturer);
   8.378 +
   8.379 +		return (buf);
   8.380 +	}
   8.381 +
   8.382 +	subid = event->midi_buffer[3];
   8.383 +	subid2 = event->midi_buffer[4];
   8.384 +
   8.385 +	if (subid == 0x01)
   8.386 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Header");
   8.387 +
   8.388 +	else if (subid == 0x02)
   8.389 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Data Packet");
   8.390 +
   8.391 +	else if (subid == 0x03)
   8.392 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Request");
   8.393 +
   8.394 +	else if (subid == 0x04 && subid2 == 0x01)
   8.395 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Master Volume");
   8.396 +
   8.397 +	else if (subid == 0x05 && subid2 == 0x01)
   8.398 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Loop Point Retransmit");
   8.399 +
   8.400 +	else if (subid == 0x05 && subid2 == 0x02)
   8.401 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Loop Point Request");
   8.402 +
   8.403 +	else if (subid == 0x06 && subid2 == 0x01)
   8.404 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Identity Request");
   8.405 +
   8.406 +	else if (subid == 0x06 && subid2 == 0x02)
   8.407 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Identity Reply");
   8.408 +
   8.409 +	else if (subid == 0x08 && subid2 == 0x00)
   8.410 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Bulk Tuning Dump Request");
   8.411 +
   8.412 +	else if (subid == 0x08 && subid2 == 0x01)
   8.413 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Bulk Tuning Dump");
   8.414 +
   8.415 +	else if (subid == 0x08 && subid2 == 0x02)
   8.416 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Single Note Tuning Change");
   8.417 +
   8.418 +	else if (subid == 0x08 && subid2 == 0x03)
   8.419 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Bulk Tuning Dump Request (Bank)");
   8.420 +
   8.421 +	else if (subid == 0x08 && subid2 == 0x04)
   8.422 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Key Based Tuning Dump");
   8.423 +
   8.424 +	else if (subid == 0x08 && subid2 == 0x05)
   8.425 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Scale/Octave Tuning Dump, 1 byte format");
   8.426 +
   8.427 +	else if (subid == 0x08 && subid2 == 0x06)
   8.428 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Scale/Octave Tuning Dump, 2 byte format");
   8.429 +
   8.430 +	else if (subid == 0x08 && subid2 == 0x07)
   8.431 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Single Note Tuning Change (Bank)");
   8.432 +
   8.433 +	else if (subid == 0x09)
   8.434 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", General MIDI %s", subid2 == 0 ? "disable" : "enable");
   8.435 +
   8.436 +	else if (subid == 0x7C)
   8.437 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Wait");
   8.438 +
   8.439 +	else if (subid == 0x7D)
   8.440 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump Cancel");
   8.441 +
   8.442 +	else if (subid == 0x7E)
   8.443 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump NAK");
   8.444 +
   8.445 +	else if (subid == 0x7F)
   8.446 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Sample Dump ACK");
   8.447 +
   8.448 +	else
   8.449 +		off += snprintf(buf + off, BUFFER_SIZE - off, ", Unknown");
   8.450 +
   8.451 +	return (buf);
   8.452 +}
   8.453 +
   8.454 +static char *
   8.455 +smf_event_decode_system_common(const smf_event_t *event)
   8.456 +{
   8.457 +	int off = 0;
   8.458 +	char *buf;
   8.459 +
   8.460 +	assert(smf_event_is_system_common(event));
   8.461 +
   8.462 +	if (smf_event_is_sysex(event))
   8.463 +		return (smf_event_decode_sysex(event));
   8.464 +
   8.465 +	buf = malloc(BUFFER_SIZE);
   8.466 +	if (buf == NULL) {
   8.467 +		fg_critical("smf_event_decode_system_realtime: malloc failed.");
   8.468 +		return (NULL);
   8.469 +	}
   8.470 +
   8.471 +	switch (event->midi_buffer[0]) {
   8.472 +		case 0xF1:
   8.473 +			off += snprintf(buf + off, BUFFER_SIZE - off, "MTC Quarter Frame");
   8.474 +			break;
   8.475 +
   8.476 +		case 0xF2:
   8.477 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Song Position Pointer");
   8.478 +			break;
   8.479 +
   8.480 +		case 0xF3:
   8.481 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Song Select");
   8.482 +			break;
   8.483 +
   8.484 +		case 0xF6:
   8.485 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Tune Request");
   8.486 +			break;
   8.487 +
   8.488 +		default:
   8.489 +			free(buf);
   8.490 +			return (NULL);
   8.491 +	}
   8.492 +
   8.493 +	return (buf);
   8.494 +}
   8.495 +
   8.496 +static void
   8.497 +note_from_int(char *buf, int note_number)
   8.498 +{
   8.499 +	int note, octave;
   8.500 +	char *names[] = {"C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"};
   8.501 +
   8.502 +	octave = note_number / 12 - 1;
   8.503 +	note = note_number % 12;
   8.504 +
   8.505 +	sprintf(buf, "%s%d", names[note], octave);
   8.506 +}
   8.507 +
   8.508 +/**
   8.509 + * \return Textual representation of the event given, or NULL, if event is unknown.
   8.510 + * Returned string looks like this:
   8.511 + *
   8.512 + * Note On, channel 1, note F#3, velocity 0
   8.513 + *
   8.514 + * You should free the returned string afterwards, using free(3).
   8.515 + */
   8.516 +char *
   8.517 +smf_event_decode(const smf_event_t *event)
   8.518 +{
   8.519 +	int off = 0, channel;
   8.520 +	char *buf, note[5];
   8.521 +
   8.522 +	if (smf_event_is_metadata(event))
   8.523 +		return (smf_event_decode_metadata(event));
   8.524 +
   8.525 +	if (smf_event_is_system_realtime(event))
   8.526 +		return (smf_event_decode_system_realtime(event));
   8.527 +
   8.528 +	if (smf_event_is_system_common(event))
   8.529 +		return (smf_event_decode_system_common(event));
   8.530 +
   8.531 +	if (!smf_event_length_is_valid(event)) {
   8.532 +		fg_critical("smf_event_decode: incorrect MIDI message length.");
   8.533 +		return (NULL);
   8.534 +	}
   8.535 +
   8.536 +	buf = malloc(BUFFER_SIZE);
   8.537 +	if (buf == NULL) {
   8.538 +		fg_critical("smf_event_decode: malloc failed.");
   8.539 +		return (NULL);
   8.540 +	}
   8.541 +
   8.542 +	/* + 1, because user-visible channels used to be in range <1-16>. */
   8.543 +	channel = (event->midi_buffer[0] & 0x0F) + 1;
   8.544 +
   8.545 +	switch (event->midi_buffer[0] & 0xF0) {
   8.546 +		case 0x80:
   8.547 +			note_from_int(note, event->midi_buffer[1]);
   8.548 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Note Off, channel %d, note %s, velocity %d",
   8.549 +					channel, note, event->midi_buffer[2]);
   8.550 +			break;
   8.551 +
   8.552 +		case 0x90:
   8.553 +			note_from_int(note, event->midi_buffer[1]);
   8.554 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Note On, channel %d, note %s, velocity %d",
   8.555 +					channel, note, event->midi_buffer[2]);
   8.556 +			break;
   8.557 +
   8.558 +		case 0xA0:
   8.559 +			note_from_int(note, event->midi_buffer[1]);
   8.560 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Aftertouch, channel %d, note %s, pressure %d",
   8.561 +					channel, note, event->midi_buffer[2]);
   8.562 +			break;
   8.563 +
   8.564 +		case 0xB0:
   8.565 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Controller, channel %d, controller %d, value %d",
   8.566 +					channel, event->midi_buffer[1], event->midi_buffer[2]);
   8.567 +			break;
   8.568 +
   8.569 +		case 0xC0:
   8.570 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Program Change, channel %d, controller %d",
   8.571 +					channel, event->midi_buffer[1]);
   8.572 +			break;
   8.573 +
   8.574 +		case 0xD0:
   8.575 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Channel Pressure, channel %d, pressure %d",
   8.576 +					channel, event->midi_buffer[1]);
   8.577 +			break;
   8.578 +
   8.579 +		case 0xE0:
   8.580 +			off += snprintf(buf + off, BUFFER_SIZE - off, "Pitch Wheel, channel %d, value %d",
   8.581 +					channel, ((int)event->midi_buffer[2] << 7) | (int)event->midi_buffer[2]);
   8.582 +			break;
   8.583 +
   8.584 +		default:
   8.585 +			free(buf);
   8.586 +			return (NULL);
   8.587 +	}
   8.588 +
   8.589 +	return (buf);
   8.590 +}
   8.591 +
   8.592 +/**
   8.593 + * \return Textual representation of the data extracted from MThd header, or NULL, if something goes wrong.
   8.594 + * Returned string looks like this:
   8.595 + *
   8.596 + * format: 1 (several simultaneous tracks); number of tracks: 4; division: 192 PPQN.
   8.597 + *
   8.598 + * You should free the returned string afterwards, using free(3).
   8.599 + */
   8.600 +char *
   8.601 +smf_decode(const smf_t *smf)
   8.602 +{
   8.603 +	int off = 0;
   8.604 +	char *buf;
   8.605 +
   8.606 +	buf = malloc(BUFFER_SIZE);
   8.607 +	if (buf == NULL) {
   8.608 +		fg_critical("smf_event_decode: malloc failed.");
   8.609 +		return (NULL);
   8.610 +	}
   8.611 +
   8.612 +	off += snprintf(buf + off, BUFFER_SIZE - off, "format: %d ", smf->format);
   8.613 +
   8.614 +	switch (smf->format) {
   8.615 +		case 0:
   8.616 +			off += snprintf(buf + off, BUFFER_SIZE - off, "(single track)");
   8.617 +			break;
   8.618 +
   8.619 +		case 1:
   8.620 +			off += snprintf(buf + off, BUFFER_SIZE - off, "(several simultaneous tracks)");
   8.621 +			break;
   8.622 +
   8.623 +		case 2:
   8.624 +			off += snprintf(buf + off, BUFFER_SIZE - off, "(several independent tracks)");
   8.625 +			break;
   8.626 +
   8.627 +		default:
   8.628 +			off += snprintf(buf + off, BUFFER_SIZE - off, "(INVALID FORMAT)");
   8.629 +			break;
   8.630 +	}
   8.631 +
   8.632 +	off += snprintf(buf + off, BUFFER_SIZE - off, "; number of tracks: %d", smf->number_of_tracks);
   8.633 +
   8.634 +	if (smf->ppqn != 0)
   8.635 +		off += snprintf(buf + off, BUFFER_SIZE - off, "; division: %d PPQN", smf->ppqn);
   8.636 +	else
   8.637 +		off += snprintf(buf + off, BUFFER_SIZE - off, "; division: %d FPS, %d resolution", smf->frames_per_second, smf->resolution);
   8.638 +
   8.639 +	return (buf);
   8.640 +}
   8.641 +
     9.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     9.2 +++ b/src/smf_load.c	Thu Jan 26 11:25:11 2012 +0200
     9.3 @@ -0,0 +1,933 @@
     9.4 +/*-
     9.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
     9.6 + * All rights reserved.
     9.7 + *
     9.8 + * Redistribution and use in source and binary forms, with or without
     9.9 + * modification, are permitted provided that the following conditions
    9.10 + * are met:
    9.11 + * 1. Redistributions of source code must retain the above copyright
    9.12 + *    notice, this list of conditions and the following disclaimer.
    9.13 + * 2. Redistributions in binary form must reproduce the above copyright
    9.14 + *    notice, this list of conditions and the following disclaimer in the
    9.15 + *    documentation and/or other materials provided with the distribution.
    9.16 + *
    9.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
    9.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
    9.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
    9.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
    9.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    9.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
    9.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
    9.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
    9.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
    9.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
    9.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    9.28 + *
    9.29 + */
    9.30 +
    9.31 +/**
    9.32 + * \file
    9.33 + *
    9.34 + * Standard MIDI File format loader.
    9.35 + *
    9.36 + */
    9.37 +
    9.38 +/* Reference: http://www.borg.com/~jglatt/tech/midifile.htm */
    9.39 +
    9.40 +#include <stdlib.h>
    9.41 +#include <string.h>
    9.42 +#include <assert.h>
    9.43 +#include <math.h>
    9.44 +#include <errno.h>
    9.45 +#include <ctype.h>
    9.46 +#ifdef __MINGW32__
    9.47 +#include <windows.h>
    9.48 +#else /* ! __MINGW32__ */
    9.49 +#include <arpa/inet.h>
    9.50 +#endif /* ! __MINGW32__ */
    9.51 +#include "smf.h"
    9.52 +#include "smf_private.h"
    9.53 +
    9.54 +/**
    9.55 + * Returns pointer to the next SMF chunk in smf->buffer, based on length of the previous one.
    9.56 + * Returns NULL in case of error.
    9.57 + */
    9.58 +static struct chunk_header_struct *
    9.59 +next_chunk(smf_t *smf)
    9.60 +{
    9.61 +	struct chunk_header_struct *chunk;
    9.62 +	void *next_chunk_ptr;
    9.63 +
    9.64 +	assert(smf->file_buffer != NULL);
    9.65 +	assert(smf->file_buffer_length > 0);
    9.66 +	assert(smf->next_chunk_offset >= 0);
    9.67 +
    9.68 +	if (smf->next_chunk_offset + sizeof(struct chunk_header_struct) >= smf->file_buffer_length) {
    9.69 +		fg_critical("SMF warning: no more chunks left.");
    9.70 +		return (NULL);
    9.71 +	}
    9.72 +
    9.73 +	next_chunk_ptr = (unsigned char *)smf->file_buffer + smf->next_chunk_offset;
    9.74 +
    9.75 +	chunk = (struct chunk_header_struct *)next_chunk_ptr;
    9.76 +
    9.77 +	if (!isalpha(chunk->id[0]) || !isalpha(chunk->id[1]) || !isalpha(chunk->id[2]) || !isalpha(chunk->id[3])) {
    9.78 +		fg_critical("SMF error: chunk signature contains at least one non-alphanumeric byte.");
    9.79 +		return (NULL);
    9.80 +	}
    9.81 +
    9.82 +	/*
    9.83 +	 * XXX: On SPARC, after compiling with "-fast" option there will be SIGBUS here.
    9.84 +	 * Please compile with -xmemalign=8i".
    9.85 +	 */
    9.86 +	smf->next_chunk_offset += sizeof(struct chunk_header_struct) + ntohl(chunk->length);
    9.87 +
    9.88 +	if (smf->next_chunk_offset > smf->file_buffer_length) {
    9.89 +		fg_critical("SMF warning: malformed chunk; truncated file?");
    9.90 +		smf->next_chunk_offset = smf->file_buffer_length;
    9.91 +	}
    9.92 +
    9.93 +	return (chunk);
    9.94 +}
    9.95 +
    9.96 +/**
    9.97 + * Returns 1, iff signature of the "chunk" is the same as string passed as "signature".
    9.98 + */
    9.99 +static int
   9.100 +chunk_signature_matches(const struct chunk_header_struct *chunk, const char *signature)
   9.101 +{
   9.102 +	if (!memcmp(chunk->id, signature, 4))
   9.103 +		return (1);
   9.104 +
   9.105 +	return (0);
   9.106 +}
   9.107 +
   9.108 +/**
   9.109 + * Verifies if MThd header looks OK.  Returns 0 iff it does.
   9.110 + */
   9.111 +static int
   9.112 +parse_mthd_header(smf_t *smf)
   9.113 +{
   9.114 +	int len;
   9.115 +	struct chunk_header_struct *mthd, *tmp_mthd;
   9.116 +
   9.117 +	/* Make sure compiler didn't do anything stupid. */
   9.118 +	assert(sizeof(struct chunk_header_struct) == 8);
   9.119 +
   9.120 +	/*
   9.121 +	 * We could just do "mthd = smf->file_buffer;" here, but this way we wouldn't
   9.122 +	 * get useful error messages.
   9.123 +	 */
   9.124 +	if (smf->file_buffer_length < 6) {
   9.125 +		fg_critical("SMF error: file is too short, it cannot be a MIDI file.");
   9.126 +
   9.127 +		return (-1);
   9.128 +	}
   9.129 +
   9.130 +	tmp_mthd = smf->file_buffer;
   9.131 +
   9.132 +	if (!chunk_signature_matches(tmp_mthd, "MThd")) {
   9.133 +		fg_critical("SMF error: MThd signature not found, is that a MIDI file?");
   9.134 +	
   9.135 +		return (-2);
   9.136 +	}
   9.137 +
   9.138 +	/* Ok, now use next_chunk(). */
   9.139 +	mthd = next_chunk(smf);
   9.140 +	if (mthd == NULL)
   9.141 +		return (-3);
   9.142 +
   9.143 +	assert(mthd == tmp_mthd);
   9.144 +
   9.145 +	len = ntohl(mthd->length);
   9.146 +	if (len != 6) {
   9.147 +		fg_critical("SMF error: MThd chunk length %d, must be 6.", len);
   9.148 +
   9.149 +		return (-4);
   9.150 +	}
   9.151 +
   9.152 +	return (0);
   9.153 +}
   9.154 +
   9.155 +/**
   9.156 + * Parses MThd chunk, filling "smf" structure with values extracted from it.  Returns 0 iff everything went OK.
   9.157 + */
   9.158 +static int
   9.159 +parse_mthd_chunk(smf_t *smf)
   9.160 +{
   9.161 +	signed char first_byte_of_division, second_byte_of_division;
   9.162 +
   9.163 +	struct mthd_chunk_struct *mthd;
   9.164 +
   9.165 +	assert(sizeof(struct mthd_chunk_struct) == 14);
   9.166 +
   9.167 +	if (parse_mthd_header(smf))
   9.168 +		return (1);
   9.169 +
   9.170 +	mthd = (struct mthd_chunk_struct *)smf->file_buffer;
   9.171 +
   9.172 +	smf->format = ntohs(mthd->format);
   9.173 +	if (smf->format < 0 || smf->format > 2) {
   9.174 +		fg_critical("SMF error: bad MThd format field value: %d, valid values are 0-2, inclusive.", smf->format);
   9.175 +		return (-1);
   9.176 +	}
   9.177 +
   9.178 +	if (smf->format == 2) {
   9.179 +		fg_critical("SMF file uses format #2, no support for that yet.");
   9.180 +		return (-2);
   9.181 +	}
   9.182 +
   9.183 +	smf->expected_number_of_tracks = ntohs(mthd->number_of_tracks);
   9.184 +	if (smf->expected_number_of_tracks <= 0) {
   9.185 +		fg_critical("SMF error: bad number of tracks: %d, must be greater than zero.", smf->expected_number_of_tracks);
   9.186 +		return (-3);
   9.187 +	}
   9.188 +
   9.189 +	/* XXX: endianess? */
   9.190 +	first_byte_of_division = *((signed char *)&(mthd->division));
   9.191 +	second_byte_of_division = *((signed char *)&(mthd->division) + 1);
   9.192 +
   9.193 +	if (first_byte_of_division >= 0) {
   9.194 +		smf->ppqn = ntohs(mthd->division);
   9.195 +		smf->frames_per_second = 0;
   9.196 +		smf->resolution = 0;
   9.197 +	} else {
   9.198 +		smf->ppqn = 0;
   9.199 +		smf->frames_per_second = - first_byte_of_division;
   9.200 +		smf->resolution = second_byte_of_division;
   9.201 +	}
   9.202 +
   9.203 +	if (smf->ppqn == 0) {
   9.204 +		fg_critical("SMF file uses FPS timing instead of PPQN, no support for that yet.");
   9.205 +		return (-4);
   9.206 +	}
   9.207 +
   9.208 +	return (0);
   9.209 +}
   9.210 +
   9.211 +/**
   9.212 + * Interprets Variable Length Quantity pointed at by "buf" and puts its value into "value" and number
   9.213 + * of bytes consumed into "len", making sure it does not read past "buf" + "buffer_length".
   9.214 + * Explanation of Variable Length Quantities is here: http://www.borg.com/~jglatt/tech/midifile/vari.htm
   9.215 + * Returns 0 iff everything went OK, different value in case of error.
   9.216 + */
   9.217 +static int
   9.218 +extract_vlq(const unsigned char *buf, const int buffer_length, int *value, int *len)
   9.219 +{
   9.220 +	int val = 0;
   9.221 +	const unsigned char *c = buf;
   9.222 +
   9.223 +	assert(buffer_length > 0);
   9.224 +
   9.225 +	for (;;) {
   9.226 +		if (c >= buf + buffer_length) {
   9.227 +			fg_critical("End of buffer in extract_vlq().");
   9.228 +			return (-1);
   9.229 +		}
   9.230 +
   9.231 +		val = (val << 7) + (*c & 0x7F);
   9.232 +
   9.233 +		if (*c & 0x80)
   9.234 +			c++;
   9.235 +		else
   9.236 +			break;
   9.237 +	};
   9.238 +
   9.239 +	*value = val;
   9.240 +	*len = c - buf + 1;
   9.241 +
   9.242 +	if (*len > 4) {
   9.243 +		fg_critical("SMF error: Variable Length Quantities longer than four bytes are not supported yet.");
   9.244 +		return (-2);
   9.245 +	}
   9.246 +
   9.247 +	return (0);
   9.248 +}
   9.249 +
   9.250 +/**
   9.251 + * Returns 1 if the given byte is a valid status byte, 0 otherwise.
   9.252 + */
   9.253 +int
   9.254 +is_status_byte(const unsigned char status)
   9.255 +{
   9.256 +	return (status & 0x80);
   9.257 +}
   9.258 +
   9.259 +static int
   9.260 +is_sysex_byte(const unsigned char status)
   9.261 +{
   9.262 +	if (status == 0xF0)
   9.263 +		return (1);
   9.264 +
   9.265 +	return (0);
   9.266 +}
   9.267 +
   9.268 +static int
   9.269 +is_escape_byte(const unsigned char status)
   9.270 +{
   9.271 +	if (status == 0xF7)
   9.272 +		return (1);
   9.273 +
   9.274 +	return (0);
   9.275 +}
   9.276 +
   9.277 +/**
   9.278 + * Just like expected_message_length(), but only for System Exclusive messages.
   9.279 + * Note that value returned by this thing here is the length of SysEx "on the wire",
   9.280 + * not the number of bytes that this sysex takes in the file - in SMF format sysex
   9.281 + * contains VLQ telling how many bytes it takes, "on the wire" format does not have
   9.282 + * this.
   9.283 + */
   9.284 +static int
   9.285 +expected_sysex_length(const unsigned char status, const unsigned char *second_byte, const int buffer_length, int *consumed_bytes)
   9.286 +{
   9.287 +	int sysex_length, len;
   9.288 +
   9.289 +	assert(status == 0xF0);
   9.290 +
   9.291 +	if (buffer_length < 3) {
   9.292 +		fg_critical("SMF error: end of buffer in expected_sysex_length().");
   9.293 +		return (-1);
   9.294 +	}
   9.295 +
   9.296 +	if (extract_vlq(second_byte, buffer_length, &sysex_length, &len))
   9.297 +		return (-1);
   9.298 +
   9.299 +	if (consumed_bytes != NULL)
   9.300 +		*consumed_bytes = len;
   9.301 +
   9.302 +	/* +1, because the length does not include status byte. */
   9.303 +	return (sysex_length + 1);
   9.304 +}
   9.305 +
   9.306 +static int
   9.307 +expected_escaped_length(const unsigned char status, const unsigned char *second_byte, const int buffer_length, int *consumed_bytes)
   9.308 +{
   9.309 +	/* -1, because we do not want to account for 0x7F status. */
   9.310 +	return (expected_sysex_length(status, second_byte, buffer_length, consumed_bytes) - 1);
   9.311 +}
   9.312 +
   9.313 +/**
   9.314 + * Returns expected length of the midi message (including the status byte), in bytes, for the given status byte.
   9.315 + * The "second_byte" points to the expected second byte of the MIDI message.  "buffer_length" is the buffer
   9.316 + * length limit, counting from "second_byte".  Returns value < 0 iff there was an error.
   9.317 + */
   9.318 +static int
   9.319 +expected_message_length(unsigned char status, const unsigned char *second_byte, const int buffer_length)
   9.320 +{
   9.321 +	/* Make sure this really is a valid status byte. */
   9.322 +	assert(is_status_byte(status));
   9.323 +
   9.324 +	/* We cannot use this routine for sysexes. */
   9.325 +	assert(!is_sysex_byte(status));
   9.326 +
   9.327 +	/* We cannot use this routine for escaped events. */
   9.328 +	assert(!is_escape_byte(status));
   9.329 +
   9.330 +	/* Buffer length may be zero, for e.g. realtime messages. */
   9.331 +	assert(buffer_length >= 0);
   9.332 +
   9.333 +	/* Is this a metamessage? */
   9.334 +	if (status == 0xFF) {
   9.335 +		if (buffer_length < 2) {
   9.336 +			fg_critical("SMF error: end of buffer in expected_message_length().");
   9.337 +			return (-1);
   9.338 +		}
   9.339 +
   9.340 +		/*
   9.341 +		 * Format of this kind of messages is like this: 0xFF 0xwhatever 0xlength and then "length" bytes.
   9.342 +		 * Second byte points to this:                        ^^^^^^^^^^
   9.343 +		 */
   9.344 +		return (*(second_byte + 1) + 3);
   9.345 +	}
   9.346 +
   9.347 +	if ((status & 0xF0) == 0xF0) {
   9.348 +		switch (status) {
   9.349 +			case 0xF2: /* Song Position Pointer. */
   9.350 +				return (3);
   9.351 +
   9.352 +			case 0xF1: /* MTC Quarter Frame. */
   9.353 +			case 0xF3: /* Song Select. */
   9.354 +				return (2);
   9.355 +
   9.356 +			case 0xF6: /* Tune Request. */
   9.357 +			case 0xF8: /* MIDI Clock. */
   9.358 +			case 0xF9: /* Tick. */
   9.359 +			case 0xFA: /* MIDI Start. */
   9.360 +			case 0xFB: /* MIDI Continue. */
   9.361 +			case 0xFC: /* MIDI Stop. */
   9.362 +			case 0xFE: /* Active Sense. */
   9.363 +				return (1);
   9.364 +
   9.365 +			default:
   9.366 +				fg_critical("SMF error: unknown 0xFx-type status byte '0x%x'.", status);
   9.367 +				return (-2);
   9.368 +		}
   9.369 +	}
   9.370 +
   9.371 +	/* Filter out the channel. */
   9.372 +	status &= 0xF0;
   9.373 +
   9.374 +	switch (status) {
   9.375 +		case 0x80: /* Note Off. */
   9.376 +		case 0x90: /* Note On. */
   9.377 +		case 0xA0: /* AfterTouch. */
   9.378 +		case 0xB0: /* Control Change. */
   9.379 +		case 0xE0: /* Pitch Wheel. */
   9.380 +			return (3);
   9.381 +
   9.382 +		case 0xC0: /* Program Change. */
   9.383 +		case 0xD0: /* Channel Pressure. */
   9.384 +			return (2);
   9.385 +
   9.386 +		default:
   9.387 +			fg_critical("SMF error: unknown status byte '0x%x'.", status);
   9.388 +			return (-3);
   9.389 +	}
   9.390 +}
   9.391 +
   9.392 +static int
   9.393 +extract_sysex_event(const unsigned char *buf, const int buffer_length, smf_event_t *event, int *len, int last_status)
   9.394 +{
   9.395 +	int status, message_length, vlq_length;
   9.396 +	const unsigned char *c = buf;
   9.397 +
   9.398 +	status = *buf;
   9.399 +
   9.400 +	assert(is_sysex_byte(status));
   9.401 +
   9.402 +	c++;
   9.403 +
   9.404 +	message_length = expected_sysex_length(status, c, buffer_length - 1, &vlq_length);
   9.405 +
   9.406 +	if (message_length < 0)
   9.407 +		return (-3);
   9.408 +
   9.409 +	c += vlq_length;
   9.410 +
   9.411 +	if (vlq_length + message_length >= buffer_length) {
   9.412 +		fg_critical("End of buffer in extract_sysex_event().");
   9.413 +		return (-5);
   9.414 +	}
   9.415 +
   9.416 +	event->midi_buffer_length = message_length;
   9.417 +	event->midi_buffer = malloc(event->midi_buffer_length);
   9.418 +	if (event->midi_buffer == NULL) {
   9.419 +		fg_critical("Cannot allocate memory in extract_sysex_event(): %s", strerror(errno));
   9.420 +		return (-4);
   9.421 +	}
   9.422 +
   9.423 +	event->midi_buffer[0] = status;
   9.424 +	memcpy(event->midi_buffer + 1, c, message_length - 1);
   9.425 +
   9.426 +	*len = vlq_length + message_length;
   9.427 +
   9.428 +	return (0);
   9.429 +}
   9.430 +
   9.431 +static int
   9.432 +extract_escaped_event(const unsigned char *buf, const int buffer_length, smf_event_t *event, int *len, int last_status)
   9.433 +{
   9.434 +	int status, message_length, vlq_length;
   9.435 +	const unsigned char *c = buf;
   9.436 +
   9.437 +	status = *buf;
   9.438 +
   9.439 +	assert(is_escape_byte(status));
   9.440 +
   9.441 +	c++;
   9.442 +
   9.443 +	message_length = expected_escaped_length(status, c, buffer_length - 1, &vlq_length);
   9.444 +
   9.445 +	if (message_length < 0)
   9.446 +		return (-3);
   9.447 +
   9.448 +	c += vlq_length;
   9.449 +
   9.450 +	if (vlq_length + message_length >= buffer_length) {
   9.451 +		fg_critical("End of buffer in extract_escaped_event().");
   9.452 +		return (-5);
   9.453 +	}
   9.454 +
   9.455 +	event->midi_buffer_length = message_length;
   9.456 +	event->midi_buffer = malloc(event->midi_buffer_length);
   9.457 +	if (event->midi_buffer == NULL) {
   9.458 +		fg_critical("Cannot allocate memory in extract_escaped_event(): %s", strerror(errno));
   9.459 +		return (-4);
   9.460 +	}
   9.461 +
   9.462 +	memcpy(event->midi_buffer, c, message_length);
   9.463 +
   9.464 +	if (smf_event_is_valid(event)) {
   9.465 +		fg_critical("Escaped event is invalid.");
   9.466 +		return (-1);
   9.467 +	}
   9.468 +
   9.469 +	if (smf_event_is_system_realtime(event) || smf_event_is_system_common(event)) {
   9.470 +		fg_warning("Escaped event is not System Realtime nor System Common.");
   9.471 +	}
   9.472 +
   9.473 +	*len = vlq_length + message_length;
   9.474 +
   9.475 +	return (0);
   9.476 +}
   9.477 +
   9.478 +
   9.479 +/**
   9.480 + * Puts MIDI data extracted from from "buf" into "event" and number of consumed bytes into "len".
   9.481 + * In case valid status is not found, it uses "last_status" (so called "running status").
   9.482 + * Returns 0 iff everything went OK, value < 0 in case of error.
   9.483 + */
   9.484 +static int
   9.485 +extract_midi_event(const unsigned char *buf, const int buffer_length, smf_event_t *event, int *len, int last_status)
   9.486 +{
   9.487 +	int status, message_length;
   9.488 +	const unsigned char *c = buf;
   9.489 +
   9.490 +	assert(buffer_length > 0);
   9.491 +
   9.492 +	/* Is the first byte the status byte? */
   9.493 +	if (is_status_byte(*c)) {
   9.494 +		status = *c;
   9.495 +		c++;
   9.496 +
   9.497 +	} else {
   9.498 +		/* No, we use running status then. */
   9.499 +		status = last_status;
   9.500 +	}
   9.501 +
   9.502 +	if (!is_status_byte(status)) {
   9.503 +		fg_critical("SMF error: bad status byte (MSB is zero).");
   9.504 +		return (-1);
   9.505 +	}
   9.506 +
   9.507 +	if (is_sysex_byte(status))
   9.508 +		return (extract_sysex_event(buf, buffer_length, event, len, last_status));
   9.509 +
   9.510 +	if (is_escape_byte(status))
   9.511 +		return (extract_escaped_event(buf, buffer_length, event, len, last_status));
   9.512 +
   9.513 +	/* At this point, "c" points to first byte following the status byte. */
   9.514 +	message_length = expected_message_length(status, c, buffer_length - (c - buf));
   9.515 +
   9.516 +	if (message_length < 0)
   9.517 +		return (-3);
   9.518 +
   9.519 +	if (message_length - 1 > buffer_length - (c - buf)) {
   9.520 +		fg_critical("End of buffer in extract_midi_event().");
   9.521 +		return (-5);
   9.522 +	}
   9.523 +
   9.524 +	event->midi_buffer_length = message_length;
   9.525 +	event->midi_buffer = malloc(event->midi_buffer_length);
   9.526 +	if (event->midi_buffer == NULL) {
   9.527 +		fg_critical("Cannot allocate memory in extract_midi_event(): %s", strerror(errno));
   9.528 +		return (-4);
   9.529 +	}
   9.530 +
   9.531 +	event->midi_buffer[0] = status;
   9.532 +	memcpy(event->midi_buffer + 1, c, message_length - 1);
   9.533 +
   9.534 +	*len = c + message_length - 1 - buf;
   9.535 +
   9.536 +	return (0);
   9.537 +}
   9.538 +
   9.539 +/**
   9.540 + * Locates, basing on track->next_event_offset, the next event data in track->buffer,
   9.541 + * interprets it, allocates smf_event_t and fills it properly.  Returns smf_event_t
   9.542 + * or NULL, if there was an error.  Allocating event means adding it to the track;
   9.543 + * see smf_event_new().
   9.544 + */
   9.545 +static smf_event_t *
   9.546 +parse_next_event(smf_track_t *track)
   9.547 +{
   9.548 +	int time = 0, len, buffer_length;
   9.549 +	unsigned char *c, *start;
   9.550 +
   9.551 +	smf_event_t *event = smf_event_new();
   9.552 +	if (event == NULL)
   9.553 +		goto error;
   9.554 +
   9.555 +	c = start = (unsigned char *)track->file_buffer + track->next_event_offset;
   9.556 +
   9.557 +	assert(track->file_buffer != NULL);
   9.558 +	assert(track->file_buffer_length > 0);
   9.559 +	assert(track->next_event_offset > 0);
   9.560 +
   9.561 +	buffer_length = track->file_buffer_length - track->next_event_offset;
   9.562 +	assert(buffer_length > 0);
   9.563 +
   9.564 +	/* First, extract time offset from previous event. */
   9.565 +	if (extract_vlq(c, buffer_length, &time, &len))
   9.566 +		goto error;
   9.567 +
   9.568 +	c += len;
   9.569 +	buffer_length -= len;
   9.570 +
   9.571 +	if (buffer_length <= 0)
   9.572 +		goto error;
   9.573 +
   9.574 +	/* Now, extract the actual event. */
   9.575 +	if (extract_midi_event(c, buffer_length, event, &len, track->last_status))
   9.576 +		goto error;
   9.577 +
   9.578 +	c += len;
   9.579 +	buffer_length -= len;
   9.580 +	track->last_status = event->midi_buffer[0];
   9.581 +	track->next_event_offset += c - start;
   9.582 +
   9.583 +	smf_track_add_event_delta_pulses(track, event, time);
   9.584 +
   9.585 +	return (event);
   9.586 +
   9.587 +error:
   9.588 +	if (event != NULL)
   9.589 +		smf_event_delete(event);
   9.590 +
   9.591 +	return (NULL);
   9.592 +}
   9.593 +
   9.594 +/**
   9.595 + * Takes "len" characters starting in "buf", making sure it does not access past the length of the buffer,
   9.596 + * and makes ordinary, zero-terminated string from it.  May return NULL if there was any problem.
   9.597 + */ 
   9.598 +static char *
   9.599 +make_string(const unsigned char *buf, const int buffer_length, int len)
   9.600 +{
   9.601 +	char *str;
   9.602 +
   9.603 +	assert(buffer_length > 0);
   9.604 +	assert(len > 0);
   9.605 +
   9.606 +	if (len > buffer_length) {
   9.607 +		fg_critical("End of buffer in make_string().");
   9.608 +
   9.609 +		len = buffer_length;
   9.610 +	}
   9.611 +
   9.612 +	str = malloc(len + 1);
   9.613 +	if (str == NULL) {
   9.614 +		fg_critical("Cannot allocate memory in make_string().");
   9.615 +		return (NULL);
   9.616 +	}
   9.617 +
   9.618 +	memcpy(str, buf, len);
   9.619 +	str[len] = '\0';
   9.620 +
   9.621 +	return (str);
   9.622 +}
   9.623 +
   9.624 +/**
   9.625 + * \return 1, if passed a metaevent containing text, that is, Text, Copyright,
   9.626 + * Sequence/Track Name, Instrument, Lyric, Marker, Cue Point, Program Name,
   9.627 + * or Device Name; 0 otherwise.
   9.628 + */
   9.629 +int
   9.630 +smf_event_is_textual(const smf_event_t *event)
   9.631 +{
   9.632 +	if (!smf_event_is_metadata(event))
   9.633 +		return (0);
   9.634 +
   9.635 +	if (event->midi_buffer_length < 4)
   9.636 +		return (0);
   9.637 +
   9.638 +	if (event->midi_buffer[3] < 1 && event->midi_buffer[3] > 9)
   9.639 +		return (0);
   9.640 +
   9.641 +	return (1);
   9.642 +}
   9.643 +
   9.644 +/**
   9.645 + * Extracts text from "textual metaevents", such as Text or Lyric.
   9.646 + *
   9.647 + * \return Zero-terminated string extracted from "text events" or NULL, if there was any problem.
   9.648 + */
   9.649 +char *
   9.650 +smf_event_extract_text(const smf_event_t *event)
   9.651 +{
   9.652 +	int string_length = -1, length_length = -1;
   9.653 +
   9.654 +	if (!smf_event_is_textual(event))
   9.655 +		return (NULL);
   9.656 +
   9.657 +	if (event->midi_buffer_length < 3) {
   9.658 +		fg_critical("smf_event_extract_text: truncated MIDI message.");
   9.659 +		return (NULL);
   9.660 +	}
   9.661 +
   9.662 +	extract_vlq((void *)&(event->midi_buffer[2]), event->midi_buffer_length - 2, &string_length, &length_length);
   9.663 +
   9.664 +	if (string_length <= 0) {
   9.665 +		fg_critical("smf_event_extract_text: truncated MIDI message.");
   9.666 +		return (NULL);
   9.667 +	}
   9.668 +
   9.669 +	return (make_string((void *)(&event->midi_buffer[2] + length_length), event->midi_buffer_length - 2 - length_length, string_length));
   9.670 +}
   9.671 +
   9.672 +/**
   9.673 + * Verify if the next chunk really is MTrk chunk, and if so, initialize some track variables and return 0.
   9.674 + * Return different value otherwise.
   9.675 + */
   9.676 +static int
   9.677 +parse_mtrk_header(smf_track_t *track)
   9.678 +{
   9.679 +	struct chunk_header_struct *mtrk;
   9.680 +
   9.681 +	/* Make sure compiler didn't do anything stupid. */
   9.682 +	assert(sizeof(struct chunk_header_struct) == 8);
   9.683 +	assert(track->smf != NULL);
   9.684 +
   9.685 +	mtrk = next_chunk(track->smf);
   9.686 +
   9.687 +	if (mtrk == NULL)
   9.688 +		return (-1);
   9.689 +
   9.690 +	if (!chunk_signature_matches(mtrk, "MTrk")) {
   9.691 +		fg_warning("SMF warning: Expected MTrk signature, got %c%c%c%c instead; ignoring this chunk.",
   9.692 +				mtrk->id[0], mtrk->id[1], mtrk->id[2], mtrk->id[3]);
   9.693 +	
   9.694 +		return (-2);
   9.695 +	}
   9.696 +
   9.697 +	track->file_buffer = mtrk;
   9.698 +	track->file_buffer_length = sizeof(struct chunk_header_struct) + ntohl(mtrk->length);
   9.699 +	track->next_event_offset = sizeof(struct chunk_header_struct);
   9.700 +
   9.701 +	return (0);
   9.702 +}
   9.703 +
   9.704 +/**
   9.705 + * Return 1 if event is end-of-the-track, 0 otherwise.
   9.706 + */
   9.707 +static int
   9.708 +event_is_end_of_track(const smf_event_t *event)
   9.709 +{
   9.710 +	if (event->midi_buffer[0] == 0xFF && event->midi_buffer[1] == 0x2F)
   9.711 +		return (1);
   9.712 +
   9.713 +	return (0);
   9.714 +}
   9.715 +
   9.716 +/**
   9.717 + * \return Nonzero, if event is as long as it should be, from the MIDI specification point of view.
   9.718 + * Does not work for SysExes - it doesn't recognize internal structure of SysEx.
   9.719 + */
   9.720 +int
   9.721 +smf_event_length_is_valid(const smf_event_t *event)
   9.722 +{
   9.723 +	assert(event);
   9.724 +	assert(event->midi_buffer);
   9.725 +
   9.726 +	if (event->midi_buffer_length < 1)
   9.727 +		return (0);
   9.728 +
   9.729 +	/* We cannot use expected_message_length on sysexes. */
   9.730 +	if (smf_event_is_sysex(event))
   9.731 +		return (1);
   9.732 +
   9.733 +	if (event->midi_buffer_length != expected_message_length(event->midi_buffer[0],
   9.734 +		&(event->midi_buffer[1]), event->midi_buffer_length - 1)) {
   9.735 +
   9.736 +		return (0);
   9.737 +	}
   9.738 +
   9.739 +	return (1);
   9.740 +}
   9.741 +
   9.742 +/**
   9.743 + * \return Nonzero, if MIDI data in the event is valid, 0 otherwise.  For example,
   9.744 + * it checks if event length is correct.
   9.745 + */
   9.746 +/* XXX: this routine requires some more work to detect more errors. */
   9.747 +int
   9.748 +smf_event_is_valid(const smf_event_t *event)
   9.749 +{
   9.750 +	assert(event);
   9.751 +	assert(event->midi_buffer);
   9.752 +	assert(event->midi_buffer_length >= 1);
   9.753 +
   9.754 +	if (!is_status_byte(event->midi_buffer[0])) {
   9.755 +		fg_critical("First byte of MIDI message is not a valid status byte.");
   9.756 +
   9.757 +		return (0);
   9.758 +	}
   9.759 +
   9.760 +	if (!smf_event_length_is_valid(event))
   9.761 +		return (0);
   9.762 +
   9.763 +	return (1);
   9.764 +}
   9.765 +
   9.766 +/**
   9.767 + * Parse events and put it on the track.
   9.768 + */
   9.769 +static int
   9.770 +parse_mtrk_chunk(smf_track_t *track)
   9.771 +{
   9.772 +	smf_event_t *event;
   9.773 +
   9.774 +	if (parse_mtrk_header(track))
   9.775 +		return (-1);
   9.776 +
   9.777 +	for (;;) {
   9.778 +		event = parse_next_event(track);
   9.779 +
   9.780 +		/* Couldn't parse an event? */
   9.781 +		if (event == NULL) {
   9.782 +			fg_critical("Unable to parse MIDI event; truncating track.");
   9.783 +			if (smf_track_add_eot_delta_pulses(track, 0) != 0) {
   9.784 +				fg_critical("smf_track_add_eot_delta_pulses failed.");
   9.785 +				return (-2);
   9.786 +			}
   9.787 +			break;
   9.788 +		}
   9.789 +
   9.790 +		assert(smf_event_is_valid(event));
   9.791 +
   9.792 +		if (event_is_end_of_track(event))
   9.793 +			break;
   9.794 +	}
   9.795 +
   9.796 +	track->file_buffer = NULL;
   9.797 +	track->file_buffer_length = 0;
   9.798 +	track->next_event_offset = -1;
   9.799 +
   9.800 +	return (0);
   9.801 +}
   9.802 +
   9.803 +/**
   9.804 + * Allocate buffer of proper size and read file contents into it.  Close file afterwards.
   9.805 + */
   9.806 +static int
   9.807 +load_file_into_buffer(void **file_buffer, int *file_buffer_length, const char *file_name)
   9.808 +{
   9.809 +	FILE *stream = fopen(file_name, "rb");
   9.810 +
   9.811 +	if (stream == NULL) {
   9.812 +		fg_critical("Cannot open input file: %s", strerror(errno));
   9.813 +
   9.814 +		return (-1);
   9.815 +	}
   9.816 +
   9.817 +	if (fseek(stream, 0, SEEK_END)) {
   9.818 +		fg_critical("fseek(3) failed: %s", strerror(errno));
   9.819 +
   9.820 +		return (-2);
   9.821 +	}
   9.822 +
   9.823 +	*file_buffer_length = ftell(stream);
   9.824 +	if (*file_buffer_length == -1) {
   9.825 +		fg_critical("ftell(3) failed: %s", strerror(errno));
   9.826 +
   9.827 +		return (-3);
   9.828 +	}
   9.829 +
   9.830 +	if (fseek(stream, 0, SEEK_SET)) {
   9.831 +		fg_critical("fseek(3) failed: %s", strerror(errno));
   9.832 +
   9.833 +		return (-4);
   9.834 +	}
   9.835 +
   9.836 +	*file_buffer = malloc(*file_buffer_length);
   9.837 +	if (*file_buffer == NULL) {
   9.838 +		fg_critical("malloc(3) failed: %s", strerror(errno));
   9.839 +
   9.840 +		return (-5);
   9.841 +	}
   9.842 +
   9.843 +	if (fread(*file_buffer, 1, *file_buffer_length, stream) != *file_buffer_length) {
   9.844 +		fg_critical("fread(3) failed: %s", strerror(errno));
   9.845 +
   9.846 +		return (-6);
   9.847 +	}
   9.848 +
   9.849 +	if (fclose(stream)) {
   9.850 +		fg_critical("fclose(3) failed: %s", strerror(errno));
   9.851 +
   9.852 +		return (-7);
   9.853 +	}
   9.854 +
   9.855 +	return (0);
   9.856 +}
   9.857 +
   9.858 +/**
   9.859 +  * Creates new SMF and fills it with data loaded from the given buffer.
   9.860 + * \return SMF or NULL, if loading failed.
   9.861 +  */
   9.862 +smf_t *
   9.863 +smf_load_from_memory(const void *buffer, const int buffer_length)
   9.864 +{
   9.865 +	int i;
   9.866 +
   9.867 +	smf_t *smf = smf_new();
   9.868 +
   9.869 +	smf->file_buffer = (void *)buffer;
   9.870 +	smf->file_buffer_length = buffer_length;
   9.871 +	smf->next_chunk_offset = 0;
   9.872 +
   9.873 +	if (parse_mthd_chunk(smf))
   9.874 +		return (NULL);
   9.875 +
   9.876 +	for (i = 1; i <= smf->expected_number_of_tracks; i++) {
   9.877 +		smf_track_t *track = smf_track_new();
   9.878 +		if (track == NULL)
   9.879 +			return (NULL);
   9.880 +
   9.881 +		smf_add_track(smf, track);
   9.882 +
   9.883 +		/* Skip unparseable chunks. */
   9.884 +		if (parse_mtrk_chunk(track)) {
   9.885 +			fg_warning("SMF warning: Cannot load track.");
   9.886 +			smf_track_delete(track);
   9.887 +		}
   9.888 +
   9.889 +		track->file_buffer = NULL;
   9.890 +		track->file_buffer_length = 0;
   9.891 +		track->next_event_offset = -1;
   9.892 +	}
   9.893 +
   9.894 +	if (smf->expected_number_of_tracks != smf->number_of_tracks) {
   9.895 +		fg_warning("SMF warning: MThd header declared %d tracks, but only %d found; continuing anyway.",
   9.896 +				smf->expected_number_of_tracks, smf->number_of_tracks);
   9.897 +
   9.898 +		smf->expected_number_of_tracks = smf->number_of_tracks;
   9.899 +	}
   9.900 +
   9.901 +	smf->file_buffer = NULL;
   9.902 +	smf->file_buffer_length = 0;
   9.903 +	smf->next_chunk_offset = -1;
   9.904 +
   9.905 +	return (smf);
   9.906 +}
   9.907 +
   9.908 +/**
   9.909 + * Loads SMF file.
   9.910 + *
   9.911 + * \param file_name Path to the file.
   9.912 + * \return SMF or NULL, if loading failed.
   9.913 + */
   9.914 +smf_t *
   9.915 +smf_load(const char *file_name)
   9.916 +{
   9.917 +	int file_buffer_length;
   9.918 +	void *file_buffer;
   9.919 +	smf_t *smf;
   9.920 +
   9.921 +	if (load_file_into_buffer(&file_buffer, &file_buffer_length, file_name))
   9.922 +		return (NULL);
   9.923 +
   9.924 +	smf = smf_load_from_memory(file_buffer, file_buffer_length);
   9.925 +
   9.926 +	memset(file_buffer, 0, file_buffer_length);
   9.927 +	free(file_buffer);
   9.928 +
   9.929 +	if (smf == NULL)
   9.930 +		return (NULL);
   9.931 +
   9.932 +	smf_rewind(smf);
   9.933 +
   9.934 +	return (smf);
   9.935 +}
   9.936 +
    10.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    10.2 +++ b/src/smf_private.h	Thu Jan 26 11:25:11 2012 +0200
    10.3 @@ -0,0 +1,83 @@
    10.4 +/*-
    10.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
    10.6 + * All rights reserved.
    10.7 + *
    10.8 + * Redistribution and use in source and binary forms, with or without
    10.9 + * modification, are permitted provided that the following conditions
   10.10 + * are met:
   10.11 + * 1. Redistributions of source code must retain the above copyright
   10.12 + *    notice, this list of conditions and the following disclaimer.
   10.13 + * 2. Redistributions in binary form must reproduce the above copyright
   10.14 + *    notice, this list of conditions and the following disclaimer in the
   10.15 + *    documentation and/or other materials provided with the distribution.
   10.16 + *
   10.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
   10.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
   10.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
   10.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
   10.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
   10.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
   10.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
   10.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
   10.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
   10.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
   10.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   10.28 + *
   10.29 + */
   10.30 +
   10.31 +#ifndef SMF_PRIVATE_H
   10.32 +#define SMF_PRIVATE_H
   10.33 +
   10.34 +#include <stdint.h>
   10.35 +#include <sys/types.h>
   10.36 +
   10.37 +#include "fake_glib.h"
   10.38 +
   10.39 +#include "config.h"
   10.40 +
   10.41 +#define SMF_VERSION PACKAGE_VERSION
   10.42 +
   10.43 +/**
   10.44 + * \file
   10.45 + *
   10.46 + * Private header.  Applications using libsmf should use smf.h.
   10.47 + *
   10.48 + */
   10.49 +
   10.50 +#if defined(__GNUC__)
   10.51 +#define ATTRIBUTE_PACKED  __attribute__((__packed__))
   10.52 +#else
   10.53 +#define ATTRIBUTE_PACKED
   10.54 +#pragma pack(1)
   10.55 +#endif
   10.56 +
   10.57 +/** SMF chunk header, used only by smf_load.c and smf_save.c. */
   10.58 +struct chunk_header_struct {
   10.59 +	char		id[4];
   10.60 +	uint32_t	length; 
   10.61 +} ATTRIBUTE_PACKED;
   10.62 +
   10.63 +/** SMF chunk, used only by smf_load.c and smf_save.c. */
   10.64 +struct mthd_chunk_struct {
   10.65 +	struct chunk_header_struct	mthd_header;
   10.66 +	uint16_t			format;
   10.67 +	uint16_t			number_of_tracks;
   10.68 +	uint16_t			division;
   10.69 +} ATTRIBUTE_PACKED;
   10.70 +
   10.71 +#if (!defined __GNUC__)
   10.72 +#pragma pack()
   10.73 +#endif
   10.74 +
   10.75 +void smf_track_add_event(smf_track_t *track, smf_event_t *event);
   10.76 +void smf_init_tempo(smf_t *smf);
   10.77 +void smf_fini_tempo(smf_t *smf);
   10.78 +void smf_create_tempo_map_and_compute_seconds(smf_t *smf);
   10.79 +void maybe_add_to_tempo_map(smf_event_t *event);
   10.80 +void remove_last_tempo_with_pulses(smf_t *smf, int pulses);
   10.81 +int smf_event_is_tempo_change_or_time_signature(const smf_event_t *event) WARN_UNUSED_RESULT;
   10.82 +int smf_event_length_is_valid(const smf_event_t *event) WARN_UNUSED_RESULT;
   10.83 +int is_status_byte(const unsigned char status) WARN_UNUSED_RESULT;
   10.84 +
   10.85 +#endif /* SMF_PRIVATE_H */
   10.86 +
    11.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    11.2 +++ b/src/smf_save.c	Thu Jan 26 11:25:11 2012 +0200
    11.3 @@ -0,0 +1,660 @@
    11.4 +/*-
    11.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
    11.6 + * All rights reserved.
    11.7 + *
    11.8 + * Redistribution and use in source and binary forms, with or without
    11.9 + * modification, are permitted provided that the following conditions
   11.10 + * are met:
   11.11 + * 1. Redistributions of source code must retain the above copyright
   11.12 + *    notice, this list of conditions and the following disclaimer.
   11.13 + * 2. Redistributions in binary form must reproduce the above copyright
   11.14 + *    notice, this list of conditions and the following disclaimer in the
   11.15 + *    documentation and/or other materials provided with the distribution.
   11.16 + *
   11.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
   11.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
   11.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
   11.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
   11.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
   11.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
   11.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
   11.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
   11.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
   11.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
   11.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   11.28 + *
   11.29 + */
   11.30 +
   11.31 +/**
   11.32 + * \file
   11.33 + *
   11.34 + * Standard MIDI File writer.
   11.35 + *
   11.36 + */
   11.37 +
   11.38 +/* Reference: http://www.borg.com/~jglatt/tech/midifile.htm */
   11.39 +
   11.40 +#include <stdlib.h>
   11.41 +#include <string.h>
   11.42 +#include <assert.h>
   11.43 +#include <math.h>
   11.44 +#include <errno.h>
   11.45 +#ifdef __MINGW32__
   11.46 +#include <windows.h>
   11.47 +#else /* ! __MINGW32__ */
   11.48 +#include <arpa/inet.h>
   11.49 +#endif /* ! __MINGW32__ */
   11.50 +#include "smf.h"
   11.51 +#include "smf_private.h"
   11.52 +
   11.53 +#define MAX_VLQ_LENGTH 128
   11.54 +
   11.55 +/**
   11.56 + * Extends (reallocates) smf->file_buffer and returns pointer to the newly added space,
   11.57 + * that is, pointer to the first byte after the previous buffer end.  Returns NULL in case
   11.58 + * of error.
   11.59 + */
   11.60 +static void *
   11.61 +smf_extend(smf_t *smf, const int length)
   11.62 +{
   11.63 +	int i, previous_file_buffer_length = smf->file_buffer_length;
   11.64 +	char *previous_file_buffer = smf->file_buffer;
   11.65 +
   11.66 +	/* XXX: Not terribly efficient. */
   11.67 +	smf->file_buffer_length += length;
   11.68 +	smf->file_buffer = realloc(smf->file_buffer, smf->file_buffer_length);
   11.69 +	if (smf->file_buffer == NULL) {
   11.70 +		fg_critical("realloc(3) failed: %s", strerror(errno));
   11.71 +		smf->file_buffer_length = 0;
   11.72 +		return (NULL);
   11.73 +	}
   11.74 +
   11.75 +	/* Fix up pointers.  XXX: omgwtf. */
   11.76 +	for (i = 1; i <= smf->number_of_tracks; i++) {
   11.77 +		smf_track_t *track;
   11.78 +		track = smf_get_track_by_number(smf, i);
   11.79 +		if (track->file_buffer != NULL)
   11.80 +			track->file_buffer = (char *)track->file_buffer + ((char *)smf->file_buffer - previous_file_buffer);
   11.81 +	}
   11.82 +
   11.83 +	return ((char *)smf->file_buffer + previous_file_buffer_length);
   11.84 +}
   11.85 +
   11.86 +/**
   11.87 + * Appends "buffer_length" bytes pointed to by "buffer" to the smf, reallocating storage as needed.  Returns 0
   11.88 + * if everything went ok, different value if there was any problem.
   11.89 + */
   11.90 +static int
   11.91 +smf_append(smf_t *smf, const void *buffer, const int buffer_length)
   11.92 +{
   11.93 +	void *dest;
   11.94 +
   11.95 +	dest = smf_extend(smf, buffer_length);
   11.96 +	if (dest == NULL) {
   11.97 +		fg_critical("Cannot extend track buffer.");
   11.98 +		return (-1);
   11.99 +	}
  11.100 +
  11.101 +	memcpy(dest, buffer, buffer_length);
  11.102 +
  11.103 +	return (0);
  11.104 +}
  11.105 +
  11.106 +/**
  11.107 + * Appends MThd header to the track.  Returns 0 if everything went ok, different value if not.
  11.108 + */
  11.109 +static int
  11.110 +write_mthd_header(smf_t *smf)
  11.111 +{
  11.112 +	struct mthd_chunk_struct mthd_chunk;
  11.113 +
  11.114 +	memcpy(mthd_chunk.mthd_header.id, "MThd", 4);
  11.115 +	mthd_chunk.mthd_header.length = htonl(6);
  11.116 +	mthd_chunk.format = htons(smf->format);
  11.117 +	mthd_chunk.number_of_tracks = htons(smf->number_of_tracks);
  11.118 +	mthd_chunk.division = htons(smf->ppqn);
  11.119 +
  11.120 +	return (smf_append(smf, &mthd_chunk, sizeof(mthd_chunk)));
  11.121 +}
  11.122 +
  11.123 +/**
  11.124 + * Extends (reallocates) track->file_buffer and returns pointer to the newly added space,
  11.125 + * that is, pointer to the first byte after the previous buffer end.  Returns NULL in case
  11.126 + * of error.
  11.127 + */
  11.128 +static void *
  11.129 +track_extend(smf_track_t *track, const int length)
  11.130 +{
  11.131 +	void *buf;
  11.132 +
  11.133 +	assert(track->smf);
  11.134 +
  11.135 +	buf = smf_extend(track->smf, length);
  11.136 +	if (buf == NULL)
  11.137 +		return (NULL);
  11.138 +
  11.139 +	track->file_buffer_length += length;
  11.140 +	if (track->file_buffer == NULL)
  11.141 +		track->file_buffer = buf;
  11.142 +
  11.143 +	return (buf);
  11.144 +}
  11.145 +
  11.146 +/**
  11.147 + * Appends "buffer_length" bytes pointed to by "buffer" to the track, reallocating storage as needed.  Returns 0
  11.148 + * if everything went ok, different value if there was any problem.
  11.149 + */
  11.150 +static int
  11.151 +track_append(smf_track_t *track, const void *buffer, const int buffer_length)
  11.152 +{
  11.153 +	void *dest;
  11.154 +
  11.155 +	dest = track_extend(track, buffer_length);
  11.156 +	if (dest == NULL) {
  11.157 +		fg_critical("Cannot extend track buffer.");
  11.158 +		return (-1);
  11.159 +	}
  11.160 +
  11.161 +	memcpy(dest, buffer, buffer_length);
  11.162 +
  11.163 +	return (0);
  11.164 +}
  11.165 +
  11.166 +static int
  11.167 +format_vlq(unsigned char *buf, int length, unsigned long value)
  11.168 +{
  11.169 +	int i;
  11.170 +	unsigned long buffer;
  11.171 +
  11.172 +	/* Taken from http://www.borg.com/~jglatt/tech/midifile/vari.htm */
  11.173 +	buffer = value & 0x7F;
  11.174 +
  11.175 +	while ((value >>= 7)) {
  11.176 +		buffer <<= 8;
  11.177 +		buffer |= ((value & 0x7F) | 0x80);
  11.178 +	}
  11.179 +
  11.180 +	for (i = 0;; i++) {
  11.181 +		buf[i] = buffer;
  11.182 +
  11.183 +		if (buffer & 0x80)
  11.184 +			buffer >>= 8;
  11.185 +		else
  11.186 +			break;
  11.187 +	}
  11.188 +
  11.189 +	assert(i <= length);
  11.190 +
  11.191 +	/* + 1, because "i" is an offset, not a count. */
  11.192 +	return (i + 1);
  11.193 +}
  11.194 +
  11.195 +smf_event_t *
  11.196 +smf_event_new_textual(int type, const char *text)
  11.197 +{
  11.198 +	int vlq_length, text_length, copied_length;
  11.199 +	smf_event_t *event;
  11.200 +
  11.201 +	assert(type >= 1 && type <= 9);
  11.202 +
  11.203 +	text_length = strlen(text);
  11.204 +
  11.205 +	event = smf_event_new();
  11.206 +	if (event == NULL)
  11.207 +		return (NULL);
  11.208 +
  11.209 +	/* "2 +" is for leading 0xFF 0xtype. */
  11.210 +	event->midi_buffer_length = 2 + text_length + MAX_VLQ_LENGTH;
  11.211 +	event->midi_buffer = malloc(event->midi_buffer_length);
  11.212 +	if (event->midi_buffer == NULL) {
  11.213 +		fg_critical("Cannot allocate MIDI buffer structure: %s", strerror(errno));
  11.214 +		smf_event_delete(event);
  11.215 +
  11.216 +		return (NULL); 
  11.217 +	}
  11.218 +
  11.219 +	event->midi_buffer[0] = 0xFF;
  11.220 +	event->midi_buffer[1] = type;
  11.221 +
  11.222 +	vlq_length = format_vlq(event->midi_buffer + 2, MAX_VLQ_LENGTH - 2, text_length);
  11.223 +	copied_length = snprintf((char *)event->midi_buffer + vlq_length + 2, event->midi_buffer_length - vlq_length - 2, "%s", text);
  11.224 +
  11.225 +	assert(copied_length == text_length);
  11.226 +
  11.227 +	event->midi_buffer_length = 2 + vlq_length + text_length;
  11.228 +
  11.229 +	return event;
  11.230 +}
  11.231 +
  11.232 +/**
  11.233 +  * Appends value, expressed as Variable Length Quantity, to event->track.
  11.234 +  */
  11.235 +static int
  11.236 +write_vlq(smf_event_t *event, unsigned long value)
  11.237 +{
  11.238 +	unsigned char buf[MAX_VLQ_LENGTH];
  11.239 +	int vlq_length;
  11.240 +
  11.241 +	vlq_length = format_vlq(buf, MAX_VLQ_LENGTH, value);
  11.242 +
  11.243 +	return (track_append(event->track, buf, vlq_length));
  11.244 +}
  11.245 +
  11.246 +/**
  11.247 + * Appends event time as Variable Length Quantity.  Returns 0 if everything went ok,
  11.248 + * different value in case of error.
  11.249 + */
  11.250 +static int
  11.251 +write_event_time(smf_event_t *event)
  11.252 +{
  11.253 +	assert(event->delta_time_pulses >= 0);
  11.254 +
  11.255 +	return (write_vlq(event, event->delta_time_pulses));
  11.256 +}
  11.257 +
  11.258 +static int
  11.259 +write_sysex_contents(smf_event_t *event)
  11.260 +{
  11.261 +	int ret;
  11.262 +	unsigned char sysex_status = 0xF0;
  11.263 +
  11.264 +	assert(smf_event_is_sysex(event));
  11.265 +
  11.266 +	ret = track_append(event->track, &sysex_status, 1);
  11.267 +	if (ret)
  11.268 +		return (ret);
  11.269 +
  11.270 +	/* -1, because length does not include status byte. */
  11.271 +	ret = write_vlq(event, event->midi_buffer_length - 1);
  11.272 +	if (ret)
  11.273 +		return (ret);
  11.274 +
  11.275 +	ret = track_append(event->track, event->midi_buffer + 1, event->midi_buffer_length - 1);
  11.276 +	if (ret)
  11.277 +		return (ret);
  11.278 +
  11.279 +	return (0);
  11.280 +}
  11.281 +
  11.282 +/**
  11.283 +  * Appends contents of event->midi_buffer wrapped into 0xF7 MIDI event.
  11.284 +  */
  11.285 +static int
  11.286 +write_escaped_event_contents(smf_event_t *event)
  11.287 +{
  11.288 +	int ret;
  11.289 +	unsigned char escape_status = 0xF7;
  11.290 +
  11.291 +	if (smf_event_is_sysex(event))
  11.292 +		return (write_sysex_contents(event));
  11.293 +
  11.294 +	ret = track_append(event->track, &escape_status, 1);
  11.295 +	if (ret)
  11.296 +		return (ret);
  11.297 +
  11.298 +	ret = write_vlq(event, event->midi_buffer_length);
  11.299 +	if (ret)
  11.300 +		return (ret);
  11.301 +
  11.302 +	ret = track_append(event->track, event->midi_buffer, event->midi_buffer_length);
  11.303 +	if (ret)
  11.304 +		return (ret);
  11.305 +
  11.306 +	return (0);
  11.307 +}
  11.308 +
  11.309 +/**
  11.310 + * Appends contents of event->midi_buffer.  Returns 0 if everything went 0,
  11.311 + * different value in case of error.
  11.312 + */
  11.313 +static int
  11.314 +write_event_contents(smf_event_t *event)
  11.315 +{
  11.316 +	if (smf_event_is_system_realtime(event) || smf_event_is_system_common(event))
  11.317 +		return (write_escaped_event_contents(event));
  11.318 +
  11.319 +	return (track_append(event->track, event->midi_buffer, event->midi_buffer_length));
  11.320 +}
  11.321 +
  11.322 +/**
  11.323 + * Writes out an event.
  11.324 + */
  11.325 +static int
  11.326 +write_event(smf_event_t *event)
  11.327 +{
  11.328 +	int ret;
  11.329 +
  11.330 +	ret = write_event_time(event);
  11.331 +	if (ret)
  11.332 +		return (ret);
  11.333 +
  11.334 +	ret = write_event_contents(event);
  11.335 +	if (ret)
  11.336 +		return (ret);
  11.337 +
  11.338 +	return (0);
  11.339 +}
  11.340 +
  11.341 +/**
  11.342 + * Writes out MTrk header, except of MTrk chunk length, which is written by write_mtrk_length().
  11.343 + */
  11.344 +static int
  11.345 +write_mtrk_header(smf_track_t *track)
  11.346 +{
  11.347 +	struct chunk_header_struct mtrk_header;
  11.348 +
  11.349 +	memcpy(mtrk_header.id, "MTrk", 4);
  11.350 +
  11.351 +	return (track_append(track, &mtrk_header, sizeof(mtrk_header)));
  11.352 +}
  11.353 +
  11.354 +/**
  11.355 + * Updates MTrk chunk length of a given track.
  11.356 + */
  11.357 +static int
  11.358 +write_mtrk_length(smf_track_t *track)
  11.359 +{
  11.360 +	struct chunk_header_struct *mtrk_header;
  11.361 +
  11.362 +	assert(track->file_buffer != NULL);
  11.363 +	assert(track->file_buffer_length >= 6);
  11.364 +
  11.365 +	mtrk_header = (struct chunk_header_struct *)track->file_buffer;
  11.366 +	mtrk_header->length = htonl(track->file_buffer_length - sizeof(struct chunk_header_struct));
  11.367 +
  11.368 +	return (0);
  11.369 +}
  11.370 +
  11.371 +/**
  11.372 + * Writes out the track.
  11.373 + */
  11.374 +static int
  11.375 +write_track(smf_track_t *track)
  11.376 +{
  11.377 +	int ret;
  11.378 +	smf_event_t *event;
  11.379 +
  11.380 +	ret = write_mtrk_header(track);
  11.381 +	if (ret)
  11.382 +		return (ret);
  11.383 +
  11.384 +	while ((event = smf_track_get_next_event(track)) != NULL) {
  11.385 +		ret = write_event(event);
  11.386 +		if (ret)
  11.387 +			return (ret);
  11.388 +	}
  11.389 +
  11.390 +	ret = write_mtrk_length(track);
  11.391 +	if (ret)
  11.392 +		return (ret);
  11.393 +
  11.394 +	return (0);
  11.395 +}
  11.396 +
  11.397 +/**
  11.398 + * Takes smf->file_buffer and saves it to the file.
  11.399 + */
  11.400 +static int
  11.401 +write_file(smf_t *smf, const char *file_name)
  11.402 +{
  11.403 +	FILE *stream;
  11.404 +
  11.405 +	stream = fopen(file_name, "wb+");
  11.406 +	if (stream == NULL) {
  11.407 +		fg_critical("Cannot open input file: %s", strerror(errno));
  11.408 +
  11.409 +		return (-1);
  11.410 +	}
  11.411 +
  11.412 +	if (fwrite(smf->file_buffer, 1, smf->file_buffer_length, stream) != smf->file_buffer_length) {
  11.413 +		fg_critical("fwrite(3) failed: %s", strerror(errno));
  11.414 +
  11.415 +		return (-2);
  11.416 +	}
  11.417 +
  11.418 +	if (fclose(stream)) {
  11.419 +		fg_critical("fclose(3) failed: %s", strerror(errno));
  11.420 +
  11.421 +		return (-3);
  11.422 +	}
  11.423 +
  11.424 +	return (0);
  11.425 +}
  11.426 +
  11.427 +static void
  11.428 +free_buffer(smf_t *smf)
  11.429 +{
  11.430 +	int i;
  11.431 +	smf_track_t *track;
  11.432 +
  11.433 +	/* Clear the pointers. */
  11.434 +	memset(smf->file_buffer, 0, smf->file_buffer_length);
  11.435 +	free(smf->file_buffer);
  11.436 +	smf->file_buffer = NULL;
  11.437 +	smf->file_buffer_length = 0;
  11.438 +
  11.439 +	for (i = 1; i <= smf->number_of_tracks; i++) {
  11.440 +		track = smf_get_track_by_number(smf, i);
  11.441 +		assert(track);
  11.442 +		track->file_buffer = NULL;
  11.443 +		track->file_buffer_length = 0;
  11.444 +	}
  11.445 +}
  11.446 +
  11.447 +#ifndef NDEBUG
  11.448 +
  11.449 +/**
  11.450 + * \return Nonzero, if all pointers supposed to be NULL are NULL.  Triggers assertion if not.
  11.451 + */
  11.452 +static int
  11.453 +pointers_are_clear(smf_t *smf)
  11.454 +{
  11.455 +	int i;
  11.456 +
  11.457 +	smf_track_t *track;
  11.458 +	assert(smf->file_buffer == NULL);
  11.459 +	assert(smf->file_buffer_length == 0);
  11.460 +
  11.461 +	for (i = 1; i <= smf->number_of_tracks; i++) {
  11.462 +		track = smf_get_track_by_number(smf, i);
  11.463 +
  11.464 +		assert(track != NULL);
  11.465 +		assert(track->file_buffer == NULL);
  11.466 +		assert(track->file_buffer_length == 0);
  11.467 +	}
  11.468 +
  11.469 +	return (1);
  11.470 +}
  11.471 +
  11.472 +#endif /* !NDEBUG */
  11.473 +
  11.474 +/**
  11.475 + * \return Nonzero, if event is End Of Track metaevent.
  11.476 + */
  11.477 +int
  11.478 +smf_event_is_eot(const smf_event_t *event)
  11.479 +{
  11.480 +	if (event->midi_buffer_length != 3)
  11.481 +		return (0);
  11.482 +
  11.483 +	if (event->midi_buffer[0] != 0xFF || event->midi_buffer[1] != 0x2F || event->midi_buffer[2] != 0x00)
  11.484 +		return (0);
  11.485 +
  11.486 +	return (1);
  11.487 +}
  11.488 +
  11.489 +/**
  11.490 + * Check if SMF is valid and add missing EOT events.
  11.491 + *
  11.492 + * \return 0, if SMF is valid.
  11.493 + */
  11.494 +static int
  11.495 +smf_validate(smf_t *smf)
  11.496 +{
  11.497 +	int trackno, eventno, eot_found;
  11.498 +	smf_track_t *track;
  11.499 +	smf_event_t *event;
  11.500 +
  11.501 +	if (smf->format < 0 || smf->format > 2) {
  11.502 +		fg_critical("SMF error: smf->format is less than zero of greater than two.");
  11.503 +		return (-1);
  11.504 +	}
  11.505 +
  11.506 +	if (smf->number_of_tracks < 1) {
  11.507 +		fg_critical("SMF error: number of tracks is less than one.");
  11.508 +		return (-2);
  11.509 +	}
  11.510 +
  11.511 +	if (smf->format == 0 && smf->number_of_tracks > 1) {
  11.512 +		fg_critical("SMF error: format is 0, but number of tracks is more than one.");
  11.513 +		return (-3);
  11.514 +	}
  11.515 +
  11.516 +	if (smf->ppqn <= 0) {
  11.517 +		fg_critical("SMF error: PPQN has to be > 0.");
  11.518 +		return (-4);
  11.519 +	}
  11.520 +
  11.521 +	for (trackno = 1; trackno <= smf->number_of_tracks; trackno++) {
  11.522 +		track = smf_get_track_by_number(smf, trackno);
  11.523 +		assert(track);
  11.524 +
  11.525 +		eot_found = 0;
  11.526 +
  11.527 +		for (eventno = 1; eventno <= track->number_of_events; eventno++) {
  11.528 +			event = smf_track_get_event_by_number(track, eventno);
  11.529 +			assert(event);
  11.530 +
  11.531 +			if (!smf_event_is_valid(event)) {
  11.532 +				fg_critical("Event #%d on track #%d is invalid.", eventno, trackno);
  11.533 +				return (-5);
  11.534 +			}
  11.535 +
  11.536 +			if (smf_event_is_eot(event)) {
  11.537 +				if (eot_found) {
  11.538 +					fg_critical("Duplicate End Of Track event on track #%d.", trackno);
  11.539 +					return (-6);
  11.540 +				}
  11.541 +
  11.542 +				eot_found = 1;
  11.543 +			}
  11.544 +		}
  11.545 +
  11.546 +		if (!eot_found) {
  11.547 +			if (smf_track_add_eot_delta_pulses(track, 0)) {
  11.548 +				fg_critical("smf_track_add_eot_delta_pulses failed.");
  11.549 +				return (-6);
  11.550 +			}
  11.551 +		}
  11.552 +
  11.553 +	}
  11.554 +
  11.555 +	return (0);
  11.556 +}
  11.557 +
  11.558 +#ifndef NDEBUG
  11.559 +
  11.560 +static void
  11.561 +assert_smf_event_is_identical(const smf_event_t *a, const smf_event_t *b)
  11.562 +{
  11.563 +	assert(a->event_number == b->event_number);
  11.564 +	assert(a->delta_time_pulses == b->delta_time_pulses);
  11.565 +	assert(abs(a->time_pulses - b->time_pulses) <= 2);
  11.566 +	assert(fabs(a->time_seconds - b->time_seconds) <= 0.01);
  11.567 +	assert(a->track_number == b->track_number);
  11.568 +	assert(a->midi_buffer_length == b->midi_buffer_length);
  11.569 +	assert(memcmp(a->midi_buffer, b->midi_buffer, a->midi_buffer_length) == 0);
  11.570 +}
  11.571 +
  11.572 +static void
  11.573 +assert_smf_track_is_identical(const smf_track_t *a, const smf_track_t *b)
  11.574 +{
  11.575 +	int i;
  11.576 +
  11.577 +	assert(a->track_number == b->track_number);
  11.578 +	assert(a->number_of_events == b->number_of_events);
  11.579 +
  11.580 +	for (i = 1; i <= a->number_of_events; i++)
  11.581 +		assert_smf_event_is_identical(smf_track_get_event_by_number(a, i), smf_track_get_event_by_number(b, i));
  11.582 +}
  11.583 +
  11.584 +static void
  11.585 +assert_smf_is_identical(const smf_t *a, const smf_t *b)
  11.586 +{
  11.587 +	int i;
  11.588 +
  11.589 +	assert(a->format == b->format);
  11.590 +	assert(a->ppqn == b->ppqn);
  11.591 +	assert(a->frames_per_second == b->frames_per_second);
  11.592 +	assert(a->resolution == b->resolution);
  11.593 +	assert(a->number_of_tracks == b->number_of_tracks);
  11.594 +
  11.595 +	for (i = 1; i <= a->number_of_tracks; i++)
  11.596 +		assert_smf_track_is_identical(smf_get_track_by_number(a, i), smf_get_track_by_number(b, i));
  11.597 +
  11.598 +	/* We do not need to compare tempos explicitly, as tempo is always computed from track contents. */
  11.599 +}
  11.600 +
  11.601 +static void
  11.602 +assert_smf_saved_correctly(const smf_t *smf, const char *file_name)
  11.603 +{
  11.604 +	smf_t *saved;
  11.605 +
  11.606 +	saved = smf_load(file_name);
  11.607 +	assert(saved != NULL);
  11.608 +
  11.609 +	assert_smf_is_identical(smf, saved);
  11.610 +
  11.611 +	smf_delete(saved);
  11.612 +}
  11.613 +
  11.614 +#endif /* !NDEBUG */
  11.615 +
  11.616 +/**
  11.617 +  * Writes the contents of SMF to the file given.
  11.618 +  * \param smf SMF.
  11.619 +  * \param file_name Path to the file.
  11.620 +  * \return 0, if saving was successfull.
  11.621 +  */
  11.622 +int
  11.623 +smf_save(smf_t *smf, const char *file_name)
  11.624 +{
  11.625 +	int i, error;
  11.626 +	smf_track_t *track;
  11.627 +
  11.628 +	smf_rewind(smf);
  11.629 +
  11.630 +	assert(pointers_are_clear(smf));
  11.631 +
  11.632 +	if (smf_validate(smf))
  11.633 +		return (-1);
  11.634 +
  11.635 +	if (write_mthd_header(smf))
  11.636 +		return (-2);
  11.637 +
  11.638 +	for (i = 1; i <= smf->number_of_tracks; i++) {
  11.639 +		track = smf_get_track_by_number(smf, i);
  11.640 +
  11.641 +		assert(track != NULL);
  11.642 +
  11.643 +		error = write_track(track);
  11.644 +		if (error) {
  11.645 +			free_buffer(smf);
  11.646 +			return (error);
  11.647 +		}
  11.648 +	}
  11.649 +
  11.650 +	error = write_file(smf, file_name);
  11.651 +
  11.652 +	free_buffer(smf);
  11.653 +
  11.654 +	if (error)
  11.655 +		return (error);
  11.656 +
  11.657 +#ifndef NDEBUG
  11.658 +	assert_smf_saved_correctly(smf, file_name);
  11.659 +#endif
  11.660 +
  11.661 +	return (0);
  11.662 +}
  11.663 +
    12.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    12.2 +++ b/src/smf_tempo.c	Thu Jan 26 11:25:11 2012 +0200
    12.3 @@ -0,0 +1,448 @@
    12.4 +/*-
    12.5 + * Copyright (c) 2007, 2008 Edward Tomasz Napierała <trasz@FreeBSD.org>
    12.6 + * All rights reserved.
    12.7 + *
    12.8 + * Redistribution and use in source and binary forms, with or without
    12.9 + * modification, are permitted provided that the following conditions
   12.10 + * are met:
   12.11 + * 1. Redistributions of source code must retain the above copyright
   12.12 + *    notice, this list of conditions and the following disclaimer.
   12.13 + * 2. Redistributions in binary form must reproduce the above copyright
   12.14 + *    notice, this list of conditions and the following disclaimer in the
   12.15 + *    documentation and/or other materials provided with the distribution.
   12.16 + *
   12.17 + * ALTHOUGH THIS SOFTWARE IS MADE OF WIN AND SCIENCE, IT IS PROVIDED BY THE
   12.18 + * AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
   12.19 + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
   12.20 + * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
   12.21 + * THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
   12.22 + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
   12.23 + * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
   12.24 + * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
   12.25 + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
   12.26 + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
   12.27 + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   12.28 + *
   12.29 + */
   12.30 +
   12.31 +/**
   12.32 + * \file
   12.33 + *
   12.34 + * Tempo map related part.
   12.35 + *
   12.36 + */
   12.37 +
   12.38 +#include <stdlib.h>
   12.39 +#include <assert.h>
   12.40 +#include <math.h>
   12.41 +#include <string.h>
   12.42 +#include "smf.h"
   12.43 +#include "smf_private.h"
   12.44 +
   12.45 +static double seconds_from_pulses(const smf_t *smf, int pulses);
   12.46 +
   12.47 +/**
   12.48 + * If there is tempo starting at "pulses" already, return it.  Otherwise,
   12.49 + * allocate new one, fill it with values from previous one (or default ones,
   12.50 + * if there is no previous one) and attach it to "smf".
   12.51 + */
   12.52 +static smf_tempo_t *
   12.53 +new_tempo(smf_t *smf, int pulses)
   12.54 +{
   12.55 +	smf_tempo_t *tempo, *previous_tempo = NULL;
   12.56 +
   12.57 +	if (smf->tempo_array->len > 0) {
   12.58 +		previous_tempo = smf_get_last_tempo(smf);
   12.59 +
   12.60 +		/* If previous tempo starts at the same time as new one, reuse it, updating in place. */
   12.61 +		if (previous_tempo->time_pulses == pulses)
   12.62 +			return (previous_tempo);
   12.63 +	}
   12.64 +
   12.65 +	tempo = malloc(sizeof(smf_tempo_t));
   12.66 +	if (tempo == NULL) {
   12.67 +		fg_critical("Cannot allocate smf_tempo_t.");
   12.68 +		return (NULL);
   12.69 +	}
   12.70 +
   12.71 +	tempo->time_pulses = pulses;
   12.72 +
   12.73 +	if (previous_tempo != NULL) {
   12.74 +		tempo->microseconds_per_quarter_note = previous_tempo->microseconds_per_quarter_note;
   12.75 +		tempo->numerator = previous_tempo->numerator;
   12.76 +		tempo->denominator = previous_tempo->denominator;
   12.77 +		tempo->clocks_per_click = previous_tempo->clocks_per_click;
   12.78 +		tempo->notes_per_note = previous_tempo->notes_per_note;
   12.79 +	} else {
   12.80 +		tempo->microseconds_per_quarter_note = 500000; /* Initial tempo is 120 BPM. */
   12.81 +		tempo->numerator = 4;
   12.82 +		tempo->denominator = 4;
   12.83 +		tempo->clocks_per_click = -1;
   12.84 +		tempo->notes_per_note = -1;
   12.85 +	}
   12.86 +
   12.87 +	fg_ptr_array_add(smf->tempo_array, tempo);
   12.88 +
   12.89 +	if (pulses == 0)
   12.90 +		tempo->time_seconds = 0.0;
   12.91 +	else
   12.92 +		tempo->time_seconds = seconds_from_pulses(smf, pulses);
   12.93 +
   12.94 +	return (tempo);
   12.95 +}
   12.96 +
   12.97 +static int
   12.98 +add_tempo(smf_t *smf, int pulses, int tempo)
   12.99 +{
  12.100 +	smf_tempo_t *smf_tempo = new_tempo(smf, pulses);
  12.101 +	if (smf_tempo == NULL)
  12.102 +		return (-1);
  12.103 +
  12.104 +	smf_tempo->microseconds_per_quarter_note = tempo;
  12.105 +
  12.106 +	return (0);
  12.107 +}
  12.108 +
  12.109 +static int
  12.110 +add_time_signature(smf_t *smf, int pulses, int numerator, int denominator, int clocks_per_click, int notes_per_note)
  12.111 +{
  12.112 +	smf_tempo_t *smf_tempo = new_tempo(smf, pulses);
  12.113 +	if (smf_tempo == NULL)
  12.114 +		return (-1);
  12.115 +
  12.116 +	smf_tempo->numerator = numerator;
  12.117 +	smf_tempo->denominator = denominator;
  12.118 +	smf_tempo->clocks_per_click = clocks_per_click;
  12.119 +	smf_tempo->notes_per_note = notes_per_note;
  12.120 +
  12.121 +	return (0);
  12.122 +}
  12.123 +
  12.124 +/**
  12.125 + * \internal
  12.126 + */
  12.127 +void
  12.128 +maybe_add_to_tempo_map(smf_event_t *event)
  12.129 +{
  12.130 +	if (!smf_event_is_metadata(event))
  12.131 +		return;
  12.132 +
  12.133 +	assert(event->track != NULL);
  12.134 +	assert(event->track->smf != NULL);
  12.135 +	assert(event->midi_buffer_length >= 1);
  12.136 +
  12.137 +	/* Tempo Change? */
  12.138 +	if (event->midi_buffer[1] == 0x51) {
  12.139 +		int new_tempo = (event->midi_buffer[3] << 16) + (event->midi_buffer[4] << 8) + event->midi_buffer[5];
  12.140 +		if (new_tempo <= 0) {
  12.141 +			fg_critical("Ignoring invalid tempo change.");
  12.142 +			return;
  12.143 +		}
  12.144 +
  12.145 +		add_tempo(event->track->smf, event->time_pulses, new_tempo);
  12.146 +	}
  12.147 +
  12.148 +	/* Time Signature? */
  12.149 +	if (event->midi_buffer[1] == 0x58) {
  12.150 +		int numerator, denominator, clocks_per_click, notes_per_note;
  12.151 +
  12.152 +		if (event->midi_buffer_length < 7) {
  12.153 +			fg_critical("Time Signature event seems truncated.");
  12.154 +			return;
  12.155 +		}
  12.156 +
  12.157 +		numerator = event->midi_buffer[3];
  12.158 +		denominator = (int)pow(2, event->midi_buffer[4]);
  12.159 +		clocks_per_click = event->midi_buffer[5];
  12.160 +		notes_per_note = event->midi_buffer[6];
  12.161 +
  12.162 +		add_time_signature(event->track->smf, event->time_pulses, numerator, denominator, clocks_per_click, notes_per_note);
  12.163 +	}
  12.164 +
  12.165 +	return;
  12.166 +}
  12.167 +
  12.168 +/**
  12.169 + * \internal
  12.170 + *
  12.171 + * This is an internal function, called from smf_track_remove_event when tempo-related
  12.172 + * event being removed does not require recreation of tempo map, i.e. there are no events
  12.173 + * after that one.
  12.174 + */
  12.175 +void
  12.176 +remove_last_tempo_with_pulses(smf_t *smf, int pulses)
  12.177 +{
  12.178 +	smf_tempo_t *tempo;
  12.179 +
  12.180 +	/* XXX: This is a partial workaround for the following problem: we have two tempo-related
  12.181 +	   events, A and B, that occur at the same time.  We remove B, then try to remove
  12.182 +	   A.  However, both tempo changes got coalesced in new_tempo(), so it is impossible
  12.183 +	   to remove B. */
  12.184 +	if (smf->tempo_array->len == 0)
  12.185 +		return;
  12.186 +
  12.187 +	tempo = smf_get_last_tempo(smf);
  12.188 +
  12.189 +	/* Workaround part two. */
  12.190 +	if (tempo->time_pulses != pulses)
  12.191 +		return;
  12.192 +
  12.193 +	memset(tempo, 0, sizeof(smf_tempo_t));
  12.194 +	free(tempo);
  12.195 +
  12.196 +	fg_ptr_array_remove_index(smf->tempo_array, smf->tempo_array->len - 1);
  12.197 +}
  12.198 +
  12.199 +static double
  12.200 +seconds_from_pulses(const smf_t *smf, int pulses)
  12.201 +{
  12.202 +	double seconds;
  12.203 +	smf_tempo_t *tempo;
  12.204 +
  12.205 +	tempo = smf_get_tempo_by_pulses(smf, pulses);
  12.206 +	assert(tempo);
  12.207 +	assert(tempo->time_pulses <= pulses);
  12.208 +
  12.209 +	seconds = tempo->time_seconds + (double)(pulses - tempo->time_pulses) *
  12.210 +		(tempo->microseconds_per_quarter_note / ((double)smf->ppqn * 1000000.0));
  12.211 +
  12.212 +	return (seconds);
  12.213 +}
  12.214 +
  12.215 +static int
  12.216 +pulses_from_seconds(const smf_t *smf, double seconds)
  12.217 +{
  12.218 +	int pulses = 0;
  12.219 +	smf_tempo_t *tempo;
  12.220 +
  12.221 +	tempo = smf_get_tempo_by_seconds(smf, seconds);
  12.222 +	assert(tempo);
  12.223 +	assert(tempo->time_seconds <= seconds);
  12.224 +
  12.225 +	pulses = tempo->time_pulses + (seconds - tempo->time_seconds) *
  12.226 +		((double)smf->ppqn * 1000000.0 / tempo->microseconds_per_quarter_note);
  12.227 +
  12.228 +	return (pulses);
  12.229 +}
  12.230 +
  12.231 +/**
  12.232 + * \internal
  12.233 + *
  12.234 + * Computes value of event->time_seconds for all events in smf.
  12.235 + * Warning: rewinds the smf.
  12.236 + */
  12.237 +void
  12.238 +smf_create_tempo_map_and_compute_seconds(smf_t *smf)
  12.239 +{
  12.240 +	smf_event_t *event;
  12.241 +
  12.242 +	smf_rewind(smf);
  12.243 +	smf_init_tempo(smf);
  12.244 +
  12.245 +	for (;;) {
  12.246 +		event = smf_get_next_event(smf);
  12.247 +
  12.248 +		if (event == NULL)
  12.249 +			return;
  12.250 +
  12.251 +		maybe_add_to_tempo_map(event);
  12.252 +
  12.253 +		event->time_seconds = seconds_from_pulses(smf, event->time_pulses);
  12.254 +	}
  12.255 +
  12.256 +	/* Not reached. */
  12.257 +}
  12.258 +
  12.259 +smf_tempo_t *
  12.260 +smf_get_tempo_by_number(const smf_t *smf, int number)
  12.261 +{
  12.262 +	assert(number >= 0);
  12.263 +
  12.264 +	if (number >= smf->tempo_array->len)
  12.265 +		return (NULL);
  12.266 +
  12.267 +	return (fg_ptr_array_index(smf->tempo_array, number));
  12.268 +}
  12.269 +
  12.270 +/**
  12.271 + * Return last tempo (i.e. tempo with greatest time_pulses) that happens before "pulses".
  12.272 + */
  12.273 +smf_tempo_t *
  12.274 +smf_get_tempo_by_pulses(const smf_t *smf, int pulses)
  12.275 +{
  12.276 +	int i;
  12.277 +	smf_tempo_t *tempo;
  12.278 +
  12.279 +	assert(pulses >= 0);
  12.280 +
  12.281 +	if (pulses == 0)
  12.282 +		return (smf_get_tempo_by_number(smf, 0));
  12.283 +
  12.284 +	assert(smf->tempo_array != NULL);
  12.285 +
  12.286 +	for (i = smf->tempo_array->len - 1; i >= 0; i--) {
  12.287 +		tempo = smf_get_tempo_by_number(smf, i);
  12.288 +
  12.289 +		assert(tempo);
  12.290 +		if (tempo->time_pulses < pulses)
  12.291 +			return (tempo);
  12.292 +	}
  12.293 +
  12.294 +	return (NULL);
  12.295 +}
  12.296 +
  12.297 +/**
  12.298 + * Return last tempo (i.e. tempo with greatest time_seconds) that happens before "seconds".
  12.299 + */
  12.300 +smf_tempo_t *
  12.301 +smf_get_tempo_by_seconds(const smf_t *smf, double seconds)
  12.302 +{
  12.303 +	int i;
  12.304 +	smf_tempo_t *tempo;
  12.305 +
  12.306 +	assert(seconds >= 0.0);
  12.307 +
  12.308 +	if (seconds == 0.0)
  12.309 +		return (smf_get_tempo_by_number(smf, 0));
  12.310 +
  12.311 +	assert(smf->tempo_array != NULL);
  12.312 +
  12.313 +	for (i = smf->tempo_array->len - 1; i >= 0; i--) {
  12.314 +		tempo = smf_get_tempo_by_number(smf, i);
  12.315 +
  12.316 +		assert(tempo);
  12.317 +		if (tempo->time_seconds < seconds)
  12.318 +			return (tempo);
  12.319 +	}
  12.320 +
  12.321 +	return (NULL);
  12.322 +}
  12.323 +
  12.324 +
  12.325 +/**
  12.326 + * Return last tempo.
  12.327 + */
  12.328 +smf_tempo_t *
  12.329 +smf_get_last_tempo(const smf_t *smf)
  12.330 +{
  12.331 +	smf_tempo_t *tempo;
  12.332 +
  12.333 +	tempo = smf_get_tempo_by_number(smf, smf->tempo_array->len - 1);
  12.334 +	assert(tempo);
  12.335 +
  12.336 +	return (tempo);
  12.337 +}
  12.338 +
  12.339 +/**
  12.340 + * \internal 
  12.341 + *
  12.342 + * Remove all smf_tempo_t structures from SMF.
  12.343 + */
  12.344 +void
  12.345 +smf_fini_tempo(smf_t *smf)
  12.346 +{
  12.347 +	smf_tempo_t *tempo;
  12.348 +
  12.349 +	while (smf->tempo_array->len > 0) {
  12.350 +		tempo = fg_ptr_array_index(smf->tempo_array, smf->tempo_array->len - 1);
  12.351 +		assert(tempo);
  12.352 +
  12.353 +		memset(tempo, 0, sizeof(smf_tempo_t));
  12.354 +		free(tempo);
  12.355 +
  12.356 +		fg_ptr_array_remove_index(smf->tempo_array, smf->tempo_array->len - 1);
  12.357 +	}
  12.358 +
  12.359 +	assert(smf->tempo_array->len == 0);
  12.360 +}
  12.361 +
  12.362 +/**
  12.363 + * \internal
  12.364 + *
  12.365 + * Remove any existing tempos and add default one.
  12.366 + *
  12.367 + * \bug This will abort (by calling fg_error) if new_tempo() (memory allocation there) fails.
  12.368 + */
  12.369 +void
  12.370 +smf_init_tempo(smf_t *smf)
  12.371 +{
  12.372 +	smf_tempo_t *tempo;
  12.373 +
  12.374 +	smf_fini_tempo(smf);
  12.375 +
  12.376 +	tempo = new_tempo(smf, 0);
  12.377 +	if (tempo == NULL)
  12.378 +		fg_error("tempo_init failed, sorry.");
  12.379 +}
  12.380 +
  12.381 +/**
  12.382 + * Returns ->time_pulses of last event on the given track, or 0, if track is empty.
  12.383 + */
  12.384 +static int
  12.385 +last_event_pulses(const smf_track_t *track)
  12.386 +{
  12.387 +	/* Get time of last event on this track. */
  12.388 +	if (track->number_of_events > 0) {
  12.389 +		smf_event_t *previous_event = smf_track_get_last_event(track);
  12.390 +		assert(previous_event);
  12.391 +		assert(previous_event->time_pulses >= 0);
  12.392 +
  12.393 +		return (previous_event->time_pulses);
  12.394 +	}
  12.395 +
  12.396 +	return (0);
  12.397 +}
  12.398 +
  12.399 +/**
  12.400 + * Adds event to the track at the time "pulses" clocks from the previous event in this track.
  12.401 + * The remaining two time fields will be computed automatically based on the third argument
  12.402 + * and current tempo map.  Note that ->delta_pulses is computed by smf.c:smf_track_add_event,
  12.403 + * not here.
  12.404 + */
  12.405 +void
  12.406 +smf_track_add_event_delta_pulses(smf_track_t *track, smf_event_t *event, int delta)
  12.407 +{
  12.408 +	assert(delta >= 0);
  12.409 +	assert(event->time_pulses == -1);
  12.410 +	assert(event->time_seconds == -1.0);
  12.411 +	assert(track->smf != NULL);
  12.412 +
  12.413 +	smf_track_add_event_pulses(track, event, last_event_pulses(track) + delta);
  12.414 +}
  12.415 +
  12.416 +/**
  12.417 + * Adds event to the track at the time "pulses" clocks from the start of song.
  12.418 + * The remaining two time fields will be computed automatically based on the third argument
  12.419 + * and current tempo map.
  12.420 + */
  12.421 +void
  12.422 +smf_track_add_event_pulses(smf_track_t *track, smf_event_t *event, int pulses)
  12.423 +{
  12.424 +	assert(pulses >= 0);
  12.425 +	assert(event->time_pulses == -1);
  12.426 +	assert(event->time_seconds == -1.0);
  12.427 +	assert(track->smf != NULL);
  12.428 +
  12.429 +	event->time_pulses = pulses;
  12.430 +	event->time_seconds = seconds_from_pulses(track->smf, pulses);
  12.431 +	smf_track_add_event(track, event);
  12.432 +}
  12.433 +
  12.434 +/**
  12.435 + * Adds event to the track at the time "seconds" seconds from the start of song.
  12.436 + * The remaining two time fields will be computed automatically based on the third argument
  12.437 + * and current tempo map.
  12.438 + */
  12.439 +void
  12.440 +smf_track_add_event_seconds(smf_track_t *track, smf_event_t *event, double seconds)
  12.441 +{
  12.442 +	assert(seconds >= 0.0);
  12.443 +	assert(event->time_pulses == -1);
  12.444 +	assert(event->time_seconds == -1.0);
  12.445 +	assert(track->smf != NULL);
  12.446 +
  12.447 +	event->time_seconds = seconds;
  12.448 +	event->time_pulses = pulses_from_seconds(track->smf, seconds);
  12.449 +	smf_track_add_event(track, event);
  12.450 +}
  12.451 +