Blame amidi/amidi.c

Packit Service a9274b
/*
Packit Service a9274b
 *  amidi.c - read from/write to RawMIDI ports
Packit Service a9274b
 *
Packit Service a9274b
 *  Copyright (c) Clemens Ladisch <clemens@ladisch.de>
Packit Service a9274b
 *
Packit Service a9274b
 *
Packit Service a9274b
 *   This program is free software; you can redistribute it and/or modify
Packit Service a9274b
 *   it under the terms of the GNU General Public License as published by
Packit Service a9274b
 *   the Free Software Foundation; either version 2 of the License, or
Packit Service a9274b
 *   (at your option) any later version.
Packit Service a9274b
 *
Packit Service a9274b
 *   This program is distributed in the hope that it will be useful,
Packit Service a9274b
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service a9274b
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service a9274b
 *   GNU General Public License for more details.
Packit Service a9274b
 *
Packit Service a9274b
 *   You should have received a copy of the GNU General Public License
Packit Service a9274b
 *   along with this program; if not, write to the Free Software
Packit Service a9274b
 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit Service a9274b
 */
Packit Service a9274b
Packit Service a9274b
#define _GNU_SOURCE
Packit Service a9274b
#include <stdio.h>
Packit Service a9274b
#include <stdlib.h>
Packit Service a9274b
#include <stdarg.h>
Packit Service a9274b
#include <string.h>
Packit Service a9274b
#include <ctype.h>
Packit Service a9274b
#include <math.h>
Packit Service a9274b
#include <getopt.h>
Packit Service a9274b
#include <errno.h>
Packit Service a9274b
#include <signal.h>
Packit Service a9274b
#include <sys/timerfd.h>
Packit Service a9274b
#include <sys/types.h>
Packit Service a9274b
#include <poll.h>
Packit Service a9274b
#include <sys/stat.h>
Packit Service a9274b
#include <unistd.h>
Packit Service a9274b
#include <fcntl.h>
Packit Service a9274b
#include <alsa/asoundlib.h>
Packit Service a9274b
#include "aconfig.h"
Packit Service a9274b
#include "version.h"
Packit Service a9274b
Packit Service a9274b
#define NSEC_PER_SEC 1000000000L
Packit Service a9274b
Packit Service a9274b
static int do_device_list, do_rawmidi_list;
Packit Service a9274b
static char *port_name = "default";
Packit Service a9274b
static char *send_file_name;
Packit Service a9274b
static char *receive_file_name;
Packit Service a9274b
static char *send_hex;
Packit Service a9274b
static char *send_data;
Packit Service a9274b
static int send_data_length;
Packit Service a9274b
static int receive_file;
Packit Service a9274b
static int dump;
Packit Service a9274b
static float timeout;
Packit Service a9274b
static int stop;
Packit Service a9274b
static int sysex_interval;
Packit Service a9274b
static snd_rawmidi_t *input, **inputp;
Packit Service a9274b
static snd_rawmidi_t *output, **outputp;
Packit Service a9274b
Packit Service a9274b
static void error(const char *format, ...)
Packit Service a9274b
{
Packit Service a9274b
	va_list ap;
Packit Service a9274b
Packit Service a9274b
	va_start(ap, format);
Packit Service a9274b
	vfprintf(stderr, format, ap);
Packit Service a9274b
	va_end(ap);
Packit Service a9274b
	putc('\n', stderr);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void usage(void)
Packit Service a9274b
{
Packit Service a9274b
	printf(
Packit Service a9274b
		"Usage: amidi options\n"
Packit Service a9274b
		"\n"
Packit Service a9274b
		"-h, --help                      this help\n"
Packit Service a9274b
		"-V, --version                   print current version\n"
Packit Service a9274b
		"-l, --list-devices              list all hardware ports\n"
Packit Service a9274b
		"-L, --list-rawmidis             list all RawMIDI definitions\n"
Packit Service a9274b
		"-p, --port=name                 select port by name\n"
Packit Service a9274b
		"-s, --send=file                 send the contents of a (.syx) file\n"
Packit Service a9274b
		"-r, --receive=file              write received data into a file\n"
Packit Service a9274b
		"-S, --send-hex=\"...\"            send hexadecimal bytes\n"
Packit Service a9274b
		"-d, --dump                      print received data as hexadecimal bytes\n"
Packit Service a9274b
		"-t, --timeout=seconds           exits when no data has been received\n"
Packit Service a9274b
		"                                for the specified duration\n"
Packit Service a9274b
		"-a, --active-sensing            include active sensing bytes\n"
Packit Service a9274b
		"-c, --clock                     include clock bytes\n"
Packit Service a9274b
		"-i, --sysex-interval=mseconds   delay in between each SysEx message\n");
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void version(void)
Packit Service a9274b
{
Packit Service a9274b
	puts("amidi version " SND_UTIL_VERSION_STR);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void *my_malloc(size_t size)
Packit Service a9274b
{
Packit Service a9274b
	void *p = malloc(size);
Packit Service a9274b
	if (!p) {
Packit Service a9274b
		error("out of memory");
Packit Service a9274b
		exit(EXIT_FAILURE);
Packit Service a9274b
	}
Packit Service a9274b
	return p;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void list_device(snd_ctl_t *ctl, int card, int device)
Packit Service a9274b
{
Packit Service a9274b
	snd_rawmidi_info_t *info;
Packit Service a9274b
	const char *name;
Packit Service a9274b
	const char *sub_name;
Packit Service a9274b
	int subs, subs_in, subs_out;
Packit Service a9274b
	int sub;
Packit Service a9274b
	int err;
Packit Service a9274b
Packit Service a9274b
	snd_rawmidi_info_alloca(&info;;
Packit Service a9274b
	snd_rawmidi_info_set_device(info, device);
Packit Service a9274b
Packit Service a9274b
	snd_rawmidi_info_set_stream(info, SND_RAWMIDI_STREAM_INPUT);
Packit Service a9274b
	err = snd_ctl_rawmidi_info(ctl, info);
Packit Service a9274b
	if (err >= 0)
Packit Service a9274b
		subs_in = snd_rawmidi_info_get_subdevices_count(info);
Packit Service a9274b
	else
Packit Service a9274b
		subs_in = 0;
Packit Service a9274b
Packit Service a9274b
	snd_rawmidi_info_set_stream(info, SND_RAWMIDI_STREAM_OUTPUT);
Packit Service a9274b
	err = snd_ctl_rawmidi_info(ctl, info);
Packit Service a9274b
	if (err >= 0)
Packit Service a9274b
		subs_out = snd_rawmidi_info_get_subdevices_count(info);
Packit Service a9274b
	else
Packit Service a9274b
		subs_out = 0;
Packit Service a9274b
Packit Service a9274b
	subs = subs_in > subs_out ? subs_in : subs_out;
Packit Service a9274b
	if (!subs)
Packit Service a9274b
		return;
Packit Service a9274b
Packit Service a9274b
	for (sub = 0; sub < subs; ++sub) {
Packit Service a9274b
		snd_rawmidi_info_set_stream(info, sub < subs_in ?
Packit Service a9274b
					    SND_RAWMIDI_STREAM_INPUT :
Packit Service a9274b
					    SND_RAWMIDI_STREAM_OUTPUT);
Packit Service a9274b
		snd_rawmidi_info_set_subdevice(info, sub);
Packit Service a9274b
		err = snd_ctl_rawmidi_info(ctl, info);
Packit Service a9274b
		if (err < 0) {
Packit Service a9274b
			error("cannot get rawmidi information %d:%d:%d: %s\n",
Packit Service a9274b
			      card, device, sub, snd_strerror(err));
Packit Service a9274b
			return;
Packit Service a9274b
		}
Packit Service a9274b
		name = snd_rawmidi_info_get_name(info);
Packit Service a9274b
		sub_name = snd_rawmidi_info_get_subdevice_name(info);
Packit Service a9274b
		if (sub == 0 && sub_name[0] == '\0') {
Packit Service a9274b
			printf("%c%c  hw:%d,%d    %s",
Packit Service a9274b
			       sub < subs_in ? 'I' : ' ',
Packit Service a9274b
			       sub < subs_out ? 'O' : ' ',
Packit Service a9274b
			       card, device, name);
Packit Service a9274b
			if (subs > 1)
Packit Service a9274b
				printf(" (%d subdevices)", subs);
Packit Service a9274b
			putchar('\n');
Packit Service a9274b
			break;
Packit Service a9274b
		} else {
Packit Service a9274b
			printf("%c%c  hw:%d,%d,%d  %s\n",
Packit Service a9274b
			       sub < subs_in ? 'I' : ' ',
Packit Service a9274b
			       sub < subs_out ? 'O' : ' ',
Packit Service a9274b
			       card, device, sub, sub_name);
Packit Service a9274b
		}
Packit Service a9274b
	}
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void list_card_devices(int card)
Packit Service a9274b
{
Packit Service a9274b
	snd_ctl_t *ctl;
Packit Service a9274b
	char name[32];
Packit Service a9274b
	int device;
Packit Service a9274b
	int err;
Packit Service a9274b
Packit Service a9274b
	sprintf(name, "hw:%d", card);
Packit Service a9274b
	if ((err = snd_ctl_open(&ctl, name, 0)) < 0) {
Packit Service a9274b
		error("cannot open control for card %d: %s", card, snd_strerror(err));
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	device = -1;
Packit Service a9274b
	for (;;) {
Packit Service a9274b
		if ((err = snd_ctl_rawmidi_next_device(ctl, &device)) < 0) {
Packit Service a9274b
			error("cannot determine device number: %s", snd_strerror(err));
Packit Service a9274b
			break;
Packit Service a9274b
		}
Packit Service a9274b
		if (device < 0)
Packit Service a9274b
			break;
Packit Service a9274b
		list_device(ctl, card, device);
Packit Service a9274b
	}
Packit Service a9274b
	snd_ctl_close(ctl);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void device_list(void)
Packit Service a9274b
{
Packit Service a9274b
	int card, err;
Packit Service a9274b
Packit Service a9274b
	card = -1;
Packit Service a9274b
	if ((err = snd_card_next(&card)) < 0) {
Packit Service a9274b
		error("cannot determine card number: %s", snd_strerror(err));
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	if (card < 0) {
Packit Service a9274b
		error("no sound card found");
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	puts("Dir Device    Name");
Packit Service a9274b
	do {
Packit Service a9274b
		list_card_devices(card);
Packit Service a9274b
		if ((err = snd_card_next(&card)) < 0) {
Packit Service a9274b
			error("cannot determine card number: %s", snd_strerror(err));
Packit Service a9274b
			break;
Packit Service a9274b
		}
Packit Service a9274b
	} while (card >= 0);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void rawmidi_list(void)
Packit Service a9274b
{
Packit Service a9274b
	snd_output_t *output;
Packit Service a9274b
	snd_config_t *config;
Packit Service a9274b
	int err;
Packit Service a9274b
Packit Service a9274b
	if ((err = snd_config_update()) < 0) {
Packit Service a9274b
		error("snd_config_update failed: %s", snd_strerror(err));
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	if ((err = snd_output_stdio_attach(&output, stdout, 0)) < 0) {
Packit Service a9274b
		error("snd_output_stdio_attach failed: %s", snd_strerror(err));
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	if (snd_config_search(snd_config, "rawmidi", &config) >= 0) {
Packit Service a9274b
		puts("RawMIDI list:");
Packit Service a9274b
		snd_config_save(config, output);
Packit Service a9274b
	}
Packit Service a9274b
	snd_output_close(output);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static int send_midi_interleaved(void)
Packit Service a9274b
{
Packit Service a9274b
	int err;
Packit Service a9274b
	char *data = send_data;
Packit Service a9274b
	size_t buffer_size;
Packit Service a9274b
	snd_rawmidi_params_t *param;
Packit Service a9274b
	snd_rawmidi_status_t *st;
Packit Service a9274b
Packit Service a9274b
	snd_rawmidi_status_alloca(&st);
Packit Service a9274b
Packit Service a9274b
	snd_rawmidi_params_alloca(¶m;;
Packit Service a9274b
	snd_rawmidi_params_current(output, param);
Packit Service a9274b
	buffer_size = snd_rawmidi_params_get_buffer_size(param);
Packit Service a9274b
Packit Service a9274b
	while (data < (send_data + send_data_length)) {
Packit Service a9274b
		int len = send_data + send_data_length - data;
Packit Service a9274b
		char *temp;
Packit Service a9274b
Packit Service a9274b
		if (data > send_data) {
Packit Service a9274b
			snd_rawmidi_status(output, st);
Packit Service a9274b
			do {
Packit Service a9274b
				/* 320 µs per byte as noted in Page 1 of MIDI spec */
Packit Service a9274b
				usleep((buffer_size - snd_rawmidi_status_get_avail(st)) * 320);
Packit Service a9274b
				snd_rawmidi_status(output, st);
Packit Service a9274b
			} while(snd_rawmidi_status_get_avail(st) < buffer_size);
Packit Service a9274b
			usleep(sysex_interval * 1000);
Packit Service a9274b
		}
Packit Service a9274b
Packit Service a9274b
		/* find end of SysEx */
Packit Service a9274b
		if ((temp = memchr(data, 0xf7, len)) != NULL)
Packit Service a9274b
			len = temp - data + 1;
Packit Service a9274b
Packit Service a9274b
		if ((err = snd_rawmidi_write(output, data, len)) < 0)
Packit Service a9274b
			return err;
Packit Service a9274b
Packit Service a9274b
		data += len;
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	return 0;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void load_file(void)
Packit Service a9274b
{
Packit Service a9274b
	int fd;
Packit Service a9274b
	off_t length;
Packit Service a9274b
Packit Service a9274b
	fd = open(send_file_name, O_RDONLY);
Packit Service a9274b
	if (fd == -1) {
Packit Service a9274b
		error("cannot open %s - %s", send_file_name, strerror(errno));
Packit Service a9274b
		return;
Packit Service a9274b
	}
Packit Service a9274b
	length = lseek(fd, 0, SEEK_END);
Packit Service a9274b
	if (length == (off_t)-1) {
Packit Service a9274b
		error("cannot determine length of %s: %s", send_file_name, strerror(errno));
Packit Service a9274b
		goto _error;
Packit Service a9274b
	}
Packit Service a9274b
	send_data = my_malloc(length);
Packit Service a9274b
	lseek(fd, 0, SEEK_SET);
Packit Service a9274b
	if (read(fd, send_data, length) != length) {
Packit Service a9274b
		error("cannot read from %s: %s", send_file_name, strerror(errno));
Packit Service a9274b
		goto _error;
Packit Service a9274b
	}
Packit Service a9274b
	if (length >= 4 && !memcmp(send_data, "MThd", 4)) {
Packit Service a9274b
		error("%s is a Standard MIDI File; use aplaymidi to send it", send_file_name);
Packit Service a9274b
		goto _error;
Packit Service a9274b
	}
Packit Service a9274b
	send_data_length = length;
Packit Service a9274b
	goto _exit;
Packit Service a9274b
_error:
Packit Service a9274b
	free(send_data);
Packit Service a9274b
	send_data = NULL;
Packit Service a9274b
_exit:
Packit Service a9274b
	close(fd);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static int hex_value(char c)
Packit Service a9274b
{
Packit Service a9274b
	if ('0' <= c && c <= '9')
Packit Service a9274b
		return c - '0';
Packit Service a9274b
	if ('A' <= c && c <= 'F')
Packit Service a9274b
		return c - 'A' + 10;
Packit Service a9274b
	if ('a' <= c && c <= 'f')
Packit Service a9274b
		return c - 'a' + 10;
Packit Service a9274b
	error("invalid character %c", c);
Packit Service a9274b
	return -1;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void parse_data(void)
Packit Service a9274b
{
Packit Service a9274b
	const char *p;
Packit Service a9274b
	int i, value;
Packit Service a9274b
Packit Service a9274b
	send_data = my_malloc(strlen(send_hex)); /* guesstimate */
Packit Service a9274b
	i = 0;
Packit Service a9274b
	value = -1; /* value is >= 0 when the first hex digit of a byte has been read */
Packit Service a9274b
	for (p = send_hex; *p; ++p) {
Packit Service a9274b
		int digit;
Packit Service a9274b
		if (isspace((unsigned char)*p)) {
Packit Service a9274b
			if (value >= 0) {
Packit Service a9274b
				send_data[i++] = value;
Packit Service a9274b
				value = -1;
Packit Service a9274b
			}
Packit Service a9274b
			continue;
Packit Service a9274b
		}
Packit Service a9274b
		digit = hex_value(*p);
Packit Service a9274b
		if (digit < 0) {
Packit Service a9274b
			send_data = NULL;
Packit Service a9274b
			return;
Packit Service a9274b
		}
Packit Service a9274b
		if (value < 0) {
Packit Service a9274b
			value = digit;
Packit Service a9274b
		} else {
Packit Service a9274b
			send_data[i++] = (value << 4) | digit;
Packit Service a9274b
			value = -1;
Packit Service a9274b
		}
Packit Service a9274b
	}
Packit Service a9274b
	if (value >= 0)
Packit Service a9274b
		send_data[i++] = value;
Packit Service a9274b
	send_data_length = i;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
/*
Packit Service a9274b
 * prints MIDI commands, formatting them nicely
Packit Service a9274b
 */
Packit Service a9274b
static void print_byte(unsigned char byte)
Packit Service a9274b
{
Packit Service a9274b
	static enum {
Packit Service a9274b
		STATE_UNKNOWN,
Packit Service a9274b
		STATE_1PARAM,
Packit Service a9274b
		STATE_1PARAM_CONTINUE,
Packit Service a9274b
		STATE_2PARAM_1,
Packit Service a9274b
		STATE_2PARAM_2,
Packit Service a9274b
		STATE_2PARAM_1_CONTINUE,
Packit Service a9274b
		STATE_SYSEX
Packit Service a9274b
	} state = STATE_UNKNOWN;
Packit Service a9274b
	int newline = 0;
Packit Service a9274b
Packit Service a9274b
	if (byte >= 0xf8)
Packit Service a9274b
		newline = 1;
Packit Service a9274b
	else if (byte >= 0xf0) {
Packit Service a9274b
		newline = 1;
Packit Service a9274b
		switch (byte) {
Packit Service a9274b
		case 0xf0:
Packit Service a9274b
			state = STATE_SYSEX;
Packit Service a9274b
			break;
Packit Service a9274b
		case 0xf1:
Packit Service a9274b
		case 0xf3:
Packit Service a9274b
			state = STATE_1PARAM;
Packit Service a9274b
			break;
Packit Service a9274b
		case 0xf2:
Packit Service a9274b
			state = STATE_2PARAM_1;
Packit Service a9274b
			break;
Packit Service a9274b
		case 0xf4:
Packit Service a9274b
		case 0xf5:
Packit Service a9274b
		case 0xf6:
Packit Service a9274b
			state = STATE_UNKNOWN;
Packit Service a9274b
			break;
Packit Service a9274b
		case 0xf7:
Packit Service a9274b
			newline = state != STATE_SYSEX;
Packit Service a9274b
			state = STATE_UNKNOWN;
Packit Service a9274b
			break;
Packit Service a9274b
		}
Packit Service a9274b
	} else if (byte >= 0x80) {
Packit Service a9274b
		newline = 1;
Packit Service a9274b
		if (byte >= 0xc0 && byte <= 0xdf)
Packit Service a9274b
			state = STATE_1PARAM;
Packit Service a9274b
		else
Packit Service a9274b
			state = STATE_2PARAM_1;
Packit Service a9274b
	} else /* b < 0x80 */ {
Packit Service a9274b
		int running_status = 0;
Packit Service a9274b
		newline = state == STATE_UNKNOWN;
Packit Service a9274b
		switch (state) {
Packit Service a9274b
		case STATE_1PARAM:
Packit Service a9274b
			state = STATE_1PARAM_CONTINUE;
Packit Service a9274b
			break;
Packit Service a9274b
		case STATE_1PARAM_CONTINUE:
Packit Service a9274b
			running_status = 1;
Packit Service a9274b
			break;
Packit Service a9274b
		case STATE_2PARAM_1:
Packit Service a9274b
			state = STATE_2PARAM_2;
Packit Service a9274b
			break;
Packit Service a9274b
		case STATE_2PARAM_2:
Packit Service a9274b
			state = STATE_2PARAM_1_CONTINUE;
Packit Service a9274b
			break;
Packit Service a9274b
		case STATE_2PARAM_1_CONTINUE:
Packit Service a9274b
			running_status = 1;
Packit Service a9274b
			state = STATE_2PARAM_2;
Packit Service a9274b
			break;
Packit Service a9274b
		default:
Packit Service a9274b
			break;
Packit Service a9274b
		}
Packit Service a9274b
		if (running_status)
Packit Service a9274b
			fputs("\n  ", stdout);
Packit Service a9274b
	}
Packit Service a9274b
	printf("%c%02X", newline ? '\n' : ' ', byte);
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void sig_handler(int dummy)
Packit Service a9274b
{
Packit Service a9274b
	stop = 1;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
static void add_send_hex_data(const char *str)
Packit Service a9274b
{
Packit Service a9274b
	int length;
Packit Service a9274b
	char *s;
Packit Service a9274b
Packit Service a9274b
	length = (send_hex ? strlen(send_hex) + 1 : 0) + strlen(str) + 1;
Packit Service a9274b
	s = my_malloc(length);
Packit Service a9274b
	if (send_hex) {
Packit Service a9274b
		strcpy(s, send_hex);
Packit Service a9274b
		strcat(s, " ");
Packit Service a9274b
	} else {
Packit Service a9274b
		s[0] = '\0';
Packit Service a9274b
	}
Packit Service a9274b
	strcat(s, str);
Packit Service a9274b
	free(send_hex);
Packit Service a9274b
	send_hex = s;
Packit Service a9274b
}
Packit Service a9274b
Packit Service a9274b
int main(int argc, char *argv[])
Packit Service a9274b
{
Packit Service a9274b
	static const char short_options[] = "hVlLp:s:r:S::dt:aci:";
Packit Service a9274b
	static const struct option long_options[] = {
Packit Service a9274b
		{"help", 0, NULL, 'h'},
Packit Service a9274b
		{"version", 0, NULL, 'V'},
Packit Service a9274b
		{"list-devices", 0, NULL, 'l'},
Packit Service a9274b
		{"list-rawmidis", 0, NULL, 'L'},
Packit Service a9274b
		{"port", 1, NULL, 'p'},
Packit Service a9274b
		{"send", 1, NULL, 's'},
Packit Service a9274b
		{"receive", 1, NULL, 'r'},
Packit Service a9274b
		{"send-hex", 2, NULL, 'S'},
Packit Service a9274b
		{"dump", 0, NULL, 'd'},
Packit Service a9274b
		{"timeout", 1, NULL, 't'},
Packit Service a9274b
		{"active-sensing", 0, NULL, 'a'},
Packit Service a9274b
		{"clock", 0, NULL, 'c'},
Packit Service a9274b
		{"sysex-interval", 1, NULL, 'i'},
Packit Service a9274b
		{0}
Packit Service a9274b
	};
Packit Service a9274b
	int c, err, ok = 0;
Packit Service a9274b
	int ignore_active_sensing = 1;
Packit Service a9274b
	int ignore_clock = 1;
Packit Service a9274b
	int do_send_hex = 0;
Packit Service a9274b
	struct itimerspec itimerspec = { .it_interval = { 0, 0 } };
Packit Service a9274b
Packit Service a9274b
	while ((c = getopt_long(argc, argv, short_options,
Packit Service a9274b
		     		long_options, NULL)) != -1) {
Packit Service a9274b
		switch (c) {
Packit Service a9274b
		case 'h':
Packit Service a9274b
			usage();
Packit Service a9274b
			return 0;
Packit Service a9274b
		case 'V':
Packit Service a9274b
			version();
Packit Service a9274b
			return 0;
Packit Service a9274b
		case 'l':
Packit Service a9274b
			do_device_list = 1;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'L':
Packit Service a9274b
			do_rawmidi_list = 1;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'p':
Packit Service a9274b
			port_name = optarg;
Packit Service a9274b
			break;
Packit Service a9274b
		case 's':
Packit Service a9274b
			send_file_name = optarg;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'r':
Packit Service a9274b
			receive_file_name = optarg;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'S':
Packit Service a9274b
			do_send_hex = 1;
Packit Service a9274b
			if (optarg)
Packit Service a9274b
				add_send_hex_data(optarg);
Packit Service a9274b
			break;
Packit Service a9274b
		case 'd':
Packit Service a9274b
			dump = 1;
Packit Service a9274b
			break;
Packit Service a9274b
		case 't':
Packit Service a9274b
			if (optarg)
Packit Service a9274b
				timeout = atof(optarg);
Packit Service a9274b
			break;
Packit Service a9274b
		case 'a':
Packit Service a9274b
			ignore_active_sensing = 0;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'c':
Packit Service a9274b
			ignore_clock = 0;
Packit Service a9274b
			break;
Packit Service a9274b
		case 'i':
Packit Service a9274b
			sysex_interval = atoi(optarg);
Packit Service a9274b
			break;
Packit Service a9274b
		default:
Packit Service a9274b
			error("Try `amidi --help' for more information.");
Packit Service a9274b
			return 1;
Packit Service a9274b
		}
Packit Service a9274b
	}
Packit Service a9274b
	if (do_send_hex) {
Packit Service a9274b
		/* data for -S can be specified as multiple arguments */
Packit Service a9274b
		if (!send_hex && !argv[optind]) {
Packit Service a9274b
			error("Please specify some data for --send-hex.");
Packit Service a9274b
			return 1;
Packit Service a9274b
		}
Packit Service a9274b
		for (; argv[optind]; ++optind)
Packit Service a9274b
			add_send_hex_data(argv[optind]);
Packit Service a9274b
	} else {
Packit Service a9274b
		if (argv[optind]) {
Packit Service a9274b
			error("%s is not an option.", argv[optind]);
Packit Service a9274b
			return 1;
Packit Service a9274b
		}
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	if (do_rawmidi_list)
Packit Service a9274b
		rawmidi_list();
Packit Service a9274b
	if (do_device_list)
Packit Service a9274b
		device_list();
Packit Service a9274b
	if (do_rawmidi_list || do_device_list)
Packit Service a9274b
		return 0;
Packit Service a9274b
Packit Service a9274b
	if (!send_file_name && !receive_file_name && !send_hex && !dump) {
Packit Service a9274b
		error("Please specify at least one of --send, --receive, --send-hex, or --dump.");
Packit Service a9274b
		return 1;
Packit Service a9274b
	}
Packit Service a9274b
	if (send_file_name && send_hex) {
Packit Service a9274b
		error("--send and --send-hex cannot be specified at the same time.");
Packit Service a9274b
		return 1;
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	if (send_file_name)
Packit Service a9274b
		load_file();
Packit Service a9274b
	else if (send_hex)
Packit Service a9274b
		parse_data();
Packit Service a9274b
	if ((send_file_name || send_hex) && !send_data)
Packit Service a9274b
		return 1;
Packit Service a9274b
Packit Service a9274b
	if (receive_file_name) {
Packit Service a9274b
		receive_file = creat(receive_file_name, 0666);
Packit Service a9274b
		if (receive_file == -1) {
Packit Service a9274b
			error("cannot create %s: %s", receive_file_name, strerror(errno));
Packit Service a9274b
			return -1;
Packit Service a9274b
		}
Packit Service a9274b
	} else {
Packit Service a9274b
		receive_file = -1;
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	if (receive_file_name || dump)
Packit Service a9274b
		inputp = &input;
Packit Service a9274b
	else
Packit Service a9274b
		inputp = NULL;
Packit Service a9274b
	if (send_data)
Packit Service a9274b
		outputp = &output;
Packit Service a9274b
	else
Packit Service a9274b
		outputp = NULL;
Packit Service a9274b
Packit Service a9274b
	if ((err = snd_rawmidi_open(inputp, outputp, port_name, SND_RAWMIDI_NONBLOCK)) < 0) {
Packit Service a9274b
		error("cannot open port \"%s\": %s", port_name, snd_strerror(err));
Packit Service a9274b
		goto _exit2;
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	if (inputp)
Packit Service a9274b
		snd_rawmidi_read(input, NULL, 0); /* trigger reading */
Packit Service a9274b
Packit Service a9274b
	if (send_data) {
Packit Service a9274b
		if ((err = snd_rawmidi_nonblock(output, 0)) < 0) {
Packit Service a9274b
			error("cannot set blocking mode: %s", snd_strerror(err));
Packit Service a9274b
			goto _exit;
Packit Service a9274b
		}
Packit Service a9274b
		if (!sysex_interval) {
Packit Service a9274b
			if ((err = snd_rawmidi_write(output, send_data, send_data_length)) < 0) {
Packit Service a9274b
				error("cannot send data: %s", snd_strerror(err));
Packit Service a9274b
				return err;
Packit Service a9274b
			}
Packit Service a9274b
		} else {
Packit Service a9274b
			if ((err = send_midi_interleaved()) < 0) {
Packit Service a9274b
				error("cannot send data: %s", snd_strerror(err));
Packit Service a9274b
				return err;
Packit Service a9274b
			}
Packit Service a9274b
		}
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	if (inputp) {
Packit Service a9274b
		int read = 0;
Packit Service a9274b
		int npfds;
Packit Service a9274b
		struct pollfd *pfds;
Packit Service a9274b
Packit Service a9274b
		npfds = 1 + snd_rawmidi_poll_descriptors_count(input);
Packit Service a9274b
		pfds = alloca(npfds * sizeof(struct pollfd));
Packit Service a9274b
Packit Service a9274b
		if (timeout > 0) {
Packit Service a9274b
			pfds[0].fd = timerfd_create(CLOCK_MONOTONIC, 0);
Packit Service a9274b
			if (pfds[0].fd == -1) {
Packit Service a9274b
				error("cannot create timer: %s", strerror(errno));
Packit Service a9274b
				goto _exit;
Packit Service a9274b
			}
Packit Service a9274b
			pfds[0].events = POLLIN;
Packit Service a9274b
		} else {
Packit Service a9274b
			pfds[0].fd = -1;
Packit Service a9274b
		}
Packit Service a9274b
Packit Service a9274b
		snd_rawmidi_poll_descriptors(input, &pfds[1], npfds - 1);
Packit Service a9274b
Packit Service a9274b
		signal(SIGINT, sig_handler);
Packit Service a9274b
Packit Service a9274b
		if (timeout > 0) {
Packit Service a9274b
			float timeout_int;
Packit Service a9274b
Packit Service a9274b
			itimerspec.it_value.tv_nsec = modff(timeout, &timeout_int) * NSEC_PER_SEC;
Packit Service a9274b
			itimerspec.it_value.tv_sec = timeout_int;
Packit Service a9274b
			err = timerfd_settime(pfds[0].fd, 0, &itimerspec, NULL);
Packit Service a9274b
			if (err < 0) {
Packit Service a9274b
				error("cannot set timer: %s", strerror(errno));
Packit Service a9274b
				goto _exit;
Packit Service a9274b
			}
Packit Service a9274b
		}
Packit Service a9274b
		for (;;) {
Packit Service a9274b
			unsigned char buf[256];
Packit Service a9274b
			int i, length;
Packit Service a9274b
			unsigned short revents;
Packit Service a9274b
Packit Service a9274b
			err = poll(pfds, npfds, -1);
Packit Service a9274b
			if (stop || (err < 0 && errno == EINTR))
Packit Service a9274b
				break;
Packit Service a9274b
			if (err < 0) {
Packit Service a9274b
				error("poll failed: %s", strerror(errno));
Packit Service a9274b
				break;
Packit Service a9274b
			}
Packit Service a9274b
Packit Service a9274b
			err = snd_rawmidi_poll_descriptors_revents(input, &pfds[1], npfds - 1, &revents);
Packit Service a9274b
			if (err < 0) {
Packit Service a9274b
				error("cannot get poll events: %s", snd_strerror(errno));
Packit Service a9274b
				break;
Packit Service a9274b
			}
Packit Service a9274b
			if (revents & (POLLERR | POLLHUP))
Packit Service a9274b
				break;
Packit Service a9274b
			if (!(revents & POLLIN)) {
Packit Service a9274b
				if (pfds[0].revents & POLLIN)
Packit Service a9274b
					break;
Packit Service a9274b
				continue;
Packit Service a9274b
			}
Packit Service a9274b
Packit Service a9274b
			err = snd_rawmidi_read(input, buf, sizeof(buf));
Packit Service a9274b
			if (err == -EAGAIN)
Packit Service a9274b
				continue;
Packit Service a9274b
			if (err < 0) {
Packit Service a9274b
				error("cannot read from port \"%s\": %s", port_name, snd_strerror(err));
Packit Service a9274b
				break;
Packit Service a9274b
			}
Packit Service a9274b
			length = 0;
Packit Service a9274b
			for (i = 0; i < err; ++i)
Packit Service a9274b
				if ((buf[i] != MIDI_CMD_COMMON_CLOCK &&
Packit Service a9274b
				     buf[i] != MIDI_CMD_COMMON_SENSING) ||
Packit Service a9274b
				    (buf[i] == MIDI_CMD_COMMON_CLOCK   && !ignore_clock) ||
Packit Service a9274b
				    (buf[i] == MIDI_CMD_COMMON_SENSING && !ignore_active_sensing))
Packit Service a9274b
					buf[length++] = buf[i];
Packit Service a9274b
			if (length == 0)
Packit Service a9274b
				continue;
Packit Service a9274b
			read += length;
Packit Service a9274b
Packit Service a9274b
			if (receive_file != -1)
Packit Service a9274b
				write(receive_file, buf, length);
Packit Service a9274b
			if (dump) {
Packit Service a9274b
				for (i = 0; i < length; ++i)
Packit Service a9274b
					print_byte(buf[i]);
Packit Service a9274b
				fflush(stdout);
Packit Service a9274b
			}
Packit Service a9274b
Packit Service a9274b
			if (timeout > 0) {
Packit Service a9274b
				err = timerfd_settime(pfds[0].fd, 0, &itimerspec, NULL);
Packit Service a9274b
				if (err < 0) {
Packit Service a9274b
					error("cannot set timer: %s", strerror(errno));
Packit Service a9274b
					break;
Packit Service a9274b
				}
Packit Service a9274b
			}
Packit Service a9274b
		}
Packit Service a9274b
		if (isatty(fileno(stdout)))
Packit Service a9274b
			printf("\n%d bytes read\n", read);
Packit Service a9274b
	}
Packit Service a9274b
Packit Service a9274b
	ok = 1;
Packit Service a9274b
_exit:
Packit Service a9274b
	if (inputp)
Packit Service a9274b
		snd_rawmidi_close(input);
Packit Service a9274b
	if (outputp)
Packit Service a9274b
		snd_rawmidi_close(output);
Packit Service a9274b
_exit2:
Packit Service a9274b
	if (receive_file != -1)
Packit Service a9274b
		close(receive_file);
Packit Service a9274b
	return !ok;
Packit Service a9274b
}