mirror of https://github.com/jetkvm/kvm.git
created feature branch
This commit is contained in:
parent
8ffe66a1bc
commit
7361289b68
|
@ -11,6 +11,14 @@ type WakeOnLanDevice struct {
|
||||||
MacAddress string `json:"macAddress"`
|
MacAddress string `json:"macAddress"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type UsbConfig struct {
|
||||||
|
UsbVendorId string `json:"usb_vendor_id"`
|
||||||
|
UsbProductId string `json:"usb_product_id"`
|
||||||
|
UsbSerialNumber string `json:"usb_serial_number"`
|
||||||
|
UsbName string `json:"usb_name"`
|
||||||
|
UsbManufacturer string `json:"usb_manufacturer"`
|
||||||
|
}
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
CloudURL string `json:"cloud_url"`
|
CloudURL string `json:"cloud_url"`
|
||||||
CloudToken string `json:"cloud_token"`
|
CloudToken string `json:"cloud_token"`
|
||||||
|
@ -22,6 +30,7 @@ type Config struct {
|
||||||
LocalAuthToken string `json:"local_auth_token"`
|
LocalAuthToken string `json:"local_auth_token"`
|
||||||
LocalAuthMode string `json:"localAuthMode"` //TODO: fix it with migration
|
LocalAuthMode string `json:"localAuthMode"` //TODO: fix it with migration
|
||||||
WakeOnLanDevices []WakeOnLanDevice `json:"wake_on_lan_devices"`
|
WakeOnLanDevices []WakeOnLanDevice `json:"wake_on_lan_devices"`
|
||||||
|
UsbConfig UsbConfig `json:"usb_config"`
|
||||||
}
|
}
|
||||||
|
|
||||||
const configPath = "/userdata/kvm_config.json"
|
const configPath = "/userdata/kvm_config.json"
|
||||||
|
|
|
@ -36,17 +36,17 @@ func switchToScreenIfDifferent(screenName string) {
|
||||||
func updateDisplay() {
|
func updateDisplay() {
|
||||||
updateLabelIfChanged("ui_Home_Content_Ip", networkState.IPv4)
|
updateLabelIfChanged("ui_Home_Content_Ip", networkState.IPv4)
|
||||||
if usbState == "configured" {
|
if usbState == "configured" {
|
||||||
updateLabelIfChanged("ui_Home_Footer_Usb_Status_Label", "Connected")
|
updateLabelIfChanged("ui_Home_Footer_Usb_Status_Label", "Connected*")
|
||||||
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Usb_Status_Label", "state": "LV_STATE_DEFAULT"})
|
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Usb_Status_Label", "state": "LV_STATE_DEFAULT"})
|
||||||
} else {
|
} else {
|
||||||
updateLabelIfChanged("ui_Home_Footer_Usb_Status_Label", "Disconnected")
|
updateLabelIfChanged("ui_Home_Footer_Usb_Status_Label", "Disconnected*")
|
||||||
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Usb_Status_Label", "state": "LV_STATE_USER_2"})
|
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Usb_Status_Label", "state": "LV_STATE_USER_2"})
|
||||||
}
|
}
|
||||||
if lastVideoState.Ready {
|
if lastVideoState.Ready {
|
||||||
updateLabelIfChanged("ui_Home_Footer_Hdmi_Status_Label", "Connected")
|
updateLabelIfChanged("ui_Home_Footer_Hdmi_Status_Label", "Connected*")
|
||||||
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Hdmi_Status_Label", "state": "LV_STATE_DEFAULT"})
|
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Hdmi_Status_Label", "state": "LV_STATE_DEFAULT"})
|
||||||
} else {
|
} else {
|
||||||
updateLabelIfChanged("ui_Home_Footer_Hdmi_Status_Label", "Disconnected")
|
updateLabelIfChanged("ui_Home_Footer_Hdmi_Status_Label", "Disconnected*")
|
||||||
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Hdmi_Status_Label", "state": "LV_STATE_USER_2"})
|
_, _ = CallCtrlAction("lv_obj_set_state", map[string]interface{}{"obj": "ui_Home_Footer_Hdmi_Status_Label", "state": "LV_STATE_USER_2"})
|
||||||
}
|
}
|
||||||
updateLabelIfChanged("ui_Home_Header_Cloud_Status_Label", fmt.Sprintf("%d active", actionSessions))
|
updateLabelIfChanged("ui_Home_Header_Cloud_Status_Label", fmt.Sprintf("%d active", actionSessions))
|
||||||
|
|
21
jsonrpc.go
21
jsonrpc.go
|
@ -478,6 +478,26 @@ func rpcSetUsbEmulationState(enabled bool) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func rpcSetUsbConfig(usbConfig UsbConfig) error {
|
||||||
|
log.Printf("[jsonrpc.go:rpcSetUsbConfig] called")
|
||||||
|
|
||||||
|
LoadConfig()
|
||||||
|
config.UsbConfig = usbConfig
|
||||||
|
|
||||||
|
err2 := UpdateGadgetConfig()
|
||||||
|
if err2 != nil {
|
||||||
|
return fmt.Errorf("failed to write gadget config: %w", err2)
|
||||||
|
}
|
||||||
|
|
||||||
|
err := SaveConfig()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to save usb config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("[jsonrpc.go:rpcSetUsbConfig] usb config set to %s", usbConfig)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func rpcGetWakeOnLanDevices() ([]WakeOnLanDevice, error) {
|
func rpcGetWakeOnLanDevices() ([]WakeOnLanDevice, error) {
|
||||||
LoadConfig()
|
LoadConfig()
|
||||||
if config.WakeOnLanDevices == nil {
|
if config.WakeOnLanDevices == nil {
|
||||||
|
@ -542,6 +562,7 @@ var rpcHandlers = map[string]RPCHandler{
|
||||||
"isUpdatePending": {Func: rpcIsUpdatePending},
|
"isUpdatePending": {Func: rpcIsUpdatePending},
|
||||||
"getUsbEmulationState": {Func: rpcGetUsbEmulationState},
|
"getUsbEmulationState": {Func: rpcGetUsbEmulationState},
|
||||||
"setUsbEmulationState": {Func: rpcSetUsbEmulationState, Params: []string{"enabled"}},
|
"setUsbEmulationState": {Func: rpcSetUsbEmulationState, Params: []string{"enabled"}},
|
||||||
|
"setUsbConfig": {Func: rpcSetUsbConfig, Params: []string{"usbConfig"}},
|
||||||
"checkMountUrl": {Func: rpcCheckMountUrl, Params: []string{"url"}},
|
"checkMountUrl": {Func: rpcCheckMountUrl, Params: []string{"url"}},
|
||||||
"getVirtualMediaState": {Func: rpcGetVirtualMediaState},
|
"getVirtualMediaState": {Func: rpcGetVirtualMediaState},
|
||||||
"getStorageSpace": {Func: rpcGetStorageSpace},
|
"getStorageSpace": {Func: rpcGetStorageSpace},
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
113
usb.go
113
usb.go
|
@ -3,6 +3,7 @@ package kvm
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
gadget "github.com/openstadia/go-usb-gadget"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
@ -11,8 +12,6 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
gadget "github.com/openstadia/go-usb-gadget"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const configFSPath = "/sys/kernel/config"
|
const configFSPath = "/sys/kernel/config"
|
||||||
|
@ -20,19 +19,6 @@ const gadgetPath = "/sys/kernel/config/usb_gadget"
|
||||||
const kvmGadgetPath = "/sys/kernel/config/usb_gadget/jetkvm"
|
const kvmGadgetPath = "/sys/kernel/config/usb_gadget/jetkvm"
|
||||||
const configC1Path = "/sys/kernel/config/usb_gadget/jetkvm/configs/c.1"
|
const configC1Path = "/sys/kernel/config/usb_gadget/jetkvm/configs/c.1"
|
||||||
|
|
||||||
func mountConfigFS() error {
|
|
||||||
_, err := os.Stat(gadgetPath)
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
err = exec.Command("mount", "-t", "configfs", "none", configFSPath).Run()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to mount configfs: %w", err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return fmt.Errorf("unable to access usb gadget path: %w", err)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
_ = os.MkdirAll(imagesFolder, 0755)
|
_ = os.MkdirAll(imagesFolder, 0755)
|
||||||
udcs := gadget.GetUdcs()
|
udcs := gadget.GetUdcs()
|
||||||
|
@ -58,6 +44,60 @@ func init() {
|
||||||
//TODO: read hid reports(capslock, numlock, etc) from keyboardHidFile
|
//TODO: read hid reports(capslock, numlock, etc) from keyboardHidFile
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func UpdateGadgetConfig() error {
|
||||||
|
LoadConfig()
|
||||||
|
gadgetAttrs := [][]string{
|
||||||
|
{"idVendor", config.UsbConfig.UsbVendorId},
|
||||||
|
{"idProduct", config.UsbConfig.UsbProductId},
|
||||||
|
}
|
||||||
|
err := writeGadgetAttrs(kvmGadgetPath, gadgetAttrs)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("Successfully updated usb gadget attributes: %v", gadgetAttrs)
|
||||||
|
|
||||||
|
strAttrs := [][]string{
|
||||||
|
{"serialnumber", config.UsbConfig.UsbSerialNumber},
|
||||||
|
{"manufacturer", config.UsbConfig.UsbManufacturer},
|
||||||
|
{"product", config.UsbConfig.UsbName},
|
||||||
|
}
|
||||||
|
gadgetStringsPath := filepath.Join(kvmGadgetPath, "strings", "0x409")
|
||||||
|
err = os.MkdirAll(gadgetStringsPath, 0755)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = writeGadgetAttrs(gadgetStringsPath, strAttrs)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("Successfully updated string attributes: %s", strAttrs)
|
||||||
|
|
||||||
|
err = rebindUsb()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func mountConfigFS() error {
|
||||||
|
logger.Infof("Checking for gadgetPath: %s ...", gadgetPath)
|
||||||
|
_, err := os.Stat(gadgetPath)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
logger.Infof("running mount command...")
|
||||||
|
err = exec.Command("mount", "-t", "configfs", "none", configFSPath).Run()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to mount configfs: %w", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("unable to access usb gadget path: %w", err)
|
||||||
|
}
|
||||||
|
logger.Infof("Successfully mounted usb gadget at %s", gadgetPath)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func writeGadgetAttrs(basePath string, attrs [][]string) error {
|
func writeGadgetAttrs(basePath string, attrs [][]string) error {
|
||||||
for _, item := range attrs {
|
for _, item := range attrs {
|
||||||
filePath := filepath.Join(basePath, item[0])
|
filePath := filepath.Join(basePath, item[0])
|
||||||
|
@ -79,30 +119,35 @@ func writeGadgetConfig() error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
err = writeGadgetAttrs(kvmGadgetPath, [][]string{
|
LoadConfig()
|
||||||
{"bcdUSB", "0x0200"}, //USB 2.0
|
gadgetAttrs := [][]string{
|
||||||
{"idVendor", "0x1d6b"}, //The Linux Foundation
|
{"bcdUSB", "0x0200"}, //USB 2.0
|
||||||
{"idProduct", "0104"}, //Multifunction Composite Gadget¬
|
{"idVendor", config.UsbConfig.UsbVendorId},
|
||||||
|
{"idProduct", config.UsbConfig.UsbProductId},
|
||||||
{"bcdDevice", "0100"},
|
{"bcdDevice", "0100"},
|
||||||
})
|
}
|
||||||
|
err = writeGadgetAttrs(kvmGadgetPath, gadgetAttrs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
logger.Infof("Successfully wrote gadget attributes: %s", gadgetAttrs)
|
||||||
gadgetStringsPath := filepath.Join(kvmGadgetPath, "strings", "0x409")
|
gadgetStringsPath := filepath.Join(kvmGadgetPath, "strings", "0x409")
|
||||||
err = os.MkdirAll(gadgetStringsPath, 0755)
|
err = os.MkdirAll(gadgetStringsPath, 0755)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
err = writeGadgetAttrs(gadgetStringsPath, [][]string{
|
strAttrs := [][]string{
|
||||||
{"serialnumber", GetDeviceID()},
|
{"serialnumber", config.UsbConfig.UsbSerialNumber},
|
||||||
{"manufacturer", "JetKVM"},
|
{"manufacturer", config.UsbConfig.UsbManufacturer},
|
||||||
{"product", "JetKVM USB Emulation Device"},
|
{"product", config.UsbConfig.UsbName},
|
||||||
})
|
}
|
||||||
|
err = writeGadgetAttrs(gadgetStringsPath, strAttrs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
logger.Infof("Successfully wrote string attributes: %s", strAttrs)
|
||||||
|
|
||||||
configC1StringsPath := path.Join(configC1Path, "strings", "0x409")
|
configC1StringsPath := path.Join(configC1Path, "strings", "0x409")
|
||||||
err = os.MkdirAll(configC1StringsPath, 0755)
|
err = os.MkdirAll(configC1StringsPath, 0755)
|
||||||
|
@ -216,11 +261,27 @@ func writeGadgetConfig() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func rebindUsb() error {
|
func rebindUsb() error {
|
||||||
|
unbindErr := unbindUsb()
|
||||||
|
if unbindErr != nil {
|
||||||
|
return unbindErr
|
||||||
|
}
|
||||||
|
bindErr := bindUsb()
|
||||||
|
if bindErr != nil {
|
||||||
|
return bindErr
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unbindUsb() error {
|
||||||
err := os.WriteFile("/sys/bus/platform/drivers/dwc3/unbind", []byte(udc), 0644)
|
err := os.WriteFile("/sys/bus/platform/drivers/dwc3/unbind", []byte(udc), 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = os.WriteFile("/sys/bus/platform/drivers/dwc3/bind", []byte(udc), 0644)
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func bindUsb() error {
|
||||||
|
err := os.WriteFile("/sys/bus/platform/drivers/dwc3/bind", []byte(udc), 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue