23 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
41bf1322ac add Interactive Mode section to README. 2024-06-27 22:07:53 +01:00
f88fb9b994 pass len of input to replace_multiple_space_with_one()
add an input prompt to interactive mode
2024-06-27 22:06:15 +01:00
9191a38745 readd comment 2024-06-27 19:20:21 +01:00
1d71f38d39 add utility function replace_multiple_space_with_one()
use it to parse the interactive input
2024-06-27 19:18:28 +01:00
2dda32ead9 up clean target 2024-06-27 10:28:09 +01:00
f60fdb4ed2 rename iVMR to vmr in vmr.c
update get() prototype. no need to return pointer.
2024-06-27 10:06:22 +01:00
6567c2c610 write error messages using log_error 2024-06-27 08:27:49 +01:00
41afc099dc sleep after, set to 50ms 2024-06-27 03:39:52 +01:00
7db10650fb do..while, sleep before first call to version() 2024-06-27 03:31:26 +01:00
06df144374 fix log_trace messages 2024-06-27 03:23:48 +01:00
8 changed files with 261 additions and 114 deletions

View File

@@ -12,11 +12,14 @@
## `Use` ## `Use`
`./vmrcli.exe [-i] [-k] [-D] <api commands>` ```powershell
./vmrcli.exe [-h] [-i] [-k] [-D] <api commands>
```
Where: Where:
- `i`: Enable interactive mode. If set any api commands passed will be ignored. - `h`: Prints the help dialogue.
- `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,11 +33,32 @@ 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`
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` ## `Script files`
@@ -44,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)

9
include/util.h Normal file
View 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__ */

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
@@ -28,7 +33,6 @@ $(BIN_DIR) $(OBJ_DIR):
pwsh -Command New-Item -Path $@ -ItemType Directory pwsh -Command New-Item -Path $@ -ItemType Directory
clean: clean:
pwsh -Command Remove-Item -Recurse $(EXE) pwsh -Command Remove-Item -Recurse $(EXE), $(OBJ_DIR)
pwsh -Command Remove-Item -Recurse $(OBJ_DIR)
-include $(OBJ:.o=.d) -include $(OBJ:.o=.d)

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];

90
src/util.c Normal file
View File

@@ -0,0 +1,90 @@
#include <stddef.h>
#include <stdio.h>
#include "vmr.h"
#include "util.h"
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;
}
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;
}

118
src/vmr.c
View File

@@ -3,141 +3,131 @@
#include <time.h> #include <time.h>
#include "vmr.h" #include "vmr.h"
#include "log.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
long login(T_VBVMR_INTERFACE *vmr, int kind)
{ {
int rep; int rep;
long v; long v;
rep = iVMR->VBVMR_Login(); rep = vmr->VBVMR_Login();
Sleep(20);
if (rep == 1) if (rep == 1)
{ {
run_voicemeeter(iVMR, 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;
endwait = time(NULL) + timeout; endwait = time(NULL) + timeout;
while (time(NULL) < endwait) do
{ {
if ((rep = version(iVMR, &v)) == 0) if ((rep = version(vmr, &v)) == 0)
break; break;
Sleep(20); Sleep(50);
} } while (time(NULL) < endwait);
} }
if (rep == 0) if (rep == 0)
{ {
version(iVMR, &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]; clear_dirty(vmr);
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);
} }
return rep; return rep;
} }
long logout(T_VBVMR_INTERFACE *iVMR) long logout(T_VBVMR_INTERFACE *vmr)
{ {
int rep; int rep;
Sleep(20); /* give time for last command */ Sleep(20); /* give time for last command */
rep = iVMR->VBVMR_Logout(); rep = vmr->VBVMR_Logout();
if (rep == 0) if (rep == 0)
log_info("Successfully logged out of the Voicemeeter API"); log_info("Successfully logged out of the Voicemeeter API");
return rep; 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); 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)"); log_trace("VBVMR_GetVoicemeeterVersion(<long> *v)");
return iVMR->VBVMR_GetVoicemeeterVersion(version); return vmr->VBVMR_GetVoicemeeterVersion(version);
} }
bool is_pdirty(T_VBVMR_INTERFACE *iVMR) bool is_pdirty(T_VBVMR_INTERFACE *vmr)
{ {
log_trace("VBVMR_IsParametersDirty()"); 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); log_trace("VBVMR_GetParameterFloat(%s, <float> *f)", param);
return iVMR->VBVMR_GetParameterFloat(param, f); 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); log_trace("VBVMR_GetParameterStringW(%s, <unsigned short> *s)", param);
return iVMR->VBVMR_GetParameterStringA(param, s); 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); log_trace("VBVMR_SetParameterFloat(%s, %.1f)", param, val);
return iVMR->VBVMR_SetParameterFloat(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); 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); 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); Sleep(30);
while (is_pdirty(iVMR)) while (is_pdirty(vmr))
Sleep(1); Sleep(1);
} }

View File

@@ -5,6 +5,7 @@
#include "cdll.h" #include "cdll.h"
#include "vmr.h" #include "vmr.h"
#include "log.h" #include "log.h"
#include "util.h"
#define MAX_LINE 512 #define MAX_LINE 512
@@ -20,16 +21,16 @@ 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);
struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res); void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res);
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
@@ -37,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)
{ {
@@ -69,10 +70,9 @@ int main(int argc, char *argv[])
} }
else else
{ {
fputs( log_error(
"-D arg out of range, expected value from 0 up to 5\n" "-D arg out of range, expected value from 0 up to 5\n"
"Log level will default to LOG_WARN (3).\n", "Log level will default to LOG_WARN (3).\n");
stderr);
} }
break; break;
default: default:
@@ -109,25 +109,42 @@ 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)
{ {
return BASIC; if (sizeof(void *) == 8)
return BASICX64;
else
return BASIC;
} }
else if (strcmp(kval, "banana") == 0) else if (strcmp(kval, "banana") == 0)
{ {
return BANANA; if (sizeof(void *) == 8)
return BANANAX64;
else
return BANANA;
} }
else if (strcmp(kval, "potato") == 0) else if (strcmp(kval, "potato") == 0)
{ {
@@ -138,7 +155,7 @@ int set_kind(char *kval)
} }
else else
{ {
fprintf(stderr, "Unknown Voicemeeter kind '%s'\n", kval); log_error("Unknown Voicemeeter kind '%s'\n", kval);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
} }
@@ -150,11 +167,11 @@ int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
{ {
if (rep == -100) if (rep == -100)
{ {
fputs("Voicemeeter is not installed", stderr); log_error("Voicemeeter is not installed");
} }
else 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; return rep;
} }
@@ -162,7 +179,7 @@ int init_voicemeeter(T_VBVMR_INTERFACE *vmr, int kind)
rep = login(vmr, kind); rep = login(vmr, kind);
if (rep != 0) if (rep != 0)
{ {
fputs("Error logging into Voicemeeter", stderr); log_error("Error logging into Voicemeeter");
return rep; return rep;
} }
@@ -171,29 +188,43 @@ 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;
int i;
size_t len;
printf(">> ");
while (fgets(input, MAX_LINE, stdin) != NULL) while (fgets(input, MAX_LINE, stdin) != NULL)
{ {
input[strcspn(input, "\n")] = 0; 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; break;
replace_multiple_space_with_one(input, len);
while (*p) while (*p)
{ {
char command[MAX_LINE]; if (isspace(*p))
int i = 0; {
p++;
continue;
}
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 */ p++; /* shift to next char */
parse_command(vmr, command); if (command[0] != '\0')
parse_command(vmr, command);
memset(command, '\0', MAX_LINE);
} }
p = input; /* reset pointer */ p = input; /* reset pointer */
memset(input, '\0', MAX_LINE); /* reset input buffer */
printf(">> ");
} }
} }
@@ -209,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)
{ {
set_parameter_float(vmr, command, 1 - res.val.f); if (res.val.f == 1 || res.val.f == 0)
set_parameter_float(vmr, command, 1 - res.val.f);
else
log_warn("%s does not appear to be a boolean parameter", command);
} }
return; return;
} }
@@ -229,16 +263,15 @@ 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:
fputs("Unexpected result type", stderr);
break; break;
} }
} }
} }
struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res) void get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
{ {
clear_dirty(vmr); clear_dirty(vmr);
if (get_parameter_float(vmr, command, &res->val.f) != 0) if (get_parameter_float(vmr, command, &res->val.f) != 0)
@@ -247,9 +280,7 @@ struct result *get(T_VBVMR_INTERFACE *vmr, char *command, struct result *res)
if (get_parameter_string(vmr, command, res->val.s) != 0) if (get_parameter_string(vmr, command, res->val.s) != 0)
{ {
res->val.s[0] = 0; res->val.s[0] = 0;
fputs("Unknown parameter", stderr); log_error("Unknown parameter '%s'", command);
} }
} }
return res;
} }