mirror of
				https://github.com/alsa-project/alsa-tools.git
				synced 2025-10-29 05:40:25 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			1740 lines
		
	
	
	
		
			40 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1740 lines
		
	
	
	
		
			40 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  EMU10k1 loader
 | |
|  *
 | |
|  *  Copyright (c) 2003,2004 by Peter Zubaj
 | |
|  *
 | |
|  *   This program is free software; you can redistribute it and/or modify
 | |
|  *   it under the terms of the GNU General Public License as published by
 | |
|  *   the Free Software Foundation; either version 2 of the License, or
 | |
|  *   (at your option) any later version.
 | |
|  *
 | |
|  *   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, write to the Free Software
 | |
|  *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include <getopt.h>
 | |
| #include <stdarg.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <ctype.h>
 | |
| #include <sys/stat.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #ifdef HAVE_CONFIG_H
 | |
| #include "config.h"
 | |
| #endif
 | |
| 
 | |
| #include <alsa/asoundlib.h>
 | |
| #include "version.h"
 | |
| #include "comm.h"
 | |
| #include "ld10k1_fnc.h"
 | |
| #include "ld10k1_error.h"
 | |
| #include "ld10k1_debug.h"
 | |
| 
 | |
| #include "liblo10k1.h"
 | |
| #include "liblo10k1ef.h"
 | |
| #include "liblo10k1lf.h"
 | |
| 
 | |
| char comm_pipe[256];
 | |
| liblo10k1_connection_t conn;
 | |
| 
 | |
| static void error(const char *fmt,...)
 | |
| {
 | |
| 	va_list va;
 | |
| 
 | |
| 	va_start(va, fmt);
 | |
| 	fprintf(stderr, "Error: ");
 | |
| 	vfprintf(stderr, fmt, va);
 | |
| 	fprintf(stderr, "\n");
 | |
| 	va_end(va);
 | |
| }
 | |
| 
 | |
| static void help(char *command)
 | |
| {
 | |
| 	fprintf(stderr,
 | |
| 		"Usage: %s [-options]\n"
 | |
| 		"\nAvailable options:\n"
 | |
| 		"  -h, --help           this help\n"
 | |
| 		"  -p, --pipe_name      connect to this, default = /tmp/.ld10k1_port\n"
 | |
| 		"  -l, --list           dump lkoaded patch\n"
 | |
| 		"  -i, --info           print some info\n"
 | |
| 		"  -s, --setup          setup DSP\n"
 | |
| 		"  -a, --add            load patch\n"
 | |
| 		"  -d, --del            unload patch\n"
 | |
| 		"  -q, --conadd         connect 2 patches\n"
 | |
| 		"  -w, --condel         delete connection\n"
 | |
| 		"      --debug          print debug information\n"
 | |
| 		"  -n, --defionames     define default in/out names for loaded patch\n"
 | |
| 		"      --ctrl           modify control parameters for loaded patch\n"
 | |
| 		"      --patch_name     load patch with this name\n"
 | |
| 		"      --where          insert patch before\n"
 | |
| 		"      --renam          rename patch, input, output, fx, patch input, patch output\n"
 | |
| 		"      --dump           dump DSP setup to file, can by loaded by dl10k1\n"
 | |
| 		"      --host           lo10k1 uses network socket instead of named socked (host,port)\n"
 | |
| 		"  -P, --path           include path\n"
 | |
| 		"      --store          store DSP setup\n"
 | |
| 		"      --restore        restore DSP setup\n"
 | |
| 		, command);
 | |
| }
 | |
| 
 | |
| typedef struct tag_path_info {
 | |
| 	char *path;
 | |
| 	struct tag_path_info *next;
 | |
| } path_t;
 | |
| 
 | |
| path_t *first_path;
 | |
| path_t *last_path;
 | |
| 
 | |
| static void add_path(char *path)
 | |
| {
 | |
| 	path_t *path_info = malloc(sizeof(path_t));
 | |
| 
 | |
| 	path_info->path = strdup(path);
 | |
| 	path_info->next = NULL;
 | |
| 
 | |
| 	if (last_path)
 | |
| 		last_path->next = path_info;
 | |
| 
 | |
| 	last_path = path_info;
 | |
| 
 | |
| 	if (!first_path)
 | |
| 		first_path = path_info;
 | |
| }
 | |
| 
 | |
| static void add_paths(char *paths)
 | |
| {
 | |
| 	char *str = strdup(paths);
 | |
| 	char *path = strtok(str, ":");
 | |
| 	
 | |
| 	while (path) {
 | |
| 		add_path(path);
 | |
| 
 | |
| 		path = strtok(NULL, ":");
 | |
| 	}
 | |
| 
 | |
| 	free (str);
 | |
| }
 | |
| 
 | |
| static void free_all_paths()
 | |
