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