209 lines
4.2 KiB
Go
209 lines
4.2 KiB
Go
|
package fastboot
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"log"
|
||
|
|
||
|
"github.com/google/gousb"
|
||
|
)
|
||
|
|
||
|
type FastbootResponseStatus string
|
||
|
|
||
|
var Status = struct {
|
||
|
OKAY FastbootResponseStatus
|
||
|
FAIL FastbootResponseStatus
|
||
|
DATA FastbootResponseStatus
|
||
|
INFO FastbootResponseStatus
|
||
|
}{
|
||
|
OKAY: "OKAY",
|
||
|
FAIL: "FAIL",
|
||
|
DATA: "DATA",
|
||
|
INFO: "INFO",
|
||
|
}
|
||
|
|
||
|
type FastbootDevice struct {
|
||
|
Serial string
|
||
|
Device *gousb.Device
|
||
|
}
|
||
|
|
||
|
func FindDevices(ctx *gousb.Context) ([]FastbootDevice, error) {
|
||
|
var fastbootDevices []FastbootDevice
|
||
|
devs, err := ctx.OpenDevices(func(desc *gousb.DeviceDesc) bool {
|
||
|
for _, cfg := range desc.Configs {
|
||
|
for _, ifc := range cfg.Interfaces {
|
||
|
for _, alt := range ifc.AltSettings {
|
||
|
return alt.Protocol == 0x03 && alt.Class == 0xff && alt.SubClass == 0x42
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
})
|
||
|
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
for _, dev := range devs {
|
||
|
serial, err := dev.SerialNumber()
|
||
|
if err != nil {
|
||
|
log.Fatalf("Error retriving serial number for device: %v", err)
|
||
|
continue
|
||
|
}
|
||
|
fastbootDevices = append(fastbootDevices, FastbootDevice{Serial: serial, Device: dev})
|
||
|
}
|
||
|
|
||
|
return fastbootDevices, nil
|
||
|
}
|
||
|
|
||
|
func FindDevice(ctx *gousb.Context, serial string) (FastbootDevice, error) {
|
||
|
var fastbootDevice FastbootDevice
|
||
|
devs, err := ctx.OpenDevices(func(desc *gousb.DeviceDesc) bool {
|
||
|
for _, cfg := range desc.Configs {
|
||
|
for _, ifc := range cfg.Interfaces {
|
||
|
for _, alt := range ifc.AltSettings {
|
||
|
return alt.Protocol == 0x03 && alt.Class == 0xff && alt.SubClass == 0x42
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
})
|
||
|
|
||
|
if err != nil {
|
||
|
return fastbootDevice, err
|
||
|
}
|
||
|
|
||
|
for _, dev := range devs {
|
||
|
serialNumber, err := dev.SerialNumber()
|
||
|
if err != nil {
|
||
|
log.Fatalf("Error retriving serial number for device: %v", err)
|
||
|
continue
|
||
|
}
|
||
|
if serial != serialNumber {
|
||
|
continue
|
||
|
}
|
||
|
return FastbootDevice{Serial: serial, Device: dev}, nil
|
||
|
}
|
||
|
|
||
|
return fastbootDevice, fmt.Errorf("Devide with serial %s not found", serial)
|
||
|
}
|
||
|
|
||
|
func (d *FastbootDevice) Send(data []byte) error {
|
||
|
intf, done, err := d.Device.DefaultInterface()
|
||
|
if err != nil {
|
||
|
return nil
|
||
|
}
|
||
|
defer done()
|
||
|
|
||
|
outEndpoint, err := intf.OutEndpoint(0x01)
|
||
|
if err != nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
_, err = outEndpoint.Write(data)
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
func (d *FastbootDevice) GetMaxPacketSize() (int, error) {
|
||
|
intf, done, err := d.Device.DefaultInterface()
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
defer done()
|
||
|
|
||
|
outEndpoint, err := intf.OutEndpoint(0x01)
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
return outEndpoint.Desc.MaxPacketSize, nil
|
||
|
}
|
||
|
|
||
|
func (d *FastbootDevice) Recv() (FastbootResponseStatus, []byte, error) {
|
||
|
intf, done, err := d.Device.DefaultInterface()
|
||
|
if err != nil {
|
||
|
return Status.FAIL, nil, err
|
||
|
}
|
||
|
defer done()
|
||
|
|
||
|
inEndpoint, err := intf.InEndpoint(0x81)
|
||
|
if err != nil {
|
||
|
return Status.FAIL, nil, err
|
||
|
}
|
||
|
|
||
|
var data []byte
|
||
|
buf := make([]byte, inEndpoint.Desc.MaxPacketSize)
|
||
|
n, err := inEndpoint.Read(buf)
|
||
|
data = append(data, buf[:n]...)
|
||
|
var status FastbootResponseStatus
|
||
|
switch string(data[:4]) {
|
||
|
case "OKAY":
|
||
|
status = Status.OKAY
|
||
|
case "FAIL":
|
||
|
status = Status.FAIL
|
||
|
case "DATA":
|
||
|
status = Status.DATA
|
||
|
case "INFO":
|
||
|
status = Status.INFO
|
||
|
}
|
||
|
return status, data[4:], nil
|
||
|
}
|
||
|
|
||
|
func (d *FastbootDevice) BootImage(data []byte) error {
|
||
|
err := d.download(data)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
err = d.Send([]byte("boot"))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
status, data, err := d.Recv()
|
||
|
switch {
|
||
|
case status != Status.OKAY:
|
||
|
return fmt.Errorf("Failed to boot image: %s %s", status, data)
|
||
|
case err != nil:
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *FastbootDevice) download(data []byte) error {
|
||
|
data_size := len(data)
|
||
|
err := d.Send([]byte(fmt.Sprintf("download:%08x", data_size)))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
status, _, err := d.Recv()
|
||
|
switch {
|
||
|
case status != Status.DATA:
|
||
|
return fmt.Errorf("Failed to start data phase: %s", status)
|
||
|
case err != nil:
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
chunk_size := 0x40040
|
||
|
|
||
|
for i := 0; i < data_size; i += chunk_size {
|
||
|
end := i + chunk_size
|
||
|
if end > data_size {
|
||
|
end = data_size
|
||
|
}
|
||
|
err := d.Send(data[i:end])
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
status, data, err = d.Recv()
|
||
|
switch {
|
||
|
case status != Status.OKAY:
|
||
|
return fmt.Errorf("Failed to finish data phase: %s %s", status, data)
|
||
|
case err != nil:
|
||
|
return err
|
||
|
}
|
||
|
return nil
|
||
|
}
|