| {
 | |
| 	path_t *path_info = first_path;
 | |
| 	path_t *path_info_n = NULL;
 | |
| 
 | |
| 	while (path_info) {
 | |
| 		path_info_n = path_info->next;
 | |
| 		free(path_info);
 | |
| 		path_info = path_info_n;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static liblo10k1_emu_patch_t *try_patch(char *file_name)
 | |
| {
 | |
| 	int en;
 | |
| 	
 | |
| 	liblo10k1_emu_patch_t *p = NULL;
 | |
| 	if ((en = liblo10k1_emu_load_patch(file_name, &p)) < 0)
 | |
| 		return NULL;
 | |
| 
 | |
| 	return p;
 | |
| }
 | |
| 
 | |
| static liblo10k1_emu_patch_t *open_patch(char *file_name)
 | |
| {
 | |
| 	liblo10k1_emu_patch_t *patch;
 | |
| 	path_t *path_info = first_path;
 | |
| 
 | |
| 	patch = try_patch(file_name);
 | |
| 	
 | |
| 	if (patch)
 | |
| 		return patch;
 | |
| 
 | |
| 	while (path_info) {
 | |
| 		char path[256]; /* FIXME */
 | |
| 
 | |
| 		memset(path, 0, sizeof(path));
 | |
| 		snprintf(path, sizeof(path)-1, "%s/%s", 
 | |
| 			 path_info->path, file_name);
 | |
| 		
 | |
| 		patch = try_patch(path);
 | |
| 
 | |
| 		if (patch) {
 | |
| 			return patch;
 | |
| 		}
 | |
| 
 | |
| 		snprintf(path, sizeof(path)-1, "%s/%s.emu10k1", 
 | |
| 			 path_info->path, file_name);
 | |
| 
 | |
| 		patch = try_patch(path);
 | |
| 
 | |
| 		if (patch) {
 | |
| 			return patch;
 | |
| 		}
 | |
| 
 | |
| 		path_info = path_info->next;
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static int load_patch(char *file_name, liblo10k1_emu_patch_t **p)
 | |
| {
 | |
| 	liblo10k1_emu_patch_t *patch;
 | |
| 	
 | |
| 	if (!(patch = open_patch(file_name))) {
 | |
| 		error("unable to load patch %s", file_name);
 | |
| 		return 1;
 | |
| 	}
 | |
| 	
 | |
| 	*p = patch;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static char get_str(char **str, char *out, int maxlen, char *sep, int isnum)
 | |
| {
 | |
| 	char ch = **str;
 | |
| 	char *tmpsep;
 | |
| 	int len = 0;
 | |
| 	int found = 0;
 | |
| 	
 | |
| 	*out = '\0';
 | |
| 	
 | |
| 	if (ch == '\0')
 | |
| 		return ch;
 | |
| 	
 | |
| 	len = 0;
 | |
| 	while (**str && len < maxlen) {
 | |
| 		found = 0;
 | |
| 		ch = **str;
 | |
| 		for (tmpsep = sep; *tmpsep; tmpsep++) {
 | |
| 			if (ch == *tmpsep) {
 | |
| 				found = 1;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		if (found)
 | |
| 			break;
 | |
|  		if (isnum && !isdigit(ch))
 | |
| 			break;
 | |
| 		
 | |
| 		*out++ = *(*str)++;
 | |
| 			len++;
 | |
| 	}
 | |
| 	
 | |
| 	*out = '\0';
 | |
| 	return ch;
 | |
| }
 | |
| 
 | |
| static int transfer_patch(int udin, char *ctrl_opt, liblo10k1_emu_patch_t *ep, liblo10k1_dsp_patch_t **p)
 | |
| {
 | |
| 	int i, j, k;
 | |
| 	
 | |
| 	char ctrl_from_concate[16][32][MAX_NAME_LEN]; /* max 32 ctrl to 1 and max 16 of this*/
 | |
| 	char ctrl_to_concate[16][MAX_NAME_LEN]; 
 | |
| 	int ctrl_to_concate_count;
 | |
| 	int ctrl_from_count[16];
 | |
| 	
 | |
| 	char ctrl_visible[16][MAX_NAME_LEN];
 | |
| 	char ctrl_visible_max[16];
 | |
| 	unsigned char ctrl_visible_count;
 | |
| 
 | |
| 	char ctrl_translate[16][MAX_NAME_LEN];
 | |
| 	char ctrl_translate_type[16];
 | |
| 	unsigned char ctrl_translate_count;
 | |
| 	
 | |
| 	char ctrl_index[16][MAX_NAME_LEN];
 | |
| 	int ctrl_index_val[16];
 | |
| 	unsigned char ctrl_index_count;
 | |
| 
 | |
| 	char ctrl_values[16][MAX_NAME_LEN];
 | |
| 	int ctrl_values_val[16][32];
 | |
| 	unsigned char ctrl_values_cnt[16];
 | |
| 	unsigned char ctrl_values_count;
 | |
| 
 | |
| 	char *tmp_str;
 | |
| 	char *tmp_num;
 | |
| 	char tmp_num_str[20];
 | |
| 	
 | |
| 	char sep;
 | |
| 
 | |
| 	liblo10k1_ctl_transform_t *tctl;
 | |
| 	int ctl_idx;
 | |
| 	
 | |
| 	liblo10k1_dsp_patch_t *np = NULL;
 | |
| 	
 | |
| 
 | |
| 	ctrl_to_concate_count = 0;
 | |
| 	ctrl_visible_count = 0;
 | |
| 	ctrl_translate_count = 0;
 | |
| 	ctrl_index_count = 0;
 | |
| 	ctrl_values_count = 0;
 | |
| 	for (i = 0; i < 16; i++) {
 | |
| 		ctrl_from_count[i] = 0;
 | |
| 		ctrl_to_concate[i][0] = '\0';
 | |
| 		ctrl_visible[i][0] = '\0';
 | |
| 		ctrl_visible_max[i] = 1;
 | |
| 		ctrl_translate[i][0] = '\0';
 | |
| 		ctrl_translate_type[i] = 1;
 | |
| 		
 | |
| 		ctrl_index[i][0] = '\0';
 | |
| 		ctrl_index_val[i] = -1;
 | |
| 
 | |
| 		ctrl_values[i][0] = '\0';
 | |
| 		ctrl_values_cnt[i] = 0;
 | |
| 		for (j = 0; j < 32; j++) {
 | |
| 			ctrl_from_concate[i][j][0] = '\0';
 | |
| 			ctrl_values_val[i][j] = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* parse ctrl opt */
 | |
| 
 | |
| 	/* TODO - check for name boundary */
 | |
| 	while (ctrl_opt && *ctrl_opt) {
 | |
| 		switch (*ctrl_opt++) {
 | |
| 			case 'c':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (c) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1) {
 | |
| 					tmp_str = ctrl_from_concate[ctrl_to_concate_count][ctrl_from_count[ctrl_to_concate_count]];
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ",:", 0);
 | |
| 					if (strlen(ctrl_from_concate[ctrl_to_concate_count][ctrl_from_count[ctrl_to_concate_count]]) == 0) {
 | |
| 						error("wrong ctrl option format (c) - wrong source ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					ctrl_from_count[ctrl_to_concate_count]++;
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					if (sep != ',') {
 | |
| 						error("wrong ctrl option format (c) - wrong separator - waiting , %c", sep);
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_str = ctrl_to_concate[ctrl_to_concate_count];
 | |
| 				/* next is new ctrl name */
 | |
| 				sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ",", 0);
 | |
| 				if (strlen(ctrl_to_concate[ctrl_to_concate_count]) == 0) {
 | |
| 					error("wrong ctrl option format (c) - wrong target ctrl name");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				ctrl_to_concate_count++;
 | |
| 				break;
 | |
| 			case 'v':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (v) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = ctrl_visible[ctrl_visible_count];
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(ctrl_visible[ctrl_visible_count]) == 0) {
 | |
| 						error("wrong ctrl option format (v) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_num = tmp_num_str;
 | |
| 				/* next is new ctrl name */
 | |
| 				sep = get_str(&ctrl_opt, tmp_num, 10, ",", 1);
 | |
| 				if (strlen(tmp_num_str) == 0) {
 | |
| 					error("wrong ctrl option format (v) - wrong vcount count");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				ctrl_visible_max[ctrl_visible_count] = atoi(tmp_num_str);
 | |
| 				ctrl_visible_count++;
 | |
| 				break;
 | |
| 			case 't':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (t) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = ctrl_translate[ctrl_translate_count];
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(ctrl_translate[ctrl_translate_count]) == 0) {
 | |
| 						error("wrong ctrl option format (t) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_num = tmp_num_str;
 | |
| 				/* next is new ctrl translate */
 | |
| 				sep = get_str(&ctrl_opt, tmp_num, 10, ",", 1);
 | |
| 				if (strlen(tmp_num_str) == 0) {
 | |
| 					error("wrong ctrl option format (t) - wrong translation function num");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				ctrl_translate_type[ctrl_translate_count] = atoi(tmp_num_str);
 | |
| 				ctrl_translate_count++;
 | |
| 				break;
 | |
| 			case 'i':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (i) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = ctrl_index[ctrl_index_count];
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(ctrl_index[ctrl_index_count]) == 0) {
 | |
| 						error("wrong ctrl option format (i) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_num = tmp_num_str;
 | |
| 				/* next is new ctrl index */
 | |
| 				sep = get_str(&ctrl_opt, tmp_num, 10, ",", 1);
 | |
| 				if (strlen(tmp_num_str) == 0) {
 | |
| 					error("wrong ctrl option format (i) - wrong index num");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				ctrl_index_val[ctrl_index_count] = atoi(tmp_num_str);
 | |
| 				ctrl_index_count++;
 | |
| 				break;
 | |
| 			case 's':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (s) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = ctrl_values[ctrl_values_count];
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(ctrl_values[ctrl_values_count]) == 0) {
 | |
| 						error("wrong ctrl option format (s) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				/* next is new ctrl name */
 | |
| 				do {
 | |
| 					tmp_num = tmp_num_str;
 | |
| 
 | |
| 					sep = get_str(&ctrl_opt, tmp_num, 10, ",#", 1);
 | |
| 					if (strlen(tmp_num_str) == 0) {
 | |
| 						error("wrong ctrl option format (s) - wrong value");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					ctrl_values_val[ctrl_values_count][ctrl_values_cnt[ctrl_values_count]] = atoi(tmp_num_str);
 | |
| 					ctrl_values_cnt[ctrl_values_count]++;
 | |
| 					if (sep != '#')
 | |
| 						break;
 | |
| 					ctrl_opt++;
 | |
| 				} while (1);
 | |
| 				ctrl_values_count++;
 | |
| 				break;
 | |
| 			default:
 | |
| 				error("wrong ctrl option format - unknown subfunction");
 | |
| 				return 1;
 | |
| 		}
 | |
| 		if (*ctrl_opt) {
 | |
| 			if (*ctrl_opt != ',') {
 | |
| 				error("wrong ctrl option format - wrong separator beetwen subfunctions");
 | |
| 				return 1;
 | |
| 			} else
 | |
| 				*ctrl_opt++;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	tctl = (liblo10k1_ctl_transform_t *)malloc(sizeof(liblo10k1_ctl_transform_t) * ctrl_to_concate_count);
 | |
| 	memset(tctl, 0, sizeof(liblo10k1_ctl_transform_t) * ctrl_to_concate_count);
 | |
| 	
 | |
| 	for (i = 0; i < ctrl_to_concate_count; i++) {
 | |
| 		/* find all controls for this ctl */
 | |
| 		for (k = 0; k < ctrl_from_count[i]; k++) {
 | |
| 			for (j = 0; j < ep->ctl_count; j++) {
 | |
| 				if (strcmp(ctrl_from_concate[i][k], ep->ctls[j].ctl_name) == 0) {
 | |
| 					tctl[i].emu_ctls[tctl[i].emu_ctl_count++] = j;
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		strcpy(tctl[i].ctl_name, ctrl_to_concate[i]);
 | |
| 	}
 | |
| 	
 | |
| 	if (liblo10k1_emu_transform_patch(ep,  tctl, ctrl_to_concate_count, &np) < 0)
 | |
| 	{
 | |
| 		error("error on liblo10k1_emu_transform_patch");
 | |
| 		return 1;
 | |
| 	}
 | |
| 	
 | |
| 	free(tctl);
 | |
| 	
 | |
| 	for (i = 0; i < ctrl_visible_count; i++) {
 | |
| 		ctl_idx = liblo10k1_patch_find_ctl_by_name(np, ctrl_visible[i]);
 | |
| 		if (ctl_idx < 0)
 | |
| 			goto err;
 | |
| 		if (liblo10k1_patch_ctl_set_vcount(np, ctl_idx, ctrl_visible_max[i]) < 0)
 | |
| 			goto err;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < ctrl_translate_count; i++) {
 | |
| 		ctl_idx = liblo10k1_patch_find_ctl_by_name(np, ctrl_translate[i]);
 | |
| 		if (ctl_idx < 0)
 | |
| 			goto err;
 | |
| 		if (liblo10k1_patch_ctl_set_trans(np, ctl_idx, ctrl_translate_type[i]) < 0)
 | |
| 			goto err;
 | |
| 	}
 | |
| 	
 | |
| 	for (i = 0; i < ctrl_index_count; i++) {
 | |
| 		ctl_idx = liblo10k1_patch_find_ctl_by_name(np, ctrl_index[i]);
 | |
| 		if (ctl_idx < 0)
 | |
| 			goto err;
 | |
| 		if (liblo10k1_patch_ctl_set_index(np, ctl_idx, ctrl_index_val[i]) < 0)
 | |
| 			goto err;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < ctrl_values_count; i++) {
 | |
| 		ctl_idx = liblo10k1_patch_find_ctl_by_name(np, ctrl_values[i]);
 | |
| 		if (ctl_idx < 0)
 | |
| 			goto err;
 | |
| 		for (j = 0; j < ctrl_values_cnt[i]; j++) {
 | |
| 			if (liblo10k1_patch_ctl_set_value(np, ctl_idx, j, ctrl_values_val[i][j]) < 0)
 | |
| 				goto err;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	*p = np;
 | |
| 	return 0;
 | |
| err:
 | |
| 	if (np)
 | |
| 		liblo10k1_patch_free(np);
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int transfer_native_patch(liblo10k1_dsp_patch_t *p, char *ctrl_opt)
 | |
| {
 | |
| 	unsigned char ctrl_values_count;
 | |
| 
 | |
| 	char tmp_name_from_str[MAX_NAME_LEN];
 | |
| 	char tmp_name_to_str[MAX_NAME_LEN];
 | |
| 	char *tmp_str;
 | |
| 	char *tmp_num;
 | |
| 	char tmp_num_str[20];
 | |
| 	
 | |
| 	char sep;
 | |
| 
 | |
| 	int ctl_idx;
 | |
| 		
 | |
| 	/* parse ctrl opt */
 | |
| 
 | |
| 	/* TODO - check for name boundary */
 | |
| 	while (ctrl_opt && *ctrl_opt) {
 | |
| 		switch (*ctrl_opt++) {
 | |
| 			case 'r':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (r) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1) {
 | |
| 					tmp_str = tmp_name_from_str;
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ",:", 0);
 | |
| 					if (strlen(tmp_name_from_str) == 0) {
 | |
| 						error("wrong ctrl option format (r) - wrong source ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_str = tmp_name_to_str;
 | |
| 				/* next is new ctrl name */
 | |
| 				sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ",", 0);
 | |
| 				if (strlen(tmp_name_to_str) == 0) {
 | |
| 					error("wrong ctrl option format (r) - wrong target ctrl name");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				
 | |
| 				ctl_idx = liblo10k1_patch_find_ctl_by_name(p, tmp_name_from_str);
 | |
| 				if (ctl_idx < 0) {
 | |
| 					error("unknown ctrl name");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				strcpy(p->ctl[ctl_idx].name, tmp_name_to_str);
 | |
| 				break;
 | |
| 			case 'i':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (i) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = tmp_name_from_str;
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(tmp_name_from_str) == 0) {
 | |
| 						error("wrong ctrl option format (i) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 
 | |
| 				tmp_num = tmp_num_str;
 | |
| 				/* next is new ctrl index */
 | |
| 				sep = get_str(&ctrl_opt, tmp_num, 10, ",", 1);
 | |
| 				if (strlen(tmp_num_str) == 0) {
 | |
| 					error("wrong ctrl option format (i) - wrong index num");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				ctl_idx = liblo10k1_patch_find_ctl_by_name(p, tmp_name_from_str);
 | |
| 				if (ctl_idx < 0) {
 | |
| 					error("unknown ctrl name");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				if (liblo10k1_patch_ctl_set_index(p, ctl_idx, atoi(tmp_num_str)) < 0) {
 | |
| 					error("can not set ctrl index");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				break;
 | |
| 			case 's':
 | |
| 				if (*ctrl_opt++ != '-') {
 | |
| 					error("wrong ctrl option format (s) - waiting -");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				while (1)
 | |
| 				{
 | |
| 					tmp_str = tmp_name_from_str;
 | |
| 					sep = get_str(&ctrl_opt, tmp_str, MAX_NAME_LEN, ":", 0);
 | |
| 					if (strlen(tmp_name_from_str) == 0) {
 | |
| 						error("wrong ctrl option format (s) - wrong ctrl name");
 | |
| 						return 1;
 | |
| 					}
 | |
| 
 | |
| 					if (sep == ':') {
 | |
| 						ctrl_opt++;
 | |
| 						break;
 | |
| 					}
 | |
| 					ctrl_opt++;
 | |
| 				}
 | |
| 				ctl_idx = liblo10k1_patch_find_ctl_by_name(p, tmp_name_from_str);
 | |
| 				if (ctl_idx < 0){
 | |
| 					error("unknown ctrl name");
 | |
| 					return 1;
 | |
| 				}
 | |
| 				
 | |
| 				/* next is value */
 | |
| 				ctrl_values_count = 0;
 | |
| 				do {
 | |
| 					tmp_num = tmp_num_str;
 | |
| 
 | |
| 					sep = get_str(&ctrl_opt, tmp_num, 10, ",#", 1);
 | |
| 					if (strlen(tmp_num_str) == 0) {
 | |
| 						error("wrong ctrl option format (s) - wrong value");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					if (liblo10k1_patch_ctl_set_value(p, ctl_idx, ctrl_values_count, atoi(tmp_num_str)) < 0){
 | |
| 						error("can not set ctrl value");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					if (sep != '#')
 | |
| 						break;
 | |
| 					ctrl_opt++;
 | |
| 					ctrl_values_count++;
 | |
| 				} while (1);
 | |
| 				break;
 | |
| 			default:
 | |
| 				error("wrong ctrl option format - unknown subfunction");
 | |
| 				return 1;
 | |
| 		}
 | |
| 		if (*ctrl_opt) {
 | |
| 			if (*ctrl_opt != ',') {
 | |
| 				error("wrong ctrl option format - wrong separator beetwen subfunctions");
 | |
| 				return 1;
 | |
| 			} else
 | |
| 				*ctrl_opt++;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int list_patch(char *file_name)
 | |
| {
 | |
| 	int err, i, j;
 | |
| 	liblo10k1_emu_patch_t *p;
 | |
| 
 | |
| 	err = load_patch(file_name, &p);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
|  	/* and now print */
 | |
| 	printf("Patch name : %s\n", p->patch_name);
 | |
| 	printf("IN:\n");
 | |
| 	for (i = 0; i < p->in_count; i++)
 | |
| 		printf("%03d: %08x\n", i, p->ins[i]);
 | |
| 	printf("OUT:\n");
 | |
| 	for (i = 0; i < p->out_count; i++)
 | |
| 		printf("%03d: %08x\n", i, p->outs[i]);
 | |
| 	
 | |
| 	printf("DYN:\n");
 | |
| 	for (i = 0; i < p->dyn_count; i++)
 | |
| 		printf("%03d: %08x\n", i, p->dyns[i]);
 | |
| 	
 | |
| 	printf("STA:\n");
 | |
| 	for (i = 0; i < p->sta_count; i++)
 | |
| 		printf("%03d: %08x  %08x\n", i, p->stas[i].sc, p->stas[i].sc_val);
 | |
| 
 | |
| 	printf("CTRL:\n");
 | |
| 	for (i = 0; i < p->ctl_count; i++)
 | |
| 		printf("%03d: %08x   %08x  %08x  %08x   %s\n", i, p->ctls[i].ctl, p->ctls[i].ctl_val, p->ctls[i].ctl_val_min, p->ctls[i].ctl_val_max, p->ctls[i].ctl_name);
 | |
| 	
 | |
| 	printf("CON:\n");
 | |
| 	for (i = 0; i < p->con_count; i++)
 | |
| 		printf("%03d: %08x  %08x\n", i, p->cons[i].sc, p->cons[i].sc_val);
 | |
| 	
 | |
| 	printf("TRAM LOOKUP:\n");
 | |
| 	for (i = 0; i < p->tram_lookup_count; i++) {
 | |
| 		printf("%03d: %08x\n", i, p->tram_lookups[i].size);
 | |
| 		for (j = 0; j < p->tram_lookups[i].read_line_count; j++)
 | |
| 			printf("  %03d: %c  %03d  %08x  %08x\n", i, 'R', j, p->tram_lookups[i].read_lines[j].line,p->tram_lookups[i].read_lines[j].line_size);
 | |
| 		for (j = 0; j < p->tram_lookups[i].write_line_count; j++)
 | |
| 			printf("  %03d: %c  %03d  %08x  %08x\n", i, 'W', j, p->tram_lookups[i].write_lines[j].line,p->tram_lookups[i].write_lines[j].line_size);
 | |
| 	}
 | |
| 	
 | |
| 	printf("TRAM DELAY:\n");
 | |
| 	for (i = 0; i < p->tram_delay_count; i++) {
 | |
| 		printf("%03d: %08x\n", i, p->tram_delays[i].size);
 | |
| 		for (j = 0; j < p->tram_delays[i].read_line_count; j++)
 | |
| 			printf("  %03d: %c  %03d  %08x  %08x\n", i, 'R', j, p->tram_delays[i].read_lines[j].line,p->tram_delays[i].read_lines[j].line_size);
 | |
| 		for (j = 0; j < p->tram_delays[i].write_line_count; j++)
 | |
| 			printf("  %03d: %c  %03d  %08x  %08x\n", i, 'W', j, p->tram_delays[i].write_lines[j].line,p->tram_delays[i].write_lines[j].line_size);
 | |
| 	}
 | |
| 		
 | |
| 	printf("INSTR:\n");
 | |
| 	for (i = 0; i < p->instr_count; i++)
 | |
| 		printf("%03d: %08x  %08x  %08x  %08x  %08x\n", i, p->instrs[i].op, p->instrs[i].arg[0], p->instrs[i].arg[1], p->instrs[i].arg[2], p->instrs[i].arg[3]);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int add_patch(char *file_name, int udin, char *ctrl_opt, char *opt_patch_name, int where)
 | |
| {
 | |
| 	int err;
 | |
| 	liblo10k1_emu_patch_t *ep;
 | |
| 	liblo10k1_dsp_patch_t *p;
 | |
| 	
 | |
| 	err = load_patch(file_name, &ep);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	err = transfer_patch(udin, ctrl_opt, ep, &p);
 | |
| 	if (err) {
 | |
| 		error("unable to transfer patch");
 | |
| 		return err;
 | |
| 	}
 | |
| 	
 | |
| 	if (opt_patch_name) {
 | |
| 		strncpy(p->patch_name, opt_patch_name, MAX_NAME_LEN - 1);
 | |
| 		p->patch_name[MAX_NAME_LEN - 1] = '\0';
 | |
| 	}
 | |
| 		
 | |
| 	if ((err = liblo10k1_patch_load(&conn, p, where, NULL, NULL)) < 0) {
 | |
| 		error("unable to load patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int load_dsp_patch(char *file_name, char *ctrl_opt, char *opt_patch_name, int where)
 | |
| {
 | |
| 	int err;
 | |
| 	liblo10k1_dsp_patch_t *p;
 | |
| 	liblo10k1_file_info_t *fi;
 | |
| 	
 | |
| 	fi = NULL;
 | |
| 	
 | |
| 	if ((err = liblo10k1lf_load_dsp_patch(&p, file_name, &fi)) < 0) {
 | |
| 		error("unable to load dsp patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	err = transfer_native_patch(p, ctrl_opt);
 | |
| 	if (err)
 | |
| 		goto err;
 | |
| 	
 | |
| 	if (opt_patch_name) {
 | |
| 		strncpy(p->patch_name, opt_patch_name, MAX_NAME_LEN - 1);
 | |
| 		p->patch_name[MAX_NAME_LEN - 1] = '\0';
 | |
| 	}
 | |
| 		
 | |
| 	if ((err = liblo10k1_patch_load(&conn, p, where, NULL, NULL)) < 0) {
 | |
| 		error("unable to load dsp patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	liblo10k1lf_file_info_free(fi);
 | |
| 	liblo10k1_patch_free(p);
 | |
| 	return 0;
 | |
| err:
 | |
| 	if (fi)
 | |
| 		liblo10k1lf_file_info_free(fi);
 | |
| 	if (p)
 | |
| 		liblo10k1_patch_free(p);
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int save_dsp_patch(char *file_name, int pn)
 | |
| {
 | |
| 	int err;
 | |
|  	
 | |
| 	liblo10k1_dsp_patch_t *p;
 | |
| 	liblo10k1_file_info_t *fi;
 | |
| 	
 | |
| 	if (pn < 0) {
 | |
| 		error("wrong patch num");
 | |
| 		return 1;
 | |
| 	}
 | |
| 	
 | |
| 	fi = liblo10k1lf_file_info_alloc();
 | |
| 	if (!fi) {
 | |
| 		error("no mem");
 | |
| 		goto err;
 | |
| 	}
 | |
| 		
 | |
| 	if ((err = liblo10k1_patch_get(&conn, pn, &p)) < 0) {
 | |
| 		error("unable to get dsp patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	fi->creater = strdup("lo10k1 - emu10k1/emu10k2 effect loader for alsa");
 | |
| 	
 | |
| 	if ((err = liblo10k1lf_save_dsp_patch(p, file_name, fi)) < 0) {
 | |
| 		error("unable to save dsp patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	liblo10k1lf_file_info_free(fi);
 | |
| 	liblo10k1_patch_free(p);
 | |
| 	return 0;
 | |
| err:
 | |
| 	if (fi)
 | |
| 		liblo10k1lf_file_info_free(fi);
 | |
| 	if (p)
 | |
| 		liblo10k1_patch_free(p);
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| void debug_print(char *str)
 | |
| {
 | |
| 	printf("%s", str);
 | |
| }
 | |
| 
 | |
| static int debug(int deb)
 | |
| {
 | |
| 	int err;
 | |
| 	
 | |
| 	if ((err = liblo10k1_debug(&conn, deb, debug_print)) < 0) {
 | |
| 		error("unable to debug (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int del_patch(char *file_name)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	if ((err = liblo10k1_patch_unload(&conn, atoi(file_name))) < 0 ) {
 | |
| 		error("unable to del patch (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int setup_dsp()
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	if ((err = liblo10k1_dsp_init(&conn)) < 0) {
 | |
| 		error("unable to setup DSP (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 	
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int is_num(char *str)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < strlen(str); i++)
 | |
| 		if (!isdigit(str[i]))
 | |
| 			return 0;
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
| 	char type;
 | |
| 	int patch;
 | |
| 	int io;
 | |
| } conn_info_t;
 | |
| 
 | |
| char *parse_connect_sym(char *con_str, char *sym, int *len, int max_len)
 | |
| {
 | |
| 	*len = 0;
 | |
| 	while (*len < max_len && *con_str && *con_str != ',' && *con_str != ')') {
 | |
| 		*sym++ = *con_str++;
 | |
| 		(*len)++;
 | |
| 	}
 | |
| 
 | |
| 	*sym++ = '\0';
 | |
| 	if (*len == 0)
 | |
| 		return NULL;
 | |
| 	return con_str;
 | |
| }
 | |
| 
 | |
| char *parse_simple_params(char *con_str, char io_type, int pn, conn_info_t *con_info, int *con_info_count, int max_con_info_count)
 | |
| {
 | |
| 	char con_arg[255];
 | |
| 	int con_arg_len = 0;
 | |
| 	int io_idx;
 | |
| 
 | |
| 	while(1) {
 | |
| 		if (*con_info_count >= max_con_info_count)
 | |
| 			return NULL;
 | |
| 		if (!(con_str = parse_connect_sym(con_str, con_arg, &con_arg_len, sizeof(con_arg) - 1)))
 | |
| 			return NULL;
 | |
| 		if (*con_str != ')' && *con_str != ',')
 | |
| 			return NULL; /* wrong format */
 | |
| 
 | |
| 		con_info[*con_info_count].type = io_type;
 | |
| 
 | |
| 		if (is_num(con_arg)) {
 | |
| 			/* input number */
 | |
| 			if (io_type == 'A' || io_type == 'B') {
 | |
| 				con_info[*con_info_count].patch = pn;
 | |
| 				con_info[(*con_info_count)++].io = atoi(con_arg);
 | |
| 			} else {
 | |
| 				con_info[*con_info_count].patch = -1;
 | |
| 				con_info[(*con_info_count)++].io = atoi(con_arg);
 | |
| 			}
 | |
| 		} else {
 | |
| 			/* input name */
 | |
| 			switch (io_type) {
 | |
| 				case 'A':
 | |
| 					if (liblo10k1_find_patch_in(&conn, pn, con_arg, &io_idx) < 0)
 | |
| 						return NULL;
 | |
| 					con_info[*con_info_count].patch = pn;
 | |
| 					con_info[(*con_info_count)++].io = io_idx;
 | |
| 					break;
 | |
| 				case 'B':
 | |
| 					if (liblo10k1_find_patch_out(&conn, pn, con_arg, &io_idx) < 0)
 | |
| 						return NULL;
 | |
| 					con_info[*con_info_count].patch = pn;
 | |
| 					con_info[(*con_info_count)++].io = io_idx;
 | |
| 					break;
 | |
| 				case 'F':
 | |
| 					if (liblo10k1_find_fx(&conn, con_arg, &io_idx) < 0)
 | |
| 						return NULL;
 | |
| 					con_info[*con_info_count].patch = -1;
 | |
| 					con_info[(*con_info_count)++].io = io_idx;
 | |
| 					break;
 | |
| 				case 'I':
 | |
| 					if (liblo10k1_find_in(&conn, con_arg, &io_idx) < 0)
 | |
| 						return NULL;
 | |
| 					con_info[*con_info_count].patch = -1;
 | |
| 					con_info[(*con_info_count)++].io = io_idx;
 | |
| 					break;
 | |
| 				case 'O':
 | |
| 					if (liblo10k1_find_out(&conn, con_arg, &io_idx) < 0)
 | |
| 						return NULL;
 | |
| 					con_info[*con_info_count].patch = -1;
 | |
| 					con_info[(*con_info_count)++].io = io_idx;
 | |
| 					break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (*con_str != ',')
 | |
| 			break;
 | |
| 		con_str++;
 | |
| 	}
 | |
| 	return con_str;
 | |
| }
 | |
| 
 | |
| char *parse_patch_params(char *con_str, char io_type, conn_info_t *con_info, int *con_info_count, int max_con_info_count)
 | |
| {
 | |
| 	char con_arg[255];
 | |
| 	int con_arg_len = 0;
 | |
| 	int i;
 | |
| 	int patch_num = -1;
 | |
| 	int io_count = 0;
 | |
| 
 | |
| 	if (!(con_str = parse_connect_sym(con_str, con_arg, &con_arg_len, sizeof(con_arg) - 1)))
 | |
| 		return NULL;
 | |
| 	if (*con_str != ')' && *con_str != ',')
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (is_num(con_arg))
 | |
| 		/* patch number */
 | |
| 		patch_num = atoi(con_arg);
 | |
| 	else
 | |
| 		/* patch name - find patch */
 | |
| 		if (liblo10k1_find_patch(&conn, con_arg, &patch_num) < 0)
 | |
| 			return NULL;
 | |
| 
 | |
| 	/* argumenty */
 | |
| 	if (*con_str == ',') {
 | |
| 		con_str++;
 | |
| 		if (*con_info_count >= max_con_info_count)
 | |
| 			return NULL;
 | |
| 		if (!(con_str = parse_simple_params(con_str, io_type, patch_num, con_info, con_info_count, max_con_info_count)))
 | |
| 			return NULL;
 | |
| 	} else {
 | |
| 		/* add all patch inputs or outputs */
 | |
| 		if (io_type == 'A') {
 | |
| 			/* get all inputs */
 | |
| 			if (liblo10k1_get_pin_count(&conn, patch_num, &io_count) < 0)
 | |
| 				return NULL;
 | |
| 		} else {
 | |
| 			/* get all outputs */
 | |
| 			if (liblo10k1_get_pout_count(&conn, patch_num, &io_count) < 0)
 | |
| 				return NULL;
 | |
| 		}
 | |
| 
 | |
| 		i = 0;
 | |
| 		while (i < io_count) {
 | |
| 			if (*con_info_count >= max_con_info_count) {
 | |
| 				return NULL;
 | |
| 			}
 | |
| 			con_info[*con_info_count].type = io_type;
 | |
| 			con_info[*con_info_count].patch = patch_num;
 | |
| 			con_info[(*con_info_count)++].io = i;
 | |
| 			i++;
 | |
| 		}
 | |
| 	}
 | |
| 	return con_str;
 | |
| }
 | |
| 
 | |
| int parse_connect(int add, char *con_str, int *multi, int *simple, conn_info_t **con_info, int *con_info_count, int max_con_info_count)
 | |
| {
 | |
| 	char con[10];
 | |
| 	int con_len;
 | |
| 
 | |
| 	int ft = 0;
 | |
| 	while (1) {
 | |
| 		con_len = 0;
 | |
| 		for(;*con_str && *con_str != '('; con_str++) {
 | |
| 			if (con_len >= sizeof(con) - 1)
 | |
| 				return 1;/* ERROR */
 | |
| 			con[con_len++] = *con_str;
 | |
| 		}
 | |
| 		con[con_len++] = '\0';
 | |
| 
 | |
| 		if (*con_str != '(')
 | |
| 			return 1;/* ERROR */
 | |
| 		con_str++;
 | |
| 
 | |
| 		if (ft && strcmp(con,"FX") == 0) {
 | |
| 			if (!(con_str = parse_simple_params(con_str, 'F', -1, con_info[ft], &(con_info_count[ft]), max_con_info_count)))
 | |
| 				return 1;/* ERROR */
 | |
| 		} else if (ft && strcmp(con,"IN") == 0) {
 | |
| 			if (!(con_str = parse_simple_params(con_str, 'I',  -1, con_info[ft], &(con_info_count[ft]), max_con_info_count)))
 | |
| 				return 1;/* ERROR */
 | |
| 		} else if (ft && strcmp(con,"OUT") == 0) {
 | |
| 			if (!(con_str = parse_simple_params(con_str, 'O', -1, con_info[ft], &(con_info_count[ft]), max_con_info_count)))
 | |
| 				return 1;/* ERROR */
 | |
| 		} else if (strcmp(con,"PIN") == 0) {
 | |
| 			if (!(con_str = parse_patch_params(con_str, 'A', con_info[ft], &(con_info_count[ft]), max_con_info_count)))
 | |
| 				return 1;/* ERROR */
 | |
| 		} else if (strcmp(con,"POUT") == 0) {
 | |
| 			if (!(con_str = parse_patch_params(con_str, 'B', con_info[ft], &(con_info_count[ft]), max_con_info_count)))
 | |
| 				return 1;/* ERROR */
 | |
| 		} else
 | |
| 			return 1;/* ERROR */
 | |
| 
 | |
| 		con_str++;
 | |
| 
 | |
| 		if (ft && !*con_str)
 | |
| 			return 0; /* OK */
 | |
| 		if (!add) {
 | |
| 			if (!*con_str)
 | |
| 				return 0;
 | |
| 			else
 | |
| 				return 1;
 | |
| 		}
 | |
| 		
 | |
| 		if (add && !ft && (*con_str == '=' || *con_str == '>' || *con_str == ':')) {
 | |
| 			ft++;
 | |
| 			if (*con_str == '=') {
 | |
| 				*multi = 0;
 | |
| 				*simple = 0;
 | |
| 			} else if (*con_str == ':') {
 | |
| 				*multi = 0;
 | |
| 				*simple = 1;
 | |
| 			} else
 | |
| 				*multi = 1;
 | |
| 
 | |
| 		} else if (add &&*con_str != '+')
 | |
| 			return 1;/* ERROR */
 | |
| 		/* process next */
 | |
| 		con_str++;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int parse_rename(char *con_str, char *io_type, int *pn, int *io, char **new_name)
 | |
| {
 | |
| 	char con[10];
 | |
| 	int con_len = 0;
 | |
| 
 | |
| 	char con_arg1[255];
 | |
| 	int con_arg_len1 = 0;
 | |
| 	int is_arg_num1 = 0;
 | |
| 	int arg_num1 = -1;
 | |
| 
 | |
| 	char con_arg2[255];
 | |
| 	int con_arg_len2 = 0;
 | |
| 	int is_arg_num2 = 0;
 | |
| 	int arg_num2 = -1;
 | |
| 
 | |
| 	for(;*con_str && *con_str != '('; con_str++) {
 | |
| 		if (con_len >= sizeof(con) - 1)
 | |
| 			return 1;/* ERROR */
 | |
| 		con[con_len++] = *con_str;
 | |
| 	}
 | |
| 	con[con_len++] = '\0';
 | |
| 
 | |
| 	if (*con_str != '(')
 | |
| 		return 1;/* ERROR */
 | |
| 	con_str++;
 | |
| 
 | |
| 	*io_type = '\0';
 | |
| 	if (strcmp(con,"FX") == 0)
 | |
| 		*io_type = 'F';
 | |
| 	else if (strcmp(con,"IN") == 0)
 | |
| 		*io_type = 'I';
 | |
| 	else if (strcmp(con,"OUT") == 0)
 | |
| 		*io_type = 'O';
 | |
| 	else if (strcmp(con,"PIN") == 0)
 | |
| 		*io_type = 'A';
 | |
| 	else if (strcmp(con,"POUT") == 0)
 | |
| 		*io_type = 'B';
 | |
| 	else if (strcmp(con,"PATCH") == 0)
 | |
| 		*io_type = 'P';
 | |
| 	else
 | |
| 		return 1;/* ERROR */
 | |
| 
 | |
| 	if (!(con_str = parse_connect_sym(con_str, con_arg1, &con_arg_len1, sizeof(con_arg1) - 1)))
 | |
| 			return 1;
 | |
| 
 | |
| 	if ((is_arg_num1 = is_num(con_arg1)))
 | |
| 		arg_num1 = atoi(con_arg1);
 | |
| 
 | |
| 	if (*io_type == 'A' || *io_type == 'B') {
 | |
| 		/* two arguments */
 | |
| 		if (*con_str != ',')
 | |
| 			return 1; /* ERROR */
 | |
| 		con_str++;
 | |
| 		if (!(con_str = parse_connect_sym(con_str, con_arg2, &con_arg_len2, sizeof(con_arg2) - 1)))
 | |
| 			return 1;
 | |
| 
 | |
| 		if ((is_arg_num2 = is_num(con_arg2)))
 | |
| 			arg_num2 = atoi(con_arg2);
 | |
| 	}
 | |
| 
 | |
| 	if (*con_str != ')')
 | |
| 		return 1; /* ERROR */
 | |
| 
 | |
| 	switch (*io_type) {
 | |
| 		case 'A':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_patch(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			if (!is_arg_num2)
 | |
| 				if (liblo10k1_find_patch_in(&conn, arg_num1, con_arg2, &arg_num2) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 		case 'B':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_patch(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			if (!is_arg_num2)
 | |
| 				if (liblo10k1_find_patch_out(&conn, arg_num1, con_arg2, &arg_num2) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 		case 'F':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_fx(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 		case 'I':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_in(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 		case 'O':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_out(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 		case 'P':
 | |
| 			if (!is_arg_num1)
 | |
| 				if (liblo10k1_find_patch(&conn, con_arg1, &arg_num1) < 0)
 | |
| 					return 1;
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	con_str++;
 | |
| 	if (*con_str != '=')
 | |
| 		return 1; /* ERROR */
 | |
| 	con_str++;
 | |
| 
 | |
| 	if (*io_type == 'A' || *io_type == 'B' || *io_type == 'P') {
 | |
| 		*pn = arg_num1;
 | |
| 		*io = arg_num2;
 | |
| 	} else {
 | |
| 		*io = arg_num1;
 | |
| 	}
 | |
| 
 | |
| 	*new_name = con_str;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int con_add(char *file_name)
 | |
| {
 | |
| 	int err, i;
 | |
| 	int multi = 0;
 | |
| 	int simple = 0;
 | |
| 
 | |
| 	conn_info_t con_infof[32];
 | |
| 	conn_info_t con_infot[32];
 | |
| 	conn_info_t *con_info[2] = {con_infof, con_infot};
 | |
| 
 | |
| 	int con_info_count[2] = {0, 0};
 | |
| 
 | |
| 	if (parse_connect(1, file_name, &multi, &simple, con_info, con_info_count, 32)) {
 | |
| 		error("wrong parameter - connection string");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	if (con_info_count[0] != con_info_count[1]) {
 | |
| 		error("wrong parameter - connection string from <> to");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	if (!con_info_count[0]) {
 | |
| 		error("wrong parameter - connection string");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < con_info_count[0]; i++) {
 | |
| 		if ((err = liblo10k1_con_add(&conn, multi, simple,
 | |
| 			con_infof[i].type, con_infof[i].patch, con_infof[i].io,
 | |
| 			con_infot[i].type, con_infot[i].patch, con_infot[i].io,
 | |
| 			NULL)) < 0) {
 | |
| 			error("unable to connect  (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 			return err;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int con_del(char *file_name)
 | |
| {
 | |
| 	conn_info_t con_info[32];
 | |
| 	int err, i;
 | |
| 
 | |
| 	conn_info_t *con_info_p = con_info;
 | |
| 
 | |
| 	int con_info_count = 0;
 | |
| 
 | |
| 	if (parse_connect(0, file_name, NULL, NULL, &con_info_p, &con_info_count, 32)) {
 | |
| 		error("wrong parameter - disconnection string");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	if (!con_info_count) {
 | |
| 		error("wrong parameter - disconnection string");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < con_info_count; i++) {
 | |
| 		if ((err = liblo10k1_con_del(&conn, con_info[i].type, con_info[i].patch, con_info[i].io, NULL)) < 0) {
 | |
| 			error("unable to connect  (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 			return err;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rename_arg(char *arg_name)
 | |
| {
 | |
| 	char io_type = '\0';
 | |
| 	int pn = -1;
 | |
| 	int io = -1;
 | |
| 	char *new_name = NULL;
 | |
| 
 | |
| 	if (parse_rename(arg_name, &io_type, &pn, &io, &new_name)) {
 | |
| 		error("wrong parameter for rename");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	switch (io_type) {
 | |
| 		case 'A':
 | |
| 			if (liblo10k1_rename_patch_in(&conn, pn, io, new_name) < 0) {
 | |
| 				error("couldn't rename patch in");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'B':
 | |
| 			if (liblo10k1_rename_patch_out(&conn, pn, io, new_name) < 0) {
 | |
| 				error("couldn't rename patch out");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'F':
 | |
| 			if (liblo10k1_rename_fx(&conn, io, new_name) < 0) {
 | |
| 				error("couldn't rename fx");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'I':
 | |
| 			if (liblo10k1_rename_in(&conn, io, new_name) < 0) {
 | |
| 				error("couldn't rename in");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'O':
 | |
| 			if (liblo10k1_rename_out(&conn, io, new_name) < 0) {
 | |
| 				error("couldn't rename out");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'P':
 | |
| 			if (liblo10k1_rename_patch(&conn, pn, new_name) < 0) {
 | |
| 				error("couldn't rename patch");
 | |
| 				return 1;
 | |
| 			}
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int dump(char *file_name)
 | |
| {
 | |
| 	int err;
 | |
| 	void *dump = NULL;
 | |
| 	int size = 0;
 | |
| 
 | |
| 	FILE *dump_file = NULL;
 | |
| 
 | |
| 	if ((err = liblo10k1_dump(&conn, &dump, &size)) < 0 ) {
 | |
| 		error("unable to dump (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	dump_file = fopen(file_name, "w");
 | |
| 	if (!dump_file) {
 | |
| 		free(dump);
 | |
| 		error("unable to open dump");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	if (fwrite(dump, 1, size, dump_file) < size) {
 | |
| 		free(dump);
 | |
| 		error("unable to write dump");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	free(dump);
 | |
| 	fclose(dump_file);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int store_dsp(char *file_name)
 | |
| {
 | |
| 	int err;
 | |
|  	
 | |
| 	liblo10k1_file_dsp_setup_t *setup;
 | |
| 	liblo10k1_file_info_t *fi;
 | |
| 	
 | |
| 	fi = liblo10k1lf_file_info_alloc();
 | |
| 	if (!fi) {
 | |
| 		error("no mem");
 | |
| 		goto err;
 | |
| 	}
 | |
| 		
 | |
| 	if ((err = liblo10k1lf_get_dsp_config(&conn, &setup)) < 0) {
 | |
| 		error("unable to get dsp config (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	fi->creater = strdup("lo10k1 - emu10k1/emu10k2 effect loader for alsa");
 | |
| 	
 | |
| 	if ((err = liblo10k1lf_save_dsp_config(setup, file_name, fi)) < 0) {
 | |
| 		error("unable to store dsp config (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	liblo10k1lf_file_info_free(fi);
 | |
| 	liblo10k1lf_dsp_config_free(setup);
 | |
| 	return 0;
 | |
| err:
 | |
| 	if (fi)
 | |
| 		liblo10k1lf_file_info_free(fi);
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int restore_dsp(char *file_name)
 | |
| {
 | |
| 	int err;
 | |
|  	
 | |
| 	liblo10k1_file_dsp_setup_t *setup;
 | |
| 	liblo10k1_file_info_t *fi;
 | |
| 	
 | |
| 	fi = NULL;
 | |
| 	
 | |
| 	if ((err = liblo10k1lf_load_dsp_config(&setup, file_name, &fi)) < 0) {
 | |
| 		error("unable to restore dsp config (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	if ((err = liblo10k1lf_put_dsp_config(&conn, setup)) < 0) {
 | |
| 		error("unable to put dsp config (ld10k1 error:%s)", liblo10k1_error_str(err));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	
 | |
| 	liblo10k1lf_file_info_free(fi);
 | |
| 	liblo10k1lf_dsp_config_free(setup);
 | |
| 	return 0;
 | |
| err:
 | |
| 	if (fi)
 | |
| 		liblo10k1lf_file_info_free(fi);
 | |
| 	return 1;
 | |
| }
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
| 	int  c;
 | |
| 
 | |
| 	int opt_list;
 | |
| 	int opt_setup;
 | |
| 	int opt_info;
 | |
| 	int opt_add;
 | |
| 	int opt_del;
 | |
| 	int opt_con_add;
 | |
| 	int opt_con_del;
 | |
| 	int opt_debug;
 | |
| 	char *opt_list_patch;
 | |
| 	int opt_use_default_io_names;
 | |
| 	char *opt_ctrl;
 | |
| 	char *opt_patch_name;
 | |
| 	char *opt_new_name;
 | |
| 	int opt_where;
 | |
| 	int option_index = 0;
 | |
| 	char *opt_dump_name;
 | |
| 	char *opt_host;
 | |
| 	char *tmp = NULL;
 | |
| 	
 | |
| 	int opt_store;
 | |
| 	int opt_restore;
 | |
| 	char *opt_store_restore_file;
 | |
| 	
 | |
| 	int opt_load_patch;
 | |
| 	int opt_save_patch;
 | |
| 	
 | |
| 	unsigned int opt_wait_for_conn;
 | |
| 
 | |
| 	liblo10k1_param params;
 | |
| 
 | |
| 	int err = 0;
 | |
| 	
 | |
|  	static struct option long_options[] = {
 | |
| 				{"pipe_name", 1, 0, 'p'},
 | |
|     				{"list", 1, 0, 'l'},
 | |
| 				{"info", 0, 0, 'i'},
 | |
| 				{"add", 1, 0, 'a'},
 | |
| 				{"del", 1, 0, 'd'},
 | |
| 				{"conadd", 1, 0, 'q'},
 | |
| 				{"condel", 1, 0, 'w'},
 | |
| 				{"debug", 1, 0, 0},
 | |
| 				{"defionames", 0, 0, 'n'},
 | |
| 				{"ctrl", 1, 0, 0},
 | |
| 				{"patch_name", 1, 0, 0},
 | |
| 				{"where", 1, 0, 0},
 | |
| 				{"setup", 1, 0, 's'},
 | |
| 				{"renam", 1, 0, 0},
 | |
| 				{"dump", 1, 0, 0},
 | |
| 				{"host", 1, 0, 0},
 | |
| 				{"path", 1, 0, 'P'},
 | |
| 				{"store", 1, 0, 0},
 | |
| 				{"restore", 1, 0, 0},
 | |
| 				{"load_patch", 1, 0, 0},
 | |
| 				{"save_patch", 1, 0, 0},
 | |
| 				{"wait", 1, 0, 0},
 | |
| 				{0, 0, 0, 0}
 | |
| 	};
 | |
| 
 | |
| 	opt_list = 0;
 | |
| 	opt_add = 0;
 | |
| 	opt_del = 0;
 | |
| 	opt_list_patch = NULL;
 | |
| 	opt_info = 0;
 | |
| 	opt_con_add = 0;
 | |
| 	opt_con_del = 0;
 | |
| 	opt_debug = 0;
 | |
| 	opt_use_default_io_names = 0;
 | |
| 	opt_ctrl = NULL;
 | |
| 	opt_patch_name = NULL;
 | |
| 	opt_new_name = NULL;
 | |
| 	opt_where = -1;
 | |
| 	opt_setup = 0;
 | |
| 	opt_dump_name = NULL;
 | |
| 	opt_host = NULL;
 | |
| 	
 | |
| 	opt_store = 0;
 | |
| 	opt_restore = 0;
 | |
| 	opt_store_restore_file = NULL;
 | |
| 	
 | |
| 	opt_load_patch = 0;
 | |
| 	opt_save_patch = 0;
 | |
| 	
 | |
| 	opt_wait_for_conn = 500;
 | |
| 
 | |
| 	strcpy(comm_pipe,"/tmp/.ld10k1_port");
 | |
| 
 | |
| 	if (argc > 1 && !strcmp(argv[1], "--help")) {
 | |
| 		help(argv[0]);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	first_path = NULL;
 | |
| #ifdef EFFECTSDIR
 | |
| 	add_paths(EFFECTSDIR);
 | |
| #endif
 | |
| 
 | |
| 	while ((c = getopt_long(argc, argv, "hil:p:a:d:q:w:nsh:P:",
 | |
| 	        long_options, &option_index)) != EOF) {
 | |
| 		switch (c) {
 | |
| 		case 0:
 | |
| 			if (strcmp(long_options[option_index].name, "debug") == 0)
 | |
| 				opt_debug = atoi(optarg);
 | |
| 			else if (strcmp(long_options[option_index].name, "ctrl") == 0)
 | |
| 				opt_ctrl = optarg;
 | |
| 			else if (strcmp(long_options[option_index].name, "patch_name") == 0)
 | |
| 				opt_patch_name = optarg;
 | |
| 			else if (strcmp(long_options[option_index].name, "where") == 0)
 | |
| 				opt_where = atoi(optarg);
 | |
| 			else if (strcmp(long_options[option_index].name, "renam") == 0)
 | |
| 				opt_new_name = optarg;
 | |
| 			else if (strcmp(long_options[option_index].name, "dump") == 0)
 | |
| 				opt_dump_name = optarg;
 | |
| 			else if (strcmp(long_options[option_index].name, "host") == 0)
 | |
| 				opt_host = optarg;
 | |
| 			else if (strcmp(long_options[option_index].name, "wait") == 0) {
 | |
| 				opt_wait_for_conn = atoi(optarg);
 | |
| 				if (opt_wait_for_conn < 0)
 | |
| 					opt_wait_for_conn = 0;
 | |
| 				else if (opt_wait_for_conn > 500)
 | |
| 					opt_wait_for_conn = 500;
 | |
| 			}
 | |
| 			else if (strcmp(long_options[option_index].name, "store") == 0) {
 | |
| 				opt_store = 1;
 | |
| 				opt_store_restore_file = optarg;
 | |
| 			} else if (strcmp(long_options[option_index].name, "restore") == 0) {
 | |
| 				opt_restore = 1;
 | |
| 				opt_store_restore_file = optarg;
 | |
| 			} else if (strcmp(long_options[option_index].name, "load_patch") == 0) {
 | |
| 				opt_load_patch = 1;
 | |
| 				opt_store_restore_file = optarg;
 | |
| 			} else if (strcmp(long_options[option_index].name, "save_patch") == 0) {
 | |
| 				opt_save_patch = 1;
 | |
| 				opt_store_restore_file = optarg;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'h':
 | |
| 			help(argv[0]);
 | |
| 			return 0;
 | |
| 		case 'l':
 | |
| 			opt_list = 1;
 | |
| 			opt_list_patch = optarg;
 | |
| 			break;
 | |
| 		case 'p':
 | |
| 			strcpy(comm_pipe, optarg);
 | |
| 			break;
 | |
| 		case 'a':
 | |
| 			opt_add = 1;
 | |
| 			opt_list_patch = optarg;
 | |
| 			break;
 | |
| 		case 'd':
 | |
| 			opt_del = 1;
 | |
| 			opt_list_patch = optarg;
 | |
| 			break;
 | |
| 		case 'i':
 | |
| 			opt_info = 1;
 | |
| 			break;
 | |
| 		case 'q':
 | |
| 			opt_con_add = 1;
 | |
| 			opt_list_patch = optarg;
 | |
| 			break;
 | |
| 		case 'w':
 | |
| 			opt_con_del = 1;
 | |
| 			opt_list_patch = optarg;
 | |
| 			break;
 | |
| 		case 'n':
 | |
| 			opt_use_default_io_names = 1;
 | |
| 			break;
 | |
| 		case 's':
 | |
| 			opt_setup = 1;
 | |
| 			break;
 | |
| 		case 'P':
 | |
| 			add_path(optarg);
 | |
| 			break;
 | |
| 		case '?':
 | |
| 			break;
 | |
| 		default:
 | |
| 			error("unknown option %c", c);
 | |
| 			return 1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	params.wfc = opt_wait_for_conn;
 | |
| 	if (opt_host) {
 | |
| 		params.type = COMM_TYPE_IP;
 | |
| 		params.name = strtok(opt_host, ":");
 | |
| 		if (!params.name)
 | |
| 			error("wrong hostname");
 | |
| 		tmp = strtok(NULL, ":");
 | |
| 		if (!tmp)
 | |
| 			error("wrong port");
 | |
| 		params.port = atoi(tmp);
 | |
| 	} else {
 | |
| 		params.type = COMM_TYPE_LOCAL;
 | |
| 		params.name = comm_pipe;
 | |
| 	}
 | |
| 
 | |
| 	params.server = 0;
 | |
| 
 | |
| 	while (1) {
 | |
| 		if ((err = liblo10k1_connect(¶ms, &conn))) {
 | |
| 			error("unable to connect ld10k1");
 | |
| 			break;
 | |
| 		}
 | |
| 		
 | |
| 		if ((err = liblo10k1_check_version(&conn))) {
 | |
| 			error("Wrong ld10k1 version");
 | |
| 			break;
 | |
| 		}
 | |
| 		
 | |
| 		if (opt_store || opt_restore) {
 | |
| 			if (opt_store) {
 | |
| 				if ((err = store_dsp(opt_store_restore_file)))
 | |
| 					break;
 | |
| 			} else {
 | |
| 				if ((err = restore_dsp(opt_store_restore_file)))
 | |
| 					break;
 | |
| 			}
 | |
| 		} else {
 | |
| 			if (opt_setup)
 | |
| 				if ((err = setup_dsp()))
 | |
| 					break;
 | |
| 			if (opt_list)
 | |
| 				if ((err = list_patch(opt_list_patch)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_add)
 | |
| 				if ((err = add_patch(opt_list_patch, opt_use_default_io_names, opt_ctrl, opt_patch_name, opt_where)))
 | |
| 					break;
 | |
| 			
 | |
| 			if (opt_load_patch)
 | |
| 				if ((err = load_dsp_patch(opt_store_restore_file, opt_ctrl, opt_patch_name, opt_where)))
 | |
| 					break;
 | |
| 					
 | |
| 			if (opt_save_patch)
 | |
| 				if ((err = save_dsp_patch(opt_store_restore_file, opt_where)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_del)
 | |
| 				if ((err = del_patch(opt_list_patch)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_con_add)
 | |
| 				if ((err = con_add(opt_list_patch)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_con_del)
 | |
| 				if ((err = con_del(opt_list_patch)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_debug)
 | |
| 				if ((err = debug(opt_debug)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_new_name)
 | |
| 				if ((err = rename_arg(opt_new_name)))
 | |
| 					break;
 | |
| 	
 | |
| 			if (opt_dump_name)
 | |
| 				if ((err = dump(opt_dump_name)))
 | |
| 					break;
 | |
| 		}
 | |
| 		break;
 | |
| 	}	
 | |
| 
 | |
| 	if (liblo10k1_is_open(&conn)) {
 | |
| 		/*send_msg(conn_num, FNC_CLOSE_CONN, NULL, 0);
 | |
| 		free_comm(conn_num);*/
 | |
| 		liblo10k1_disconnect(&conn);
 | |
| 	}
 | |
| 	
 | |
| 	free_all_paths();
 | |
| 
 | |
| 	return err;
 | |
| }
 | 
