2024-06-25 04:34:28 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <getopt.h>
|
|
|
|
#include "cdll.h"
|
|
|
|
#include "vmr.h"
|
2024-06-26 16:44:28 +01:00
|
|
|
#include "log.h"
|
2024-06-27 19:18:28 +01:00
|
|
|
#include "util.h"
|
2024-06-25 04:34:28 +01:00
|
|
|
|
2024-06-26 05:00:30 +01:00
|
|
|
#define MAX_LINE 512
|
2024-06-25 04:34:28 +01:00
|
|
|
|
2024-06-25 23:32:32 +01:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
FLOAT_T,
|
|
|
|
STRING_T,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct result
|
|
|
|
{
|
|
|
|
int type;
|
|
|
|
union val
|
|
|
|
{
|
|
|
|
float f;
|
2024-06-27 23:26:46 +01:00
|
|
|
wchar_t s[MAX_LINE];
|
2024-06-25 23:32:32 +01:00
|
|
|
} val;
|
|
|
|
};
|
|
|
|
|
|
|
|
void help(void);
|
2024-06-25 16:46:35 +01:00
|
|
|
int set_kind(char *kval);
|
|
|
|
int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind);
|
2024-06-25 04:34:28 +01:00
|
|
|
void interactive(T_VBVMR_INTERFACE *vmr);
|
2024-06-25 23:32:32 +01:00
|
|
|
void parse_command(T_VBVMR_INTERFACE *vmr, char *command);
|
2024-06-27 10:06:22 +01:00
|
|
|
void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res);
|
2024-06-25 04:34:28 +01:00
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
bool iflag = false;
|
2024-06-25 20:01:38 +01:00
|
|
|
int opt;
|
|
|
|
char *kvalue = "";
|
2024-06-26 16:44:28 +01:00
|
|
|
int dvalue;
|
2024-06-25 16:46:35 +01:00
|
|
|
int kind = BANANA;
|
2024-06-25 04:34:28 +01:00
|
|
|
|
2024-06-26 16:58:10 +01:00
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
help();
|
2024-06-27 01:17:59 +01:00
|
|
|
exit(EXIT_SUCCESS);
|
2024-06-26 16:58:10 +01:00
|
|
|
}
|
|
|
|
|
2024-06-27 01:17:59 +01:00
|
|
|
log_set_level(LOG_WARN);
|
2024-06-26 16:44:28 +01:00
|
|
|
|
|
|
|
while ((opt = getopt(argc, argv, "k:ihD:")) != -1)
|
2024-06-25 04:34:28 +01:00
|
|
|
{
|
2024-06-25 20:01:38 +01:00
|
|
|
switch (opt)
|
2024-06-25 04:34:28 +01:00
|
|
|
{
|
|
|
|
case 'i':
|
|
|
|
iflag = true;
|
|
|
|
break;
|
2024-06-25 16:46:35 +01:00
|
|
|
case 'k':
|
|
|
|
kvalue = optarg;
|
2024-06-26 16:44:28 +01:00
|
|
|
kind = set_kind(kvalue);
|
2024-06-25 16:46:35 +01:00
|
|
|
break;
|
2024-06-25 20:01:38 +01:00
|
|
|
case 'h':
|
2024-06-25 23:32:32 +01:00
|
|
|
help();
|
2024-06-25 20:01:38 +01:00
|
|
|
exit(EXIT_SUCCESS);
|
2024-06-26 16:44:28 +01:00
|
|
|
case 'D':
|
2024-06-27 01:17:59 +01:00
|
|
|
dvalue = atoi(optarg);
|
|
|
|
if (dvalue >= LOG_TRACE && dvalue <= LOG_FATAL)
|
2024-06-26 16:44:28 +01:00
|
|
|
{
|
|
|
|
log_set_level(dvalue);
|
|
|
|
}
|
2024-06-27 01:17:59 +01:00
|
|
|
else
|
|
|
|
{
|
2024-06-27 08:27:49 +01:00
|
|
|
log_error(
|
2024-06-27 01:17:59 +01:00
|
|
|
"-D arg out of range, expected value from 0 up to 5\n"
|
2024-06-27 08:27:49 +01:00
|
|
|
"Log level will default to LOG_WARN (3).\n");
|
2024-06-27 01:17:59 +01:00
|
|
|
}
|
2024-06-26 16:44:28 +01:00
|
|
|
break;
|
2024-06-25 04:34:28 +01:00
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-26 06:01:01 +01:00
|
|
|
T_VBVMR_INTERFACE iVMR;
|
2024-06-25 04:34:28 +01:00
|
|
|
T_VBVMR_INTERFACE *vmr = &iVMR;
|
|
|
|
|
2024-06-25 16:46:35 +01:00
|
|
|
int rep = init_voicemeeter(vmr, kind);
|
2024-06-25 17:22:46 +01:00
|
|
|
if (rep != 0)
|
2024-06-25 04:34:28 +01:00
|
|
|
{
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iflag)
|
|
|
|
{
|
|
|
|
puts("Interactive mode enabled. Enter 'Q' to exit.");
|
|
|
|
interactive(vmr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int i = optind; i < argc; i++)
|
|
|
|
{
|
2024-06-25 23:32:32 +01:00
|
|
|
parse_command(vmr, argv[i]);
|
2024-06-25 04:34:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rep = logout(vmr);
|
2024-06-25 17:22:46 +01:00
|
|
|
if (rep == 0)
|
2024-06-25 04:34:28 +01:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
else
|
|
|
|
return EXIT_FAILURE;
|
2024-06-25 16:46:35 +01:00
|
|
|
}
|
|
|
|
|
2024-06-25 23:32:32 +01:00
|
|
|
void help()
|
2024-06-25 20:01:38 +01:00
|
|
|
{
|
2024-06-25 23:32:32 +01:00
|
|
|
puts(
|
2024-06-26 19:31:08 +01:00
|
|
|
"Usage: ./vmrcli.exe [-i] [-k] [-D] <api commands>\n"
|
2024-06-25 20:01:38 +01:00
|
|
|
"Where: \n"
|
|
|
|
"\ti: Enable interactive mode\n"
|
2024-06-26 16:44:28 +01:00
|
|
|
"\tk: The kind of Voicemeeter (basic, banana, potato)\n"
|
2024-06-26 16:58:10 +01:00
|
|
|
"\tD: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL");
|
2024-06-25 20:01:38 +01:00
|
|
|
}
|
|
|
|
|
2024-06-25 16:46:35 +01:00
|
|
|
int set_kind(char *kval)
|
|
|
|
{
|
|
|
|
if (strcmp(kval, "basic") == 0)
|
|
|
|
{
|
|
|
|
return BASIC;
|
|
|
|
}
|
|
|
|
else if (strcmp(kval, "banana") == 0)
|
|
|
|
{
|
|
|
|
return BANANA;
|
|
|
|
}
|
|
|
|
else if (strcmp(kval, "potato") == 0)
|
|
|
|
{
|
|
|
|
if (sizeof(void *) == 8)
|
|
|
|
return POTATOX64;
|
|
|
|
else
|
|
|
|
return POTATO;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-06-27 08:27:49 +01:00
|
|
|
log_error("Unknown Voicemeeter kind '%s'\n", kval);
|
2024-06-25 16:46:35 +01:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2024-06-25 04:34:28 +01:00
|
|
|
}
|
|
|
|
|
2024-06-25 16:46:35 +01:00
|
|
|
int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
|
2024-06-25 04:34:28 +01:00
|
|
|
{
|
|
|
|
int rep = initialize_dll_interfaces(vmr);
|
|
|
|
if (rep < 0)
|
|
|
|
{
|
|
|
|
if (rep == -100)
|
|
|
|
{
|
2024-06-27 08:27:49 +01:00
|
|
|
log_error("Voicemeeter is not installed");
|
2024-06-25 04:34:28 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-06-27 08:27:49 +01:00
|
|
|
log_error("Error loading Voicemeeter dll with code %d\n", rep);
|
2024-06-25 04:34:28 +01:00
|
|
|
}
|
|
|
|
return rep;
|
|
|
|
}
|
|
|
|
|
2024-06-25 16:46:35 +01:00
|
|
|
rep = login(vmr, kind);
|
2024-06-25 04:34:28 +01:00
|
|
|
if (rep != 0)
|
|
|
|
{
|
2024-06-27 08:27:49 +01:00
|
|
|
log_error("Error logging into Voicemeeter");
|
2024-06-25 04:34:28 +01:00
|
|
|
return rep;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interactive(T_VBVMR_INTERFACE *vmr)
|
|
|
|
{
|
2024-06-28 01:23:40 +01:00
|
|
|
char input[MAX_LINE], command[MAX_LINE];
|
2024-06-26 05:54:13 +01:00
|
|
|
char *p = input;
|
2024-06-27 19:18:28 +01:00
|
|
|
int i;
|
2024-06-27 22:06:15 +01:00
|
|
|
size_t len;
|
2024-06-25 04:34:28 +01:00
|
|
|
|
2024-06-27 22:06:15 +01:00
|
|
|
printf(">> ");
|
2024-06-25 04:34:28 +01:00
|
|
|
while (fgets(input, MAX_LINE, stdin) != NULL)
|
|
|
|
{
|
2024-06-26 16:44:28 +01:00
|
|
|
input[strcspn(input, "\n")] = 0;
|
2024-06-27 22:06:15 +01:00
|
|
|
len = strlen(input);
|
|
|
|
if (len == 1 && toupper(input[0]) == 'Q')
|
2024-06-25 04:34:28 +01:00
|
|
|
break;
|
|
|
|
|
2024-06-27 22:06:15 +01:00
|
|
|
replace_multiple_space_with_one(input, len);
|
2024-06-26 05:54:13 +01:00
|
|
|
while (*p)
|
|
|
|
{
|
2024-06-27 22:06:15 +01:00
|
|
|
if (isspace(*p))
|
|
|
|
{
|
|
|
|
p++;
|
|
|
|
continue;
|
|
|
|
}
|
2024-06-27 19:18:28 +01:00
|
|
|
i = 0;
|
2024-06-27 22:56:58 +01:00
|
|
|
log_trace("commands still in buffer: %s", p);
|
2024-06-26 05:54:13 +01:00
|
|
|
|
2024-06-26 20:17:33 +01:00
|
|
|
while (!isspace(*p))
|
2024-06-26 05:54:13 +01:00
|
|
|
command[i++] = *p++;
|
|
|
|
command[i] = '\0';
|
2024-06-28 01:23:40 +01:00
|
|
|
p++; /* shift to next char */
|
2024-06-26 05:54:13 +01:00
|
|
|
|
2024-06-27 19:18:28 +01:00
|
|
|
if (command[0] != '\0')
|
|
|
|
parse_command(vmr, command);
|
2024-06-28 01:23:40 +01:00
|
|
|
memset(command, '\0', MAX_LINE);
|
2024-06-26 05:54:13 +01:00
|
|
|
}
|
|
|
|
|
2024-06-28 01:23:40 +01:00
|
|
|
p = input; /* reset pointer */
|
|
|
|
memset(input, '\0', MAX_LINE); /* reset input buffer */
|
2024-06-27 22:06:15 +01:00
|
|
|
printf(">> ");
|
2024-06-25 23:32:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
|
|
|
|
{
|
2024-06-26 16:44:28 +01:00
|
|
|
log_debug("Parsing %s", command);
|
|
|
|
|
2024-06-25 23:32:32 +01:00
|
|
|
if (command[0] == '!') /* toggle */
|
|
|
|
{
|
2024-06-26 00:03:50 +01:00
|
|
|
command++;
|
|
|
|
struct result res = {.type = FLOAT_T};
|
|
|
|
|
|
|
|
get(vmr, command, &res);
|
|
|
|
if (res.type == FLOAT_T)
|
|
|
|
{
|
|
|
|
set_parameter_float(vmr, command, 1 - res.val.f);
|
|
|
|
}
|
2024-06-25 23:32:32 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strchr(command, '=') != NULL) /* set */
|
|
|
|
{
|
|
|
|
set_parameters(vmr, command);
|
|
|
|
}
|
|
|
|
else /* get */
|
|
|
|
{
|
|
|
|
struct result res = {.type = FLOAT_T};
|
|
|
|
|
|
|
|
get(vmr, command, &res);
|
|
|
|
switch (res.type)
|
|
|
|
{
|
|
|
|
case FLOAT_T:
|
2024-06-27 02:00:17 +01:00
|
|
|
printf("%.1f\n", res.val.f);
|
2024-06-25 23:32:32 +01:00
|
|
|
break;
|
|
|
|
case STRING_T:
|
2024-06-27 23:26:46 +01:00
|
|
|
printf("%ls\n", res.val.s);
|
2024-06-25 23:32:32 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-27 10:06:22 +01:00
|
|
|
void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
|
2024-06-25 23:32:32 +01:00
|
|
|
{
|
2024-06-26 00:03:50 +01:00
|
|
|
clear_dirty(vmr);
|
2024-06-25 23:32:32 +01:00
|
|
|
if (get_parameter_float(vmr, command, &res->val.f) != 0)
|
|
|
|
{
|
|
|
|
res->type = STRING_T;
|
2024-06-26 05:54:13 +01:00
|
|
|
if (get_parameter_string(vmr, command, res->val.s) != 0)
|
|
|
|
{
|
|
|
|
res->val.s[0] = 0;
|
2024-06-27 22:06:15 +01:00
|
|
|
log_error("Unknown parameter '%s'", command);
|
2024-06-26 05:54:13 +01:00
|
|
|
}
|
2024-06-25 04:34:28 +01:00
|
|
|
}
|
|
|
|
}
|