598 lines
15 KiB
C
598 lines
15 KiB
C
/*
|
|
* Copyright (c) 2009 Tomasz Moń <desowin@gmail.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; under version 3 of the License.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses>.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <gtk/gtk.h>
|
|
#include <getopt.h>
|
|
#include <alsa/asoundlib.h>
|
|
#include <alloca.h>
|
|
#include "gdigi.h"
|
|
#include "gui.h"
|
|
|
|
static unsigned char device_id = 0x7F;
|
|
static unsigned char family_id = 0x7F;
|
|
static unsigned char product_id = 0x7F;
|
|
|
|
static snd_rawmidi_t *output = NULL;
|
|
static snd_rawmidi_t *input = NULL;
|
|
static char *device = "hw:1,0,0";
|
|
|
|
/**
|
|
* calculate_checksum:
|
|
* @array: data to calculate checksum
|
|
* @length: data length
|
|
*
|
|
* Calculates message checksum
|
|
*
|
|
* Return value: calculated checksum
|
|
**/
|
|
static char calculate_checksum(gchar *array, gint length)
|
|
{
|
|
int x;
|
|
int checksum = 0;
|
|
|
|
for (x = 0; x<length; x++) {
|
|
checksum ^= array[x];
|
|
}
|
|
|
|
return checksum;
|
|
}
|
|
|
|
/**
|
|
* open_device:
|
|
*
|
|
* Opens MIDI device. This function modifies global input and output variables.
|
|
*
|
|
* Return value: FALSE on success, TRUE on error.
|
|
**/
|
|
gboolean open_device()
|
|
{
|
|
int err;
|
|
|
|
err = snd_rawmidi_open(&input, &output, device, SND_RAWMIDI_NONBLOCK);
|
|
if (err) {
|
|
fprintf(stderr, "snd_rawmidi_open %s failed: %d\n", device, err);
|
|
return TRUE;
|
|
}
|
|
|
|
err = snd_rawmidi_nonblock(output, 0);
|
|
if (err) {
|
|
fprintf(stderr, "snd_rawmidi_nonblock failed: %d\n", err);
|
|
return TRUE;
|
|
}
|
|
|
|
snd_rawmidi_read(input, NULL, 0); /* trigger reading */
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/**
|
|
* send_data:
|
|
* @data: data to be sent
|
|
* @length: data length
|
|
*
|
|
* Sends data to device. This function uses global output variable.
|
|
**/
|
|
void send_data(char *data, int length)
|
|
{
|
|
snd_rawmidi_write(output, data, length);
|
|
}
|
|
|
|
/**
|
|
* pack_data:
|
|
* @data: data to be packed
|
|
* @len: data length
|
|
*
|
|
* Packs data using method used on all newer DigiTech products.
|
|
*
|
|
* Return value: GString containing packed data
|
|
**/
|
|
GString *pack_data(gchar *data, gint len)
|
|
{
|
|
GString *packed;
|
|
gint i;
|
|
gint new_len;
|
|
unsigned char status;
|
|
gint offset;
|
|
gint status_byte;
|
|
|
|
new_len = len + (len/7);
|
|
packed = g_string_sized_new(new_len);
|
|
status = 0;
|
|
offset = -1;
|
|
status_byte = 0;
|
|
|
|
for (i=0; i<len; i++) {
|
|
if ((i % 7) == 0) {
|
|
packed->str[status_byte] = status;
|
|
status = 0;
|
|
status_byte = packed->len;
|
|
g_string_append_c(packed, '\0');
|
|
}
|
|
g_string_append_c(packed, (data[i] & 0x7F));
|
|
status |= (data[i] & 0x80) >> ((i%7) + 1);
|
|
}
|
|
packed->str[status_byte] = status;
|
|
|
|
return packed;
|
|
}
|
|
|
|
/**
|
|
* unpack_message:
|
|
* @msg: message to unpack
|
|
*
|
|
* Unpacks message data. This function modifies given GString.
|
|
**/
|
|
static void unpack_message(GString *msg)
|
|
{
|
|
int offset;
|
|
int x;
|
|
int i;
|
|
unsigned char status;
|
|
unsigned char *str;
|
|
gboolean stop = FALSE;
|
|
|
|
g_return_if_fail(msg != NULL);
|
|
g_return_if_fail(msg->len > 9);
|
|
|
|
offset = 1;
|
|
x = 0;
|
|
i = 8;
|
|
|
|
str = (unsigned char*)msg->str;
|
|
do {
|
|
offset += 8;
|
|
status = str[offset-1];
|
|
for (x=0; x<7; x++) {
|
|
if (offset+x >= msg->len) {
|
|
stop = TRUE;
|
|
break;
|
|
}
|
|
if (str[offset+x] == 0xF7) {
|
|
str[i] = 0xF7;
|
|
stop = TRUE;
|
|
}
|
|
|
|
str[i] = (((status << (x+1)) & 0x80) | str[x+offset]);
|
|
i++;
|
|
}
|
|
} while (!stop && (offset+x < msg->len));
|
|
|
|
g_string_truncate(msg, i);
|
|
}
|
|
|
|
/**
|
|
* read_data:
|
|
*
|
|
* Reads data from MIDI IN. This function uses global input variable.
|
|
*
|
|
* Return value: GString containing data, or NULL when no data was read.
|
|
**/
|
|
GString* read_data()
|
|
{
|
|
/* This is mostly taken straight from alsa-utils-1.0.19 amidi/amidi.c
|
|
by Clemens Ladisch <clemens@ladisch.de> */
|
|
int err;
|
|
int npfds;
|
|
gboolean stop = FALSE;
|
|
struct pollfd *pfds;
|
|
GString *string = NULL;
|
|
|
|
npfds = snd_rawmidi_poll_descriptors_count(input);
|
|
pfds = alloca(npfds * sizeof(struct pollfd));
|
|
snd_rawmidi_poll_descriptors(input, pfds, npfds);
|
|
|
|
do {
|
|
char buf[20];
|
|
int i, length;
|
|
unsigned short revents;
|
|
|
|
err = poll(pfds, npfds, 200);
|
|
if (err < 0 && errno == EINTR)
|
|
break;
|
|
if (err < 0) {
|
|
g_error("poll failed: %s", strerror(errno));
|
|
break;
|
|
}
|
|
if ((err = snd_rawmidi_poll_descriptors_revents(input, pfds, npfds, &revents)) < 0) {
|
|
g_error("cannot get poll events: %s", snd_strerror(errno));
|
|
break;
|
|
}
|
|
if (revents & (POLLERR | POLLHUP))
|
|
break;
|
|
if (!(revents & POLLIN))
|
|
continue;
|
|
|
|
err = snd_rawmidi_read(input, buf, sizeof(buf));
|
|
if (err == -EAGAIN)
|
|
continue;
|
|
if (err < 0) {
|
|
g_error("cannot read: %s", snd_strerror(err));
|
|
break;
|
|
}
|
|
|
|
length = 0;
|
|
for (i = 0; i < err; ++i)
|
|
if (buf[i] != 0xFE) /* ignore active sensing */
|
|
buf[length++] = buf[i];
|
|
|
|
if ((unsigned char)buf[length-1] == 0xF7)
|
|
stop = TRUE;
|
|
|
|
if (length != 0) {
|
|
if (string == NULL) {
|
|
string = g_string_new_len(buf, length);
|
|
} else {
|
|
string = g_string_append_len(string, buf, length);
|
|
}
|
|
}
|
|
} while ((err != 0) && (stop == FALSE));
|
|
|
|
return string;
|
|
}
|
|
|
|
/**
|
|
* send_message:
|
|
* @procedure: procedure ID
|
|
* @data: unpacked message data
|
|
* @len: data length
|
|
*
|
|
* Creates SysEx message then sends it. This function uses folowing global variables: device_id, family_id and product_id.
|
|
**/
|
|
void send_message(gint procedure, gchar *data, gint len)
|
|
{
|
|
GString *msg = g_string_new_len("\xF0" /* SysEx status byte */
|
|
"\x00\x00\x10", /* Manufacturer ID */
|
|
4);
|
|
g_string_append_printf(msg,
|
|
"%c%c%c" /* device, family, product ID */
|
|
"%c", /* procedure */
|
|
device_id, family_id, product_id,
|
|
procedure);
|
|
|
|
if (len > 0) {
|
|
GString *tmp = pack_data(data, len);
|
|
g_string_append_len(msg, tmp->str, tmp->len);
|
|
g_string_free(tmp, TRUE);
|
|
}
|
|
|
|
g_string_append_printf(msg, "%c\xF7",
|
|
calculate_checksum(&msg->str[1], msg->len - 1));
|
|
|
|
send_data(msg->str, msg->len);
|
|
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* get_message_id:
|
|
* @msg: SysEx message
|
|
*
|
|
* Checks message ID.
|
|
*
|
|
* Return value: MessageID, or -1 on error.
|
|
**/
|
|
static MessageID get_message_id(GString *msg)
|
|
{
|
|
/* TODO: sanity checks */
|
|
g_return_val_if_fail(msg != NULL, -1);
|
|
|
|
if (msg->len > 7) {
|
|
return (unsigned char)msg->str[7];
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* get_message_by_id:
|
|
* @id: MessageID of requested message
|
|
*
|
|
* Reads data from MIDI IN until message with matching id is found.
|
|
*
|
|
* Return value: GString containing unpacked message.
|
|
**/
|
|
GString *get_message_by_id(MessageID id)
|
|
{
|
|
GString *data = NULL;
|
|
|
|
do {
|
|
if (data)
|
|
g_string_free(data, TRUE);
|
|
data = read_data();
|
|
} while (get_message_id(data) != id);
|
|
|
|
unpack_message(data);
|
|
|
|
return data;
|
|
}
|
|
|
|
/**
|
|
* append_value:
|
|
* @msg: message to append value
|
|
* @value: value to append
|
|
*
|
|
* Packs value using scheme used on all newer DigiTech products.
|
|
**/
|
|
void append_value(GString *msg, guint value)
|
|
{
|
|
/* check how many bytes long the value is */
|
|
guint temp = value;
|
|
gint n = 0;
|
|
do {
|
|
n++;
|
|
temp = temp >> 8;
|
|
} while (temp);
|
|
|
|
if (n == 1) {
|
|
if (value & 0x80)
|
|
n = 2;
|
|
else
|
|
g_string_append_printf(msg, "%c", value);
|
|
}
|
|
|
|
if (n > 1) {
|
|
gint x;
|
|
g_string_append_printf(msg, "%c", (n | 0x80));
|
|
for (x=0; x<n; x++) {
|
|
g_string_append_printf(msg, "%c",
|
|
((value >> (8*(n-x-1))) & 0xFF));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* set_option:
|
|
* @id: Parameter ID
|
|
* @position: Parameter position
|
|
* @value: Parameter value
|
|
*
|
|
* Forms SysEx message to set parameter then sends it to device.
|
|
**/
|
|
void set_option(guint id, guint position, guint value)
|
|
{
|
|
GString *msg = g_string_sized_new(9);
|
|
g_string_append_printf(msg, "%c%c%c",
|
|
((id & 0xFF00) >> 8), (id & 0xFF),
|
|
position);
|
|
append_value(msg, value);
|
|
send_message(RECEIVE_PARAMETER_VALUE, msg->str, msg->len);
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* switch_preset:
|
|
* @bank: preset bank
|
|
* @x: preset index
|
|
*
|
|
* Switches to selected preset.
|
|
**/
|
|
void switch_preset(guint bank, guint x)
|
|
{
|
|
GString *msg = g_string_sized_new(6);
|
|
g_string_append_printf(msg, "%c%c%c%c%c%c",
|
|
bank, x, /* source */
|
|
PRESETS_EDIT_BUFFER, 0, /* destination */
|
|
0, /* keep existing name */
|
|
1); /* load */
|
|
send_message(MOVE_PRESET, msg->str, msg->len);
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* store_preset_name:
|
|
* @x: preset index
|
|
* @name: preset name
|
|
*
|
|
* Stores current edit buffer in user presets bank.
|
|
**/
|
|
void store_preset_name(int x, const gchar *name)
|
|
{
|
|
GString *msg = g_string_sized_new(6);
|
|
g_string_append_printf(msg, "%c%c%c%c%s%c%c",
|
|
PRESETS_EDIT_BUFFER, 0, /* source */
|
|
PRESETS_USER, x, /* destination */
|
|
name, 0, /* name */
|
|
1); /* load */
|
|
send_message(MOVE_PRESET, msg->str, msg->len);
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* set_preset_name:
|
|
* @x: preset index
|
|
* @name: preset name
|
|
*
|
|
* Sets preset name.
|
|
**/
|
|
void set_preset_name(int x, gchar *name)
|
|
{
|
|
GString *msg = g_string_sized_new(12);
|
|
g_string_append_printf(msg, "%c%c%s%c",
|
|
PRESETS_USER, /* preset bank */
|
|
x, /* preset index */
|
|
name, 0); /* name */
|
|
send_message(RECEIVE_PRESET_NAME, msg->str, msg->len);
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* query_preset_names:
|
|
* @bank: preset bank
|
|
*
|
|
* Queries preset names.
|
|
*
|
|
* Return value: GStrv which must be freed with g_strfreev, or NULL on error.
|
|
**/
|
|
GStrv query_preset_names(gchar bank)
|
|
{
|
|
GString *data = NULL;
|
|
int x; /* used to iterate over whole reply */
|
|
int n = 0; /* current preset number */
|
|
int n_total; /* total number of presets */
|
|
gchar **str_array = NULL;
|
|
|
|
/* query user preset names */
|
|
send_message(REQUEST_PRESET_NAMES, &bank, 1);
|
|
|
|
/* read reply */
|
|
data = get_message_by_id(RECEIVE_PRESET_NAMES);
|
|
|
|
if (data != NULL) {
|
|
if (data->len >= 10) {
|
|
n_total = data->str[9];
|
|
str_array = g_new(gchar*, n_total + 1);
|
|
str_array[n_total] = NULL;
|
|
}
|
|
|
|
for (x=10; ((x<data->len) && (n<n_total)); x++) {
|
|
if (data->str[x] == 0xF7) /* every message ends with 0xF7 */
|
|
break;
|
|
|
|
str_array[n] = g_strdup(&data->str[x]);
|
|
x += strlen(str_array[n]);
|
|
n++;
|
|
}
|
|
g_string_free(data, TRUE);
|
|
}
|
|
return str_array;
|
|
}
|
|
|
|
/**
|
|
* get_current_preset:
|
|
*
|
|
* Queries current edit buffer.
|
|
*
|
|
* Return value: GString containing RECEIVE_PRESET_PARAMETERS SysEx message.
|
|
**/
|
|
GString *get_current_preset()
|
|
{
|
|
GString *data = NULL;
|
|
|
|
send_message(REQUEST_PRESET, "\x04\x00", 3);
|
|
|
|
/* read reply */
|
|
data = get_message_by_id(RECEIVE_PRESET_PARAMETERS);
|
|
|
|
return data;
|
|
}
|
|
|
|
/**
|
|
* request_who_am_i:
|
|
* @device_id: Variable to hold device ID
|
|
* @family_id: Variable to hold family ID
|
|
* @product_id: Variable to hold product ID
|
|
*
|
|
* Requests device information.
|
|
*
|
|
* Return value: TRUE on success, FALSE on error.
|
|
**/
|
|
static gboolean request_who_am_i(unsigned char *device_id, unsigned char *family_id,
|
|
unsigned char *product_id)
|
|
{
|
|
send_message(REQUEST_WHO_AM_I, NULL, 0);
|
|
|
|
GString *data = read_data();
|
|
if (data != NULL) {
|
|
if ((data->len == 15) && (data->str[7] == RECEIVE_WHO_AM_I)) {
|
|
*device_id = data->str[9];
|
|
*family_id = data->str[10];
|
|
*product_id = data->str[11];
|
|
g_string_free(data, TRUE);
|
|
return TRUE;
|
|
}
|
|
g_string_free(data, TRUE);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void request_device_configuration()
|
|
{
|
|
gint os_major, os_minor;
|
|
gint cpu_major, cpu_minor;
|
|
gint protocol_version;
|
|
gint current_bank, current_preset;
|
|
gint media_card;
|
|
|
|
send_message(REQUEST_DEVICE_CONFIGURATION, NULL, 0);
|
|
|
|
GString *data = get_message_by_id(RECEIVE_DEVICE_CONFIGURATION);
|
|
|
|
if (data->len > 14) {
|
|
os_major = data->str[8];
|
|
os_minor = (((data->str[9] & 0xF0) >> 4) * 10) |
|
|
(data->str[9] & 0x0F);
|
|
g_message("OS version: %d.%d", os_major, os_minor);
|
|
|
|
cpu_major = data->str[10];
|
|
cpu_minor = data->str[11];
|
|
g_message("CPU version: %d.%d", cpu_major, cpu_minor);
|
|
|
|
protocol_version = data->str[12];
|
|
g_message("Protocol version: %d", protocol_version);
|
|
|
|
current_bank = data->str[13];
|
|
current_preset = data->str[14];
|
|
g_message("Active bank: %d", current_bank);
|
|
g_message("Active preset: %d", current_preset);
|
|
|
|
if (os_major >= 1) {
|
|
media_card = data->str[15];
|
|
g_message("Media card present: %d", media_card);
|
|
}
|
|
}
|
|
}
|
|
|
|
static GOptionEntry options[] = {
|
|
{"device", 'd', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING, &device, "MIDI device port to use", NULL},
|
|
{NULL}
|
|
};
|
|
|
|
int main(int argc, char *argv[]) {
|
|
GError *error = NULL;
|
|
GOptionContext *context;
|
|
context = g_option_context_new(NULL);
|
|
g_option_context_add_main_entries(context, options, NULL);
|
|
g_option_context_add_group(context, gtk_get_option_group(TRUE));
|
|
|
|
if (!g_option_context_parse(context, &argc, &argv, &error)) {
|
|
g_message("option parsing failed: %s\n", error->message);
|
|
g_error_free(error);
|
|
g_option_context_free(context);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
g_option_context_free(context);
|
|
|
|
if (open_device() == TRUE) {
|
|
show_error_message(NULL, "Failed to open MIDI device");
|
|
} else {
|
|
if (request_who_am_i(&device_id, &family_id, &product_id) == FALSE) {
|
|
show_error_message(NULL, "No suitable reply from device - is it connected?");
|
|
} else {
|
|
gui_create();
|
|
gtk_main();
|
|
gui_free();
|
|
}
|
|
}
|
|
|
|
if (output != NULL)
|
|
snd_rawmidi_close(output);
|
|
if (input != NULL)
|
|
snd_rawmidi_close(input);
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|