mirror of
https://github.com/onyx-and-iris/vmrcli.git
synced 2026-04-07 17:33:33 +00:00
Compare commits
33 Commits
v0.2.0
...
faad5bc2c8
| Author | SHA1 | Date | |
|---|---|---|---|
| faad5bc2c8 | |||
| cc0ec73ef4 | |||
| c45df11286 | |||
| a383aaa36b | |||
| 28945b72c5 | |||
| 97fc9ca9ce | |||
| 12a55a52f2 | |||
| d4afbfa881 | |||
| ea1f05d323 | |||
| 900ed23ebf | |||
| 815a19210b | |||
| adaf3a7837 | |||
| 1199b31e2c | |||
| 2740c6c82d | |||
| 0bda368f59 | |||
| 49604b874b | |||
| 161b1061c4 | |||
| 3c46b3d9f3 | |||
| accab93fba | |||
| 947abb3c01 | |||
| e06a26f87b | |||
| 25bf542b46 | |||
| 50271edd8f | |||
| 41bf1322ac | |||
| f88fb9b994 | |||
| 9191a38745 | |||
| 1d71f38d39 | |||
| 2dda32ead9 | |||
| f60fdb4ed2 | |||
| 6567c2c610 | |||
| 41afc099dc | |||
| 7db10650fb | |||
| 06df144374 |
21
LICENSE
Normal file
21
LICENSE
Normal file
@@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2024 Onyx and Iris
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
54
README.md
54
README.md
@@ -1,4 +1,4 @@
|
||||
# VMRCLI Command Line Utility
|
||||
# Voicemeeter Remote Command Line Utility
|
||||
|
||||
## `Tested against`
|
||||
|
||||
@@ -12,13 +12,17 @@
|
||||
|
||||
## `Use`
|
||||
|
||||
`./vmrcli.exe [-i] [-k] [-D] <api commands>`
|
||||
```powershell
|
||||
./vmrcli.exe [-h] [-i] [-k] [-D] [-v] <api commands>
|
||||
```
|
||||
|
||||
Where:
|
||||
|
||||
- `i`: Enable interactive mode. If set any api commands passed will be ignored.
|
||||
- `h`: Prints the help message.
|
||||
- `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.
|
||||
- `D`: Set log level 0=TRACE, 1=DEBUG, 2=INFO, 3=WARN, 4=ERROR, 5=FATAL
|
||||
- `v`: Enable extra console output (toggle, set messages)
|
||||
|
||||
## `API Commands`
|
||||
|
||||
@@ -30,11 +34,32 @@ Examples:
|
||||
|
||||
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
|
||||
|
||||
`./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`
|
||||
|
||||
Running the following command in Powershell:
|
||||
|
||||
```powershell
|
||||
./vmrcli.exe -i
|
||||
```
|
||||
|
||||
Will open an interactive prompt:
|
||||
|
||||
```powershell
|
||||
Interactive mode enabled. Enter 'Q' to exit.
|
||||
>>
|
||||
```
|
||||
|
||||
API commands follow the same rules as listed above. Entering `Q` or `q` will exit the program.
|
||||
|
||||
## `Script files`
|
||||
|
||||
@@ -44,10 +69,25 @@ Scripts can be loaded from text files, for example in Powershell:
|
||||
./vmrcli.exe -D1 $(Get-Content .\example_commands.txt)
|
||||
```
|
||||
|
||||
Multiple API commands can be in a single line but they should be space separated.
|
||||
|
||||
## `Build`
|
||||
|
||||
Run the included `makefile` with [GNU Make](https://www.gnu.org/software/make/).
|
||||
|
||||
The binary in [Releases][releases] is compiled with coloured logging enabled. To disable this you can override the `LOG_USE_COLOR` variable, for example:
|
||||
|
||||
`make LOG_USE_COLOR=no`
|
||||
|
||||
## `Official Documentation`
|
||||
|
||||
- [Voicemeeter Remote C API](https://github.com/onyx-and-iris/Voicemeeter-SDK/blob/main/VoicemeeterRemoteAPI.pdf)
|
||||
- [Voicemeeter Remote C API][remoteapi-docs]
|
||||
|
||||
## `Special Thanks`
|
||||
|
||||
- [rxi](https://github.com/rxi) for writing the [log.c](https://github.com/rxi/log.c) package
|
||||
- [rxi][rxi-user] for writing the [log.c][log-c] package
|
||||
|
||||
[releases]: https://github.com/onyx-and-iris/vmrcli/releases
|
||||
[remoteapi-docs]: https://github.com/onyx-and-iris/Voicemeeter-SDK/blob/main/VoicemeeterRemoteAPI.pdf
|
||||
[rxi-user]: https://github.com/rxi
|
||||
[log-c]: https://github.com/rxi/log.c
|
||||
|
||||
@@ -1,8 +1,3 @@
|
||||
strip[0].mute
|
||||
!strip[0].mute
|
||||
strip[0].mute
|
||||
strip[1].mute=1
|
||||
strip[1].mute
|
||||
strip[0].gain
|
||||
strip[0].label=podmic
|
||||
strip[0].label
|
||||
strip[0].mute !strip[0].mute strip[0].mute strip[0].gain strip[0].label=podmic strip[0].label
|
||||
strip[1].mute=1 strip[1].mute
|
||||
bus[0].label bus[0].gain=-8.3
|
||||
9
include/util.h
Normal file
9
include/util.h
Normal file
@@ -0,0 +1,9 @@
|
||||
#ifndef __UTIL_H__
|
||||
#define __UTIL_H__
|
||||
|
||||
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__ */
|
||||
@@ -9,26 +9,28 @@ enum kind
|
||||
BASIC = 1,
|
||||
BANANA,
|
||||
POTATO,
|
||||
POTATOX64 = 6
|
||||
BASICX64,
|
||||
BANANAX64,
|
||||
POTATOX64,
|
||||
};
|
||||
|
||||
long login(T_VBVMR_INTERFACE *iVMR, int kind);
|
||||
long logout(T_VBVMR_INTERFACE *iVMR);
|
||||
long run_voicemeeter(T_VBVMR_INTERFACE *iVMR, int kind);
|
||||
long type(T_VBVMR_INTERFACE *iVMR, long *type);
|
||||
long version(T_VBVMR_INTERFACE *iVMR, long *version);
|
||||
long login(T_VBVMR_INTERFACE *vmr, int kind);
|
||||
long logout(T_VBVMR_INTERFACE *vmr);
|
||||
long run_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind);
|
||||
long type(T_VBVMR_INTERFACE *vmr, long *type);
|
||||
long version(T_VBVMR_INTERFACE *vmr, long *version);
|
||||
|
||||
bool is_pdirty(T_VBVMR_INTERFACE *iVMR);
|
||||
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 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_parameters(T_VBVMR_INTERFACE *iVMR, char *command);
|
||||
bool is_pdirty(T_VBVMR_INTERFACE *vmr);
|
||||
long get_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float *f);
|
||||
long get_parameter_string(T_VBVMR_INTERFACE *vmr, char *param, unsigned short *s);
|
||||
long set_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float val);
|
||||
long set_parameter_string(T_VBVMR_INTERFACE *vmr, char *param, char *s);
|
||||
long set_parameters(T_VBVMR_INTERFACE *vmr, char *command);
|
||||
|
||||
bool is_mdirty(T_VBVMR_INTERFACE *iVMR);
|
||||
long macrobutton_getstatus(T_VBVMR_INTERFACE *iVMR, long n, float *val, long mode);
|
||||
long macrobutton_setstatus(T_VBVMR_INTERFACE *iVMR, long n, float val, long mode);
|
||||
bool is_mdirty(T_VBVMR_INTERFACE *vmr);
|
||||
long macrobutton_getstatus(T_VBVMR_INTERFACE *vmr, long n, float *val, long mode);
|
||||
long macrobutton_setstatus(T_VBVMR_INTERFACE *vmr, long n, float val, long mode);
|
||||
|
||||
void clear_dirty(T_VBVMR_INTERFACE *iVMR);
|
||||
void clear_dirty(T_VBVMR_INTERFACE *vmr);
|
||||
|
||||
#endif
|
||||
#endif /* __VMR_H__ */
|
||||
10
makefile
10
makefile
@@ -9,7 +9,12 @@ EXE := $(BIN_DIR)/$(program).exe
|
||||
SRC := $(wildcard $(SRC_DIR)/*.c)
|
||||
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
|
||||
LDFLAGS := -Llib
|
||||
LDLIBS := -lm
|
||||
@@ -28,7 +33,6 @@ $(BIN_DIR) $(OBJ_DIR):
|
||||
pwsh -Command New-Item -Path $@ -ItemType Directory
|
||||
|
||||
clean:
|
||||
pwsh -Command Remove-Item -Recurse $(EXE)
|
||||
pwsh -Command Remove-Item -Recurse $(OBJ_DIR)
|
||||
pwsh -Command Remove-Item -Recurse $(EXE), $(OBJ_DIR)
|
||||
|
||||
-include $(OBJ:.o=.d)
|
||||
|
||||
13
src/cdll.c
13
src/cdll.c
@@ -1,6 +1,7 @@
|
||||
#include <stdbool.h>
|
||||
#include <stdio.h>
|
||||
#include "cdll.h"
|
||||
#include "util.h"
|
||||
|
||||
/*******************************************************************************/
|
||||
/** GET VOICEMEETER DIRECTORY **/
|
||||
@@ -12,18 +13,6 @@
|
||||
#define KEY_WOW64_32KEY 0x0200
|
||||
#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)
|
||||
{
|
||||
char szKey[256];
|
||||
|
||||
103
src/util.c
Normal file
103
src/util.c
Normal file
@@ -0,0 +1,103 @@
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include "vmr.h"
|
||||
#include "util.h"
|
||||
|
||||
/**
|
||||
* @brief Removes the last part of a path
|
||||
*
|
||||
* @param szPath
|
||||
*/
|
||||
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 count = 0;
|
||||
|
||||
if (len == 1 && (s[0] == ' ' || s[0] == '\t'))
|
||||
{
|
||||
s[0] = '\0';
|
||||
return len;
|
||||
}
|
||||
|
||||
if (len < 2)
|
||||
return len;
|
||||
|
||||
for (int i = 0; s[i] != '\0'; i++)
|
||||
{
|
||||
if (s[i] == ' ' || s[i] == '\t')
|
||||
{
|
||||
count++;
|
||||
}
|
||||
|
||||
if (s[i] != ' ' && s[i] != '\t')
|
||||
{
|
||||
if (count >= 1)
|
||||
{
|
||||
count = 0;
|
||||
s[j++] = ' ';
|
||||
}
|
||||
s[j++] = s[i];
|
||||
}
|
||||
}
|
||||
s[j] = '\0';
|
||||
return j;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief
|
||||
*
|
||||
* @param s Pointer to a character buffer
|
||||
* @param kind The kind of Voicemeeter.
|
||||
* @param n maximum number of characters to be written to the buffer
|
||||
* @return char* The kind of Voicemeeter as a string
|
||||
*/
|
||||
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;
|
||||
}
|
||||
134
src/vmr.c
134
src/vmr.c
@@ -3,141 +3,147 @@
|
||||
#include <time.h>
|
||||
#include "vmr.h"
|
||||
#include "log.h"
|
||||
#include "util.h"
|
||||
|
||||
long login(T_VBVMR_INTERFACE *iVMR, int kind)
|
||||
#define VERSION_STR_LEN 128
|
||||
#define KIND_STR_LEN 64
|
||||
|
||||
/**
|
||||
* @brief Logs into the API.
|
||||
* Tests for valid connection for up to 2 seconds.
|
||||
* If successful initializes the dirty parameters.
|
||||
*
|
||||
* @param vmr
|
||||
* @param kind
|
||||
* @return long
|
||||
*/
|
||||
long login(T_VBVMR_INTERFACE *vmr, int kind)
|
||||
{
|
||||
int rep;
|
||||
long v;
|
||||
|
||||
rep = iVMR->VBVMR_Login();
|
||||
Sleep(20);
|
||||
rep = vmr->VBVMR_Login();
|
||||
if (rep == 1)
|
||||
{
|
||||
run_voicemeeter(iVMR, kind);
|
||||
switch (kind)
|
||||
{
|
||||
case BASIC:
|
||||
log_info("Launching Voicemeeter Basic GUI");
|
||||
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;
|
||||
}
|
||||
run_voicemeeter(vmr, kind);
|
||||
char kind_s[KIND_STR_LEN];
|
||||
log_info(
|
||||
"Launching Voicemeeter %s GUI",
|
||||
kind_as_string(kind_s, kind, KIND_STR_LEN));
|
||||
|
||||
time_t endwait;
|
||||
int timeout = 2;
|
||||
|
||||
endwait = time(NULL) + timeout;
|
||||
while (time(NULL) < endwait)
|
||||
do
|
||||
{
|
||||
if ((rep = version(iVMR, &v)) == 0)
|
||||
if ((rep = version(vmr, &v)) == 0)
|
||||
break;
|
||||
Sleep(20);
|
||||
}
|
||||
Sleep(50);
|
||||
} while (time(NULL) < endwait);
|
||||
}
|
||||
if (rep == 0)
|
||||
{
|
||||
version(iVMR, &v);
|
||||
long v1 = (v & 0xFF000000) >> 24,
|
||||
v2 = (v & 0x00FF0000) >> 16,
|
||||
v3 = (v & 0x0000FF00) >> 8,
|
||||
v4 = (v & 0x000000FF);
|
||||
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(iVMR);
|
||||
version(vmr, &v);
|
||||
char version_s[VERSION_STR_LEN];
|
||||
log_info(
|
||||
"Successfully logged into the Voicemeeter API v%s",
|
||||
version_as_string(version_s, v, VERSION_STR_LEN));
|
||||
clear_dirty(vmr);
|
||||
}
|
||||
return rep;
|
||||
}
|
||||
|
||||
long logout(T_VBVMR_INTERFACE *iVMR)
|
||||
/**
|
||||
* @brief Logs out of the API giving a short wait to allow a
|
||||
* final instruction to complete.
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
* @return long VBVMR_Logout return value
|
||||
*/
|
||||
long logout(T_VBVMR_INTERFACE *vmr)
|
||||
{
|
||||
int rep;
|
||||
|
||||
Sleep(20); /* give time for last command */
|
||||
rep = iVMR->VBVMR_Logout();
|
||||
rep = vmr->VBVMR_Logout();
|
||||
if (rep == 0)
|
||||
log_info("Successfully logged out of the Voicemeeter API");
|
||||
return rep;
|
||||
}
|
||||
|
||||
long run_voicemeeter(T_VBVMR_INTERFACE *iVMR, int kind)
|
||||
long run_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
|
||||
{
|
||||
log_trace("VBVMR_RunVoicemeeter(%d)", kind);
|
||||
return iVMR->VBVMR_RunVoicemeeter((long)kind);
|
||||
return vmr->VBVMR_RunVoicemeeter((long)kind);
|
||||
}
|
||||
|
||||
long type(T_VBVMR_INTERFACE *iVMR, long *type)
|
||||
long type(T_VBVMR_INTERFACE *vmr, long *type)
|
||||
{
|
||||
return iVMR->VBVMR_GetVoicemeeterType(type);
|
||||
log_trace("VBVMR_GetVoicemeeterType(<long> *t)");
|
||||
return vmr->VBVMR_GetVoicemeeterType(type);
|
||||
}
|
||||
|
||||
long version(T_VBVMR_INTERFACE *iVMR, long *version)
|
||||
long version(T_VBVMR_INTERFACE *vmr, long *version)
|
||||
{
|
||||
log_trace("VBVMR_GetVoicemeeterType(<long> *v)");
|
||||
return iVMR->VBVMR_GetVoicemeeterVersion(version);
|
||||
log_trace("VBVMR_GetVoicemeeterVersion(<long> *v)");
|
||||
return vmr->VBVMR_GetVoicemeeterVersion(version);
|
||||
}
|
||||
|
||||
bool is_pdirty(T_VBVMR_INTERFACE *iVMR)
|
||||
bool is_pdirty(T_VBVMR_INTERFACE *vmr)
|
||||
{
|
||||
log_trace("VBVMR_IsParametersDirty()");
|
||||
return iVMR->VBVMR_IsParametersDirty() == 1;
|
||||
return vmr->VBVMR_IsParametersDirty() == 1;
|
||||
}
|
||||
|
||||
long get_parameter_float(T_VBVMR_INTERFACE *iVMR, 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);
|
||||
return iVMR->VBVMR_GetParameterFloat(param, f);
|
||||
log_trace("VBVMR_GetParameterFloat(%s, <float> *f)", param);
|
||||
return vmr->VBVMR_GetParameterFloat(param, f);
|
||||
}
|
||||
|
||||
long get_parameter_string(T_VBVMR_INTERFACE *iVMR, 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);
|
||||
return iVMR->VBVMR_GetParameterStringA(param, s);
|
||||
log_trace("VBVMR_GetParameterStringW(%s, <unsigned short> *s)", param);
|
||||
return vmr->VBVMR_GetParameterStringW(param, s);
|
||||
}
|
||||
|
||||
long set_parameter_float(T_VBVMR_INTERFACE *iVMR, char *param, float val)
|
||||
long set_parameter_float(T_VBVMR_INTERFACE *vmr, char *param, float val)
|
||||
{
|
||||
log_trace("VBVMR_SetParameterFloat(%s, %.2f)", param, val);
|
||||
return iVMR->VBVMR_SetParameterFloat(param, val);
|
||||
log_trace("VBVMR_SetParameterFloat(%s, %.1f)", param, val);
|
||||
return vmr->VBVMR_SetParameterFloat(param, val);
|
||||
}
|
||||
|
||||
long set_parameter_string(T_VBVMR_INTERFACE *iVMR, char *param, char *s)
|
||||
long set_parameter_string(T_VBVMR_INTERFACE *vmr, char *param, char *s)
|
||||
{
|
||||
log_trace("VBVMR_SetParameterStringA(%s, %s)", param, s);
|
||||
return iVMR->VBVMR_SetParameterStringA(param, s);
|
||||
return vmr->VBVMR_SetParameterStringA(param, s);
|
||||
}
|
||||
|
||||
long set_parameters(T_VBVMR_INTERFACE *iVMR, char *command)
|
||||
long set_parameters(T_VBVMR_INTERFACE *vmr, char *command)
|
||||
{
|
||||
log_trace("VBVMR_SetParameters(%s)", command);
|
||||
return iVMR->VBVMR_SetParameters(command);
|
||||
return vmr->VBVMR_SetParameters(command);
|
||||
}
|
||||
|
||||
bool is_mdirty(T_VBVMR_INTERFACE *iVMR)
|
||||
bool is_mdirty(T_VBVMR_INTERFACE *vmr)
|
||||
{
|
||||
return iVMR->VBVMR_MacroButton_IsDirty() == 1;
|
||||
return vmr->VBVMR_MacroButton_IsDirty() == 1;
|
||||
}
|
||||
|
||||
long macrobutton_getstatus(T_VBVMR_INTERFACE *iVMR, long n, float *val, long mode)
|
||||
long macrobutton_getstatus(T_VBVMR_INTERFACE *vmr, long n, float *val, long mode)
|
||||
{
|
||||
return iVMR->VBVMR_MacroButton_GetStatus(n, val, mode);
|
||||
return vmr->VBVMR_MacroButton_GetStatus(n, val, mode);
|
||||
}
|
||||
|
||||
long macrobutton_setstatus(T_VBVMR_INTERFACE *iVMR, long n, float val, long mode)
|
||||
long macrobutton_setstatus(T_VBVMR_INTERFACE *vmr, long n, float val, long mode)
|
||||
{
|
||||
return iVMR->VBVMR_MacroButton_SetStatus(n, val, mode);
|
||||
return vmr->VBVMR_MacroButton_SetStatus(n, val, mode);
|
||||
}
|
||||
|
||||
void clear_dirty(T_VBVMR_INTERFACE *iVMR)
|
||||
void clear_dirty(T_VBVMR_INTERFACE *vmr)
|
||||
{
|
||||
Sleep(30);
|
||||
while (is_pdirty(iVMR))
|
||||
while (is_pdirty(vmr))
|
||||
Sleep(1);
|
||||
}
|
||||
162
src/vmrcli.c
162
src/vmrcli.c
@@ -5,31 +5,44 @@
|
||||
#include "cdll.h"
|
||||
#include "vmr.h"
|
||||
#include "log.h"
|
||||
#include "util.h"
|
||||
|
||||
#define MAX_LINE 512
|
||||
|
||||
/**
|
||||
* @brief An enum used to define the kind of value
|
||||
* a 'get' call returns.
|
||||
*
|
||||
*/
|
||||
enum
|
||||
{
|
||||
FLOAT_T,
|
||||
STRING_T,
|
||||
};
|
||||
|
||||
/**
|
||||
* @brief A struct holding the result of a get call.
|
||||
*
|
||||
*/
|
||||
struct result
|
||||
{
|
||||
int type;
|
||||
union val
|
||||
{
|
||||
float f;
|
||||
char s[MAX_LINE];
|
||||
wchar_t s[MAX_LINE];
|
||||
} val;
|
||||
};
|
||||
|
||||
void help(void);
|
||||
int set_kind(char *kval);
|
||||
enum kind set_kind(char *kval);
|
||||
int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind);
|
||||
void interactive(T_VBVMR_INTERFACE *vmr);
|
||||
void parse_input(T_VBVMR_INTERFACE *vmr, char *input, int len);
|
||||
void parse_command(T_VBVMR_INTERFACE *vmr, char *command);
|
||||
struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res);
|
||||
void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res);
|
||||
|
||||
bool vflag = false;
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
@@ -37,7 +50,7 @@ int main(int argc, char *argv[])
|
||||
int opt;
|
||||
char *kvalue = "";
|
||||
int dvalue;
|
||||
int kind = BANANA;
|
||||
enum kind kind = BANANAX64;
|
||||
|
||||
if (argc == 1)
|
||||
{
|
||||
@@ -47,7 +60,7 @@ int main(int argc, char *argv[])
|
||||
|
||||
log_set_level(LOG_WARN);
|
||||
|
||||
while ((opt = getopt(argc, argv, "k:ihD:")) != -1)
|
||||
while ((opt = getopt(argc, argv, "k:ihD:v")) != -1)
|
||||
{
|
||||
switch (opt)
|
||||
{
|
||||
@@ -69,12 +82,14 @@ int main(int argc, char *argv[])
|
||||
}
|
||||
else
|
||||
{
|
||||
fputs(
|
||||
log_error(
|
||||
"-D arg out of range, expected value from 0 up to 5\n"
|
||||
"Log level will default to LOG_WARN (3).\n",
|
||||
stderr);
|
||||
"Log level will default to LOG_WARN (3).\n");
|
||||
}
|
||||
break;
|
||||
case 'v':
|
||||
vflag = true;
|
||||
break;
|
||||
default:
|
||||
abort();
|
||||
}
|
||||
@@ -98,7 +113,7 @@ int main(int argc, char *argv[])
|
||||
{
|
||||
for (int i = optind; i < argc; i++)
|
||||
{
|
||||
parse_command(vmr, argv[i]);
|
||||
parse_input(vmr, argv[i], strlen(argv[i]));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -109,24 +124,42 @@ int main(int argc, char *argv[])
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief prints the help message
|
||||
*
|
||||
*/
|
||||
void help()
|
||||
{
|
||||
puts(
|
||||
"Usage: ./vmrcli.exe [-i] [-k] [-D] <api commands>\n"
|
||||
"Usage: ./vmrcli.exe [-h] [-i] [-k] [-D] [-v] <api commands>\n"
|
||||
"Where: \n"
|
||||
"\th: Prints the help message\n"
|
||||
"\ti: Enable interactive mode\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\n"
|
||||
"\tv: Enable extra console output (toggle, set messages)");
|
||||
}
|
||||
|
||||
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 (sizeof(void *) == 8)
|
||||
return BASICX64;
|
||||
else
|
||||
return BASIC;
|
||||
}
|
||||
else if (strcmp(kval, "banana") == 0)
|
||||
{
|
||||
if (sizeof(void *) == 8)
|
||||
return BANANAX64;
|
||||
else
|
||||
return BANANA;
|
||||
}
|
||||
else if (strcmp(kval, "potato") == 0)
|
||||
@@ -138,11 +171,19 @@ int set_kind(char *kval)
|
||||
}
|
||||
else
|
||||
{
|
||||
fprintf(stderr, "Unknown Voicemeeter kind '%s'\n", kval);
|
||||
log_error("Unknown Voicemeeter kind '%s'\n", kval);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Defines the DLL interface as a struct.
|
||||
* Logs into the API.
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
* @param kind
|
||||
* @return int
|
||||
*/
|
||||
int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
|
||||
{
|
||||
int rep = initialize_dll_interfaces(vmr);
|
||||
@@ -150,11 +191,11 @@ int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
|
||||
{
|
||||
if (rep == -100)
|
||||
{
|
||||
fputs("Voicemeeter is not installed", stderr);
|
||||
log_error("Voicemeeter is not installed");
|
||||
}
|
||||
else
|
||||
{
|
||||
fprintf(stderr, "Error loading Voicemeeter dll with code %d\n", rep);
|
||||
log_error("Error loading Voicemeeter dll with code %d\n", rep);
|
||||
}
|
||||
return rep;
|
||||
}
|
||||
@@ -162,41 +203,69 @@ int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
|
||||
rep = login(vmr, kind);
|
||||
if (rep != 0)
|
||||
{
|
||||
fputs("Error logging into Voicemeeter", stderr);
|
||||
log_error("Error logging into Voicemeeter");
|
||||
return rep;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Continuously read lines from stdin.
|
||||
* Break if 'Q' is entered on the interactive prompt.
|
||||
* Each line is passed to parse_input()
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
*/
|
||||
void interactive(T_VBVMR_INTERFACE *vmr)
|
||||
{
|
||||
char input[MAX_LINE];
|
||||
char *p = input;
|
||||
size_t len;
|
||||
|
||||
printf(">> ");
|
||||
while (fgets(input, MAX_LINE, stdin) != NULL)
|
||||
{
|
||||
input[strcspn(input, "\n")] = 0;
|
||||
if (strlen(input) == 1 && (strncmp(input, "Q", 1) == 0 || strncmp(input, "q", 1) == 0))
|
||||
len = strlen(input);
|
||||
if (len == 1 && toupper(input[0]) == 'Q')
|
||||
break;
|
||||
|
||||
while (*p)
|
||||
{
|
||||
char command[MAX_LINE];
|
||||
int i = 0;
|
||||
parse_input(vmr, input, len);
|
||||
|
||||
while (!isspace(*p))
|
||||
command[i++] = *p++;
|
||||
command[i] = '\0';
|
||||
p++; /* shift to next char */
|
||||
|
||||
parse_command(vmr, command);
|
||||
}
|
||||
|
||||
p = input; /* reset pointer */
|
||||
memset(input, '\0', MAX_LINE); /* reset input buffer */
|
||||
printf(">> ");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Walks through each line split by a space delimiter.
|
||||
* Each token is passed to parse_command()
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
* @param input Each input line, from stdin or CLI args
|
||||
* @param len The length of the input line
|
||||
*/
|
||||
void parse_input(T_VBVMR_INTERFACE *vmr, char *input, int len)
|
||||
{
|
||||
char *token;
|
||||
|
||||
replace_multiple_space_with_one(input, len);
|
||||
token = strtok(input, " ");
|
||||
while (token != NULL)
|
||||
{
|
||||
parse_command(vmr, token);
|
||||
token = strtok(NULL, " ");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Execute each command according to type.
|
||||
* See command type definitions in:
|
||||
* https://github.com/onyx-and-iris/vmrcli?tab=readme-ov-file#api-commands
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
* @param command Each token from the input line as its own command string
|
||||
*/
|
||||
void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
|
||||
{
|
||||
log_debug("Parsing %s", command);
|
||||
@@ -208,8 +277,17 @@ void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
|
||||
|
||||
get(vmr, command, &res);
|
||||
if (res.type == FLOAT_T)
|
||||
{
|
||||
if (res.val.f == 1 || res.val.f == 0)
|
||||
{
|
||||
set_parameter_float(vmr, command, 1 - res.val.f);
|
||||
if (vflag)
|
||||
{
|
||||
printf("Toggling %s\n", command);
|
||||
}
|
||||
}
|
||||
else
|
||||
log_warn("%s does not appear to be a boolean parameter", command);
|
||||
}
|
||||
return;
|
||||
}
|
||||
@@ -217,6 +295,10 @@ void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
|
||||
if (strchr(command, '=') != NULL) /* set */
|
||||
{
|
||||
set_parameters(vmr, command);
|
||||
if (vflag)
|
||||
{
|
||||
printf("Setting %s\n", command);
|
||||
}
|
||||
}
|
||||
else /* get */
|
||||
{
|
||||
@@ -226,19 +308,25 @@ void parse_command(T_VBVMR_INTERFACE *vmr, char *command)
|
||||
switch (res.type)
|
||||
{
|
||||
case FLOAT_T:
|
||||
printf("%.1f\n", res.val.f);
|
||||
printf("%s: %.1f\n", command, res.val.f);
|
||||
break;
|
||||
case STRING_T:
|
||||
puts(res.val.s);
|
||||
printf("%s: %ls\n", command, res.val.s);
|
||||
break;
|
||||
default:
|
||||
fputs("Unexpected result type", stderr);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
|
||||
/**
|
||||
* @brief
|
||||
*
|
||||
* @param vmr The API interface as a struct
|
||||
* @param command A parsed 'get' command as a string
|
||||
* @param res A struct holding the result of the API call.
|
||||
*/
|
||||
void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
|
||||
{
|
||||
clear_dirty(vmr);
|
||||
if (get_parameter_float(vmr, command, &res->val.f) != 0)
|
||||
@@ -247,9 +335,7 @@ struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
|
||||
if (get_parameter_string(vmr, command, res->val.s) != 0)
|
||||
{
|
||||
res->val.s[0] = 0;
|
||||
fputs("Unknown parameter", stderr);
|
||||
log_error("Unknown parameter '%s'", command);
|
||||
}
|
||||
}
|
||||
|
||||
return res;
|
||||
}
|
||||
Reference in New Issue
Block a user