13 Commits

Author SHA1 Message Date
815a19210b protect against unsafe gain changes 2024-07-01 03:40:33 +01:00
adaf3a7837 break up long line 2024-06-29 03:08:03 +01:00
1199b31e2c extra X64 kinds added to enum kind
log message updated to reflect new kinds

replace_multiple_space_with_one() now returns new line length

kind_as_string() added to util.c
2024-06-29 03:05:51 +01:00
2740c6c82d move remove_name_in_path() into util.c
add version_as_string() to util.c
2024-06-28 03:21:38 +01:00
0bda368f59 add -h flag to help dialogue
add -h flag to Use section in README

update example markdown in README
2024-06-28 03:04:14 +01:00
49604b874b ensure we don't step past the NUL terminator here. 2024-06-28 01:54:04 +01:00
161b1061c4 remove unnecessary args in log_trace calls
ensure we shift a char after copying next command from input buffer
2024-06-28 01:23:40 +01:00
3c46b3d9f3 Use VBVMR_GetParameterStringW instead of VBVMR_GetParameterStringA
update struct result definition

fixes bug with unknown parameters
2024-06-27 23:26:46 +01:00
accab93fba reword 2024-06-27 23:09:10 +01:00
947abb3c01 clear the input buffer before reading from stdin again 2024-06-27 23:07:29 +01:00
e06a26f87b add Build section to README. 2024-06-27 23:02:34 +01:00
25bf542b46 add conditional override var LOG_USE_COLOR to makefile 2024-06-27 22:57:35 +01:00
50271edd8f add log_trace message while traversing input buffer 2024-06-27 22:56:58 +01:00
8 changed files with 147 additions and 64 deletions

View File

@@ -12,10 +12,13 @@
## `Use` ## `Use`
`./vmrcli.exe [-i] [-k] [-D] <api commands>` ```powershell
./vmrcli.exe [-h] [-i] [-k] [-D] <api commands>
```
Where: Where:
- `h`: Prints the help dialogue.
- `i`: Enable interactive mode. If set, any api commands passed on the command line will be ignored. - `i`: Enable interactive mode. If set, any api commands passed on the command line will be ignored.
- `k`: The kind of Voicemeeter (basic, banana or potato). Use this to launch the GUI. - `k`: The kind of Voicemeeter (basic, banana or potato). Use this to launch the GUI.
- `D`: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL - `D`: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL
@@ -30,17 +33,23 @@ Examples:
Launch basic GUI, set log level to INFO, Toggle Strip 0 Mute, then print its new value Launch basic GUI, set log level to INFO, Toggle Strip 0 Mute, then print its new value
`./vmrcli.exe -kbasic -D2 !strip[0].mute strip[0].mute` ```powershell
./vmrcli.exe -kbasic -D2 !strip[0].mute strip[0].mute
```
Launch banana GUI, set log level to DEBUG, set Strip 0 label to podmic then print Strip 2 label Launch banana GUI, set log level to DEBUG, set Strip 0 label to podmic then print Strip 2 label
`./vmrcli.exe -kbanana -D1 strip[0].label=podmic strip[2].label` ```powershell
./vmrcli.exe -kbanana -D1 strip[0].label=podmic strip[2].label
```
## `Interactive Mode` ## `Interactive Mode`
Running the following command in Powershell: Running the following command in Powershell:
`./vmrcli.exe -i` ```powershell
./vmrcli.exe -i
```
Will open an interactive prompt: Will open an interactive prompt:
@@ -59,6 +68,14 @@ Scripts can be loaded from text files, for example in Powershell:
./vmrcli.exe -D1 $(Get-Content .\example_commands.txt) ./vmrcli.exe -D1 $(Get-Content .\example_commands.txt)
``` ```
## `Build`
Run the included `makefile` with [GNU Make](https://www.gnu.org/software/make/).
By default the log.c module is built with coloured logging enabled. To disable this you can override the `LOG_USE_COLOR` variable, for example:
`make LOG_USE_COLOR=no`
## `Official Documentation` ## `Official Documentation`
- [Voicemeeter Remote C API](https://github.com/onyx-and-iris/Voicemeeter-SDK/blob/main/VoicemeeterRemoteAPI.pdf) - [Voicemeeter Remote C API](https://github.com/onyx-and-iris/Voicemeeter-SDK/blob/main/VoicemeeterRemoteAPI.pdf)

View File

@@ -1,6 +1,9 @@
#ifndef __UTIL_H__ #ifndef __UTIL_H__
#define __UTIL_H__ #define __UTIL_H__
void replace_multiple_space_with_one(char *s, size_t len); void remove_name_in_path(char *szPath);
int replace_multiple_space_with_one(char *s, size_t len);
char *kind_as_string(char *s, enum kind kind, int n);
char *version_as_string(char *, long v, int n);
#endif /* __UTIL_H__ */ #endif /* __UTIL_H__ */

View File

@@ -9,7 +9,9 @@ enum kind
BASIC = 1, BASIC = 1,
BANANA, BANANA,
POTATO, POTATO,
POTATOX64 = 6 BASICX64,
BANANAX64,
POTATOX64,
}; };
long login(T_VBVMR_INTERFACE *iVMR, int kind); long login(T_VBVMR_INTERFACE *iVMR, int kind);
@@ -20,7 +22,7 @@ long version(T_VBVMR_INTERFACE *iVMR, long *version);
bool is_pdirty(T_VBVMR_INTERFACE *iVMR); bool is_pdirty(T_VBVMR_INTERFACE *iVMR);
long get_parameter_float(T_VBVMR_INTERFACE *iVMR, char *param, float *f); long get_parameter_float(T_VBVMR_INTERFACE *iVMR, char *param, float *f);
long get_parameter_string(T_VBVMR_INTERFACE *iVMR, char *param, char *s); long get_parameter_string(T_VBVMR_INTERFACE *iVMR, char *param, unsigned short *s);
long set_parameter_float(T_VBVMR_INTERFACE *iVMR, char *param, float val); long set_parameter_float(T_VBVMR_INTERFACE *iVMR, char *param, float val);
long set_parameter_string(T_VBVMR_INTERFACE *iVMR, char *param, char *s); long set_parameter_string(T_VBVMR_INTERFACE *iVMR, char *param, char *s);
long set_parameters(T_VBVMR_INTERFACE *iVMR, char *command); long set_parameters(T_VBVMR_INTERFACE *iVMR, char *command);

