voicemeeter/base.go

351 lines
8.9 KiB
Go
Raw Normal View History

package voicemeeter
import (
"bytes"
"fmt"
"math"
"os"
"strings"
"syscall"
"time"
"unsafe"
)
var (
mod = syscall.NewLazyDLL(getDllPath())
2022-06-23 14:01:39 +01:00
vmLogin = mod.NewProc("VBVMR_Login")
vmLogout = mod.NewProc("VBVMR_Logout")
vmRunvm = mod.NewProc("VBVMR_RunVoicemeeter")
vmGetvmType = mod.NewProc("VBVMR_GetVoicemeeterType")
vmGetvmVersion = mod.NewProc("VBVMR_GetVoicemeeterVersion")
vmPdirty = mod.NewProc("VBVMR_IsParametersDirty")
vmGetParamFloat = mod.NewProc("VBVMR_GetParameterFloat")
vmGetParamString = mod.NewProc("VBVMR_GetParameterStringA")
2022-06-30 20:08:46 +01:00
vmGetLevelFloat = mod.NewProc("VBVMR_GetLevel")
2022-06-23 14:01:39 +01:00
vmSetParamFloat = mod.NewProc("VBVMR_SetParameterFloat")
vmSetParameters = mod.NewProc("VBVMR_SetParameters")
vmSetParamString = mod.NewProc("VBVMR_SetParameterStringA")
vmGetDevNumIn = mod.NewProc("VBVMR_Input_GetDeviceNumber")
vmGetDevDescIn = mod.NewProc("VBVMR_Input_GetDeviceDescA")
vmGetDevNumOut = mod.NewProc("VBVMR_Output_GetDeviceNumber")
vmGetDevDescOut = mod.NewProc("VBVMR_Output_GetDeviceDescA")
2022-06-23 14:01:39 +01:00
vmMdirty = mod.NewProc("VBVMR_MacroButton_IsDirty")
vmGetMacroStatus = mod.NewProc("VBVMR_MacroButton_GetStatus")
vmSetMacroStatus = mod.NewProc("VBVMR_MacroButton_SetStatus")
vmGetMidiMessage = mod.NewProc("VBVMR_GetMidiMessage")
)
// login logs into the API,
// attempts to launch Voicemeeter if it's not running,
// initializes dirty parameters.
func login(kindId string) {
res, _, _ := vmLogin.Call()
if res == 1 {
runVoicemeeter(kindId)
time.Sleep(time.Second)
} else if res != 0 {
err := fmt.Errorf("VBVMR_Login returned %d", res)
fmt.Println(err)
os.Exit(1)
}
fmt.Println("Logged into API")
for pdirty() || mdirty() {
}
}
// logout logs out of the API,
// delayed for 100ms to allow final operation to complete.
func logout() {
time.Sleep(100 * time.Millisecond)
res, _, _ := vmLogout.Call()
if res != 0 {
err := fmt.Errorf("VBVMR_Logout returned %d", res)
fmt.Println(err)
os.Exit(1)
}
fmt.Println("Logged out of API")
}
// runVoicemeeter attempts to launch a Voicemeeter GUI of a kind.
func runVoicemeeter(kindId string) {
vals := map[string]uint64{
"basic": 1,
"banana": 2,
"potato": 3,
}
res, _, _ := vmRunvm.Call(uintptr(vals[kindId]))
if res != 0 {
err := fmt.Errorf("VBVMR_RunVoicemeeter returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
// getVersion returns the version of Voicemeeter as a string
func getVersion() string {
var ver uint64
res, _, _ := vmGetvmVersion.Call(uintptr(unsafe.Pointer(&ver)))
if res != 0 {
err := fmt.Errorf("VBVMR_GetVoicemeeterVersion returned %d", res)
fmt.Println(err)
os.Exit(1)
}
v1 := (ver & 0xFF000000) >> 24
v2 := (ver & 0x00FF0000) >> 16
v3 := (ver & 0x0000FF00) >> 8
v4 := ver & 0x000000FF
return fmt.Sprintf("%d.%d.%d.%d", v1, v2, v3, v4)
}
// pdirty returns true iff a parameter value has changed
func pdirty() bool {
res, _, _ := vmPdirty.Call()
return int(res) == 1
}
// mdirty returns true iff a macrobutton value has changed
func mdirty() bool {
res, _, _ := vmMdirty.Call()
return int(res) == 1
}
// ldirty returns true iff a level value has changed
func ldirty(k *kind) bool {
_levelCache.stripLevelsBuff = make([]float32, (2*k.PhysIn)+(8*k.VirtIn))
_levelCache.busLevelsBuff = make([]float32, 8*k.NumBus())
for i := 0; i < (2*k.PhysIn)+(8*k.VirtIn); i++ {
_levelCache.stripLevelsBuff[i] = float32(getLevel(_levelCache.stripMode, i))
_levelCache.stripComp[i] = _levelCache.stripLevelsBuff[i] == _levelCache.stripLevels[i]
}
for i := 0; i < 8*k.NumBus(); i++ {
_levelCache.busLevelsBuff[i] = float32(getLevel(3, i))
_levelCache.busComp[i] = _levelCache.busLevelsBuff[i] == _levelCache.busLevels[i]
}
return !(allTrue(_levelCache.stripComp, (2*k.PhysIn)+(8*k.VirtIn)) && allTrue(_levelCache.busComp, 8*k.NumBus()))
}
// getVMType returns the type of Voicemeeter, as a string
func getVMType() string {
var type_ uint64
res, _, _ := vmGetvmType.Call(
uintptr(unsafe.Pointer(&type_)),
)
if res != 0 {
err := fmt.Errorf("VBVMR_GetVoicemeeterType returned %d", res)
fmt.Println(err)
os.Exit(1)
}
vals := map[uint64]string{
1: "basic",
2: "banana",
3: "potato",
}
return vals[type_]
}
// getParameterFloat gets the value of a float parameter
func getParameterFloat(name string) float64 {
var value float32
b := append([]byte(name), 0)
2022-06-23 14:01:39 +01:00
res, _, _ := vmGetParamFloat.Call(
uintptr(unsafe.Pointer(&b[0])),
uintptr(unsafe.Pointer(&value)),
)
if res != 0 {
err := fmt.Errorf("VBVMR_GetParameterFloat returned %d", res)
fmt.Println(err)
os.Exit(1)
}
return math.Round(float64(value)*10) / 10
}
// setParameterFloat sets the value of a float parameter
func setParameterFloat(name string, value float32) {
b1 := append([]byte(name), 0)
b2 := math.Float32bits(value)
2022-06-23 14:01:39 +01:00
res, _, _ := vmSetParamFloat.Call(
uintptr(unsafe.Pointer(&b1[0])),
uintptr(b2),
)
if res != 0 {
err := fmt.Errorf("VBVMR_SetParameterFloat returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
// getParameterString gets the value of a string parameter
func getParameterString(name string) string {
b1 := append([]byte(name), 0)
var b2 [512]byte
2022-06-23 14:01:39 +01:00
res, _, _ := vmGetParamString.Call(
uintptr(unsafe.Pointer(&b1[0])),
uintptr(unsafe.Pointer(&b2[0])),
)
if res != 0 {
err := fmt.Errorf("VBVMR_GetParameterStringA returned %d", res)
fmt.Println(err)
os.Exit(1)
}
str := bytes.Trim(b2[:], "\x00")
return string(str)
}
// getParameterString sets the value of a string parameter
func setParameterString(name, value string) {
b1 := append([]byte(name), 0)
b2 := append([]byte(value), 0)
2022-06-23 14:01:39 +01:00
res, _, _ := vmSetParamString.Call(
uintptr(unsafe.Pointer(&b1[0])),
uintptr(unsafe.Pointer(&b2[0])),
)
if res != 0 {
err := fmt.Errorf("VBVMR_SetParameterStringA returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
// setParametersMulti sets multiple parameters with a script
func setParametersMulti(script string) {
b1 := append([]byte(script), 0)
2022-06-23 14:01:39 +01:00
res, _, _ := vmSetParameters.Call(
uintptr(unsafe.Pointer(&b1[0])),
)
if res != 0 {
err := fmt.Errorf("VBVMR_SetParameters returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
// getMacroStatus gets a macrobutton value
func getMacroStatus(id, mode int) float32 {
var state float32
2022-06-23 14:01:39 +01:00
res, _, _ := vmGetMacroStatus.Call(
uintptr(id),
uintptr(unsafe.Pointer(&state)),
uintptr(mode),
)
if res != 0 {
err := fmt.Errorf("VBVMR_MacroButton_GetStatus returned %d", res)
fmt.Println(err)
os.Exit(1)
}
return state
}
// setMacroStatus sets a macrobutton value
func setMacroStatus(id, state, mode int) {
2022-06-23 14:01:39 +01:00
res, _, _ := vmSetMacroStatus.Call(
uintptr(id),
uintptr(state),
uintptr(mode),
)
if res != 0 {
err := fmt.Errorf("VBVMR_MacroButton_SetStatus returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
// getNumDevices returns the number of hardware input/output devices
func getNumDevices(dir string) uint64 {
if strings.Compare(dir, "in") == 0 {
res, _, _ := vmGetDevNumIn.Call()
return uint64(res)
} else {
res, _, _ := vmGetDevNumOut.Call()
return uint64(res)
}
}
// getDeviceDescription returns name, driver type and hwid for a given device
func getDeviceDescription(i int, dir string) (string, uint64, string) {
var t_ uint64
var b1 [512]byte
var b2 [512]byte
if strings.Compare(dir, "in") == 0 {
res, _, _ := vmGetDevDescIn.Call(
uintptr(i),
uintptr(unsafe.Pointer(&t_)),
uintptr(unsafe.Pointer(&b1[0])),
uintptr(unsafe.Pointer(&b2[0])),
)
if res != 0 {
err := fmt.Errorf("VBVMR_Input_GetDeviceDescA returned %d", res)
fmt.Println(err)
os.Exit(1)
}
} else {
res, _, _ := vmGetDevDescOut.Call(
uintptr(i),
uintptr(unsafe.Pointer(&t_)),
uintptr(unsafe.Pointer(&b1[0])),
uintptr(unsafe.Pointer(&b2[0])),
)
if res != 0 {
err := fmt.Errorf("VBVMR_Output_GetDeviceDescA returned %d", res)
fmt.Println(err)
os.Exit(1)
}
}
name := bytes.Trim(b1[:], "\x00")
hwid := bytes.Trim(b2[:], "\x00")
return string(name), t_, string(hwid)
}
// getLevel returns a single level value of type type_ for channel[i]
func getLevel(type_, i int) float32 {
var val float32
2022-06-30 20:08:46 +01:00
res, _, _ := vmGetLevelFloat.Call(
uintptr(type_),
uintptr(i),
uintptr(unsafe.Pointer(&val)),
)
if res != 0 {
err := fmt.Errorf("VBVMR_GetLevel returned %d", res)
fmt.Println(err)
os.Exit(1)
}
return val
}
// getMidiMessage gets midi channel, pitch and velocity for a single midi input
func getMidiMessage() bool {
var midi = newMidi()
var b1 [1024]byte
res, _, _ := vmGetMidiMessage.Call(
uintptr(unsafe.Pointer(&b1[0])),
uintptr(1024),
)
if int(res) < 0 {
err := fmt.Errorf("VBVMR_GetMidiMessage returned %d", res)
fmt.Println(err)
os.Exit(1)
}
msg := bytes.Trim(b1[:], "\x00")
if len(msg) > 0 {
for i := 0; i < len(msg)%3; i++ {
msg = append(msg, 0)
}
for i := 0; i < len(msg); i += 3 {
var ch = int(msg[i])
var pitch = int(msg[i+1])
var vel = int(msg[i+2])
midi.channel = ch
midi.current = pitch
midi.cache[pitch] = vel
}
}
return len(msg) > 0
}