View File

@@ -9,7 +9,12 @@ EXE := $(BIN_DIR)/$(program).exe
SRC := $(wildcard $(SRC_DIR)/*.c) SRC := $(wildcard $(SRC_DIR)/*.c)
OBJ := $(SRC:$(SRC_DIR)/%.c=$(OBJ_DIR)/%.o) OBJ := $(SRC:$(SRC_DIR)/%.c=$(OBJ_DIR)/%.o)
CPPFLAGS := -Iinclude -MMD -MP LOG_USE_COLOR ?= yes
ifeq ($(LOG_USE_COLOR), yes)
CPPFLAGS := -Iinclude -MMD -MP -DLOG_USE_COLOR
else
CPPFLAGS := -Iinclude -MMD -MP
endif
CFLAGS = -O -Wall -W -pedantic -ansi -std=c99 CFLAGS = -O -Wall -W -pedantic -ansi -std=c99
LDFLAGS := -Llib LDFLAGS := -Llib
LDLIBS := -lm LDLIBS := -lm

View File

@@ -1,6 +1,7 @@
#include <stdbool.h> #include <stdbool.h>
#include <stdio.h> #include <stdio.h>
#include "cdll.h" #include "cdll.h"
#include "util.h"
/*******************************************************************************/ /*******************************************************************************/
/** GET VOICEMEETER DIRECTORY **/ /** GET VOICEMEETER DIRECTORY **/
@@ -12,18 +13,6 @@
#define KEY_WOW64_32KEY 0x0200 #define KEY_WOW64_32KEY 0x0200
#endif #endif
void remove_name_in_path(char *szPath)
{
char *p = szPath;
while (*p++)
;
while (p > szPath && *p != '\\')
p--;
if (*p == '\\')
*p = '\0';
}
bool __cdecl registry_get_voicemeeter_folder(char *szDir) bool __cdecl registry_get_voicemeeter_folder(char *szDir)
{ {
char szKey[256]; char szKey[256];

View File

@@ -1,6 +1,28 @@
#include <stddef.h> #include <stddef.h>
#include <stdio.h>
#include "vmr.h"
#include "util.h"
void replace_multiple_space_with_one(char *s, size_t len) void remove_name_in_path(char *szPath)
{
char *p = szPath;
while (*p++)
;
while (p > szPath && *p != '\\')
p--;
if (*p == '\\')
*p = '\0';
}
/**
* @brief replaces multiple newlines and tabs with single spaces
*
* @param s the string to be reduced
* @param len current length of the string
* @return int new length of the string
*/
int replace_multiple_space_with_one(char *s, size_t len)
{ {
int j = 0; int j = 0;
int count = 0; int count = 0;
@@ -8,11 +30,11 @@ void replace_multiple_space_with_one(char *s, size_t len)
if (len == 1 && (s[0] == ' ' || s[0] == '\t')) if (len == 1 && (s[0] == ' ' || s[0] == '\t'))
{ {
s[0] = '\0'; s[0] = '\0';
return; return len;
} }
if (len < 2) if (len < 2)
return; return len;
for (int i = 0; s[i] != '\0'; i++) for (int i = 0; s[i] != '\0'; i++)
{ {
@@ -32,4 +54,37 @@ void replace_multiple_space_with_one(char *s, size_t len)
} }
} }
s[j] = '\0'; s[j] = '\0';
return j;
}
char *kind_as_string(char *s, enum kind kind, int n)
{
char *kinds[] = {
"Basic",
"Banana",
"Potato",
"Basic x64",
"Banana x64",
"Potato x64",
};
snprintf(s, n, kinds[kind - 1]);
return s;
}
/**
* @brief returns Voicemeeter's version as a string
*
* @param s string buffer the version will be written to
* @param v unprocessed version as a long int
* @param n maximum number of characters to be written to the buffer
* @return char*
*/
char *version_as_string(char *s, long v, int n)
{
long v1 = (v & 0xFF000000) >> 24,
v2 = (v & 0x00FF0000) >> 16,
v3 = (v & 0x0000FF00) >> 8,
v4 = (v & 0x000000FF);
snprintf(s, n, "%i.%i.%i.%i", (int)v1, (int)v2, (int)v3, (int)v4);
return s;
} }

View File

@@ -3,6 +3,10 @@
#include <time.h> #include <time.h>
#include "vmr.h" #include "vmr.h"
#include "log.h" #include "log.h"
#include "util.h"
#define VERSION_STR_LEN 128
#define KIND_STR_LEN 64
long login(T_VBVMR_INTERFACE *vmr, int kind) long login(T_VBVMR_INTERFACE *vmr, int kind)
{ {
@@ -13,21 +17,10 @@ long login(T_VBVMR_INTERFACE *vmr, int kind)
if (rep == 1) if (rep == 1)
{ {
run_voicemeeter(vmr, kind); run_voicemeeter(vmr, kind);
switch (kind) char kind_s[KIND_STR_LEN];
{ log_info(
case BASIC: "Launching Voicemeeter %s GUI",
log_info("Launching Voicemeeter Basic GUI"); kind_as_string(kind_s, kind, KIND_STR_LEN));
break;
case BANANA:
log_info("Launching Voicemeeter Banana GUI");
break;
case POTATO:
log_info("Launching Voicemeeter Potato GUI");
break;
case POTATOX64:
log_info("Launching Voicemeeter Potato x64 GUI");
break;
}
time_t endwait; time_t endwait;
int timeout = 2; int timeout = 2;
@@ -43,13 +36,10 @@ long login(T_VBVMR_INTERFACE *vmr, int kind)
if (rep == 0) if (rep == 0)
{ {
version(vmr, &v); version(vmr, &v);
long v1 = (v & 0xFF000000) >> 24, char version_s[VERSION_STR_LEN];
v2 = (v & 0x00FF0000) >> 16, log_info(
v3 = (v & 0x0000FF00) >> 8, "Successfully logged into the Voicemeeter API v%s",
v4 = (v & 0x000000FF); version_as_string(version_s, v, VERSION_STR_LEN));
char version_s[128];
sprintf(version_s, "%i.%i.%i.%i", (int)v1, (int)v2, (int)v3, (int)v4);
log_info("Successfully logged into the Voicemeeter API v%s", version_s);
clear_dirty(vmr); clear_dirty(vmr);
} }
return rep; return rep;
@@ -92,14 +82,14 @@ bool is_pdirty(T_VBVMR_INTERFACE *vmr)
long get_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float *f) long get_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float *f)
{ {
log_trace("VBVMR_GetParameterFloat(%s, <float> *f)", param, f); log_trace("VBVMR_GetParameterFloat(%s, <float> *f)", param);
return vmr->VBVMR_GetParameterFloat(param, f); return vmr->VBVMR_GetParameterFloat(param, f);
} }
long get_parameter_string(T_VBVMR_INTERFACE *vmr, char *param, char *s) long get_parameter_string(T_VBVMR_INTERFACE *vmr, char *param, unsigned short *s)
{ {
log_trace("VBVMR_GetParameterStringA(%s, <char> *s)", param, s); log_trace("VBVMR_GetParameterStringW(%s, <unsigned short> *s)", param);
return vmr->VBVMR_GetParameterStringA(param, s); return vmr->VBVMR_GetParameterStringW(param, s);
} }
long set_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float val) long set_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float val)

View File

@@ -21,12 +21,12 @@ struct result
union val union val
{ {
float f; float f;
char s[MAX_LINE]; wchar_t s[MAX_LINE];
} val; } val;
}; };
void help(void); void help(void);
int set_kind(char *kval); enum kind set_kind(char *kval);
int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind); int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind);
void interactive(T_VBVMR_INTERFACE *vmr); void interactive(T_VBVMR_INTERFACE *vmr);
void parse_command(T_VBVMR_INTERFACE *vmr, char *command); void parse_command(T_VBVMR_INTERFACE *vmr, char *command);
@@ -38,7 +38,7 @@ int main(int argc, char *argv[])
int opt; int opt;
char *kvalue = ""; char *kvalue = "";
int dvalue; int dvalue;
int kind = BANANA; enum kind kind = BANANAX64;
if (argc == 1) if (argc == 1)
{ {
@@ -109,24 +109,41 @@ int main(int argc, char *argv[])
return EXIT_FAILURE; return EXIT_FAILURE;
} }
/**
* @brief prints the help dialogue
*
*/
void help() void help()
{ {
puts( puts(
"Usage: ./vmrcli.exe [-i] [-k] [-D] <api commands>\n" "Usage: ./vmrcli.exe [-h] [-i] [-k] [-D] <api commands>\n"
"Where: \n" "Where: \n"
"\th: Prints the help dialogue\n"
"\ti: Enable interactive mode\n" "\ti: Enable interactive mode\n"
"\tk: The kind of Voicemeeter (basic, banana, potato)\n" "\tk: The kind of Voicemeeter (basic, banana, potato)\n"
"\tD: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL"); "\tD: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL");
} }
int set_kind(char *kval) /**
* @brief Set the kind object
*
* @param kval
* @return enum kind
*/
enum kind set_kind(char *kval)
{ {
if (strcmp(kval, "basic") == 0) if (strcmp(kval, "basic") == 0)
{ {
if (sizeof(void *) == 8)
return BASICX64;
else
return BASIC; return BASIC;
} }
else if (strcmp(kval, "banana") == 0) else if (strcmp(kval, "banana") == 0)
{ {
if (sizeof(void *) == 8)
return BANANAX64;
else
return BANANA; return BANANA;
} }
else if (strcmp(kval, "potato") == 0) else if (strcmp(kval, "potato") == 0)
@@ -171,9 +188,8 @@ int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
void interactive(T_VBVMR_INTERFACE *vmr) void interactive(T_VBVMR_INTERFACE *vmr)
{ {
char input[MAX_LINE]; char input[MAX_LINE], command[MAX_LINE];
char *p = input; char *p = input;
char command[MAX_LINE];
int i; int i;
size_t len; size_t len;
@@ -193,18 +209,21 @@ void interactive(T_VBVMR_INTERFACE *vmr)
p++; p++;
continue; continue;
} }
i = 0; log_trace("commands still in buffer: %s", p);
while (!isspace(*p)) i = 0;
while (*p && !isspace(*p))
command[i++] = *p++; command[i++] = *p++;
command[i] = '\0'; command[i] = '\0';
p++; /* shift to next char */
if (command[0] != '\0') if (command[0] != '\0')
parse_command(vmr, command); parse_command(vmr, command);
memset(command, '\0', sizeof(command)); memset(command, '\0', MAX_LINE);
} }
p = input; /* reset pointer */ p = input; /* reset pointer */
memset(input, '\0', MAX_LINE); /* reset input buffer */
printf(">> "); printf(">> ");
} }
} }
@@ -221,7 +240,10 @@ void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
get(vmr, command, &res); get(vmr, command, &res);
if (res.type == FLOAT_T) if (res.type == FLOAT_T)
{ {
if (res.val.f == 1 || res.val.f == 0)
set_parameter_float(vmr, command, 1 - res.val.f); set_parameter_float(vmr, command, 1 - res.val.f);
else
log_warn("%s does not appear to be a boolean parameter", command);
} }
return; return;
} }
@@ -241,7 +263,7 @@ void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
printf("%.1f\n", res.val.f); printf("%.1f\n", res.val.f);
break; break;
case STRING_T: case STRING_T:
puts(res.val.s); printf("%ls\n", res.val.s);
break; break;
default: default:
break; break;