accounts/usbwallet: two phase Ledger refreshes to avoid Windows bug

This commit is contained in:
Péter Szilágyi 2017-01-30 12:21:39 +02:00
parent 470b79385b
commit b3c0e9d3cc
No known key found for this signature in database
GPG Key ID: E9AE538CEDF8293D

View File

@ -234,32 +234,11 @@ func (hub *LedgerHub) rescan() {
hub.lock.Lock() hub.lock.Lock()
defer hub.lock.Unlock() defer hub.lock.Unlock()
// Iterate over all attached devices and fetch those seemingly Ledger // Iterate over all connected Ledger devices and do a heartbeat test
present := make(map[uint16]bool)
devices, _ := hub.ctx.ListDevices(func(desc *usb.Descriptor) bool {
// Discard all devices not advertizing as Ledger
ledger := false
for _, id := range ledgerDeviceIDs {
if desc.Vendor == id.Vendor && desc.Product == id.Product {
ledger = true
}
}
if !ledger {
return false
}
// If we have a Ledger, mark as still present, or open as new
id := uint16(desc.Bus)<<8 + uint16(desc.Address)
if _, known := hub.wallets[id]; known {
// Track it's presence, but don't open again
present[id] = true
return false
}
// New Ledger device, open it for communication
return true
})
// Drop any tracker wallet which disconnected
for id, wallet := range hub.wallets { for id, wallet := range hub.wallets {
if !present[id] { // If the device doesn't respond (io error on Windows, no device on Linux), drop
if err := wallet.resolveVersion(); err == usb.ERROR_IO || err == usb.ERROR_NO_DEVICE {
// Wallet disconnected or at least in a useless state
if wallet.address == (common.Address{}) { if wallet.address == (common.Address{}) {
glog.V(logger.Info).Infof("ledger wallet [%03d.%03d] disconnected", wallet.device.Bus, wallet.device.Address) glog.V(logger.Info).Infof("ledger wallet [%03d.%03d] disconnected", wallet.device.Bus, wallet.device.Address)
} else { } else {
@ -279,6 +258,26 @@ func (hub *LedgerHub) rescan() {
wallet.device.Close() wallet.device.Close()
} }
} }
// Iterate over all attached devices and fetch those seemingly Ledger
devices, _ := hub.ctx.ListDevices(func(desc *usb.Descriptor) bool {
// Discard all devices not advertizing as Ledger
ledger := false
for _, id := range ledgerDeviceIDs {
if desc.Vendor == id.Vendor && desc.Product == id.Product {
ledger = true
}
}
if !ledger {
return false
}
// If we have an already known Ledger, skip opening it
id := uint16(desc.Bus)<<8 + uint16(desc.Address)
if _, known := hub.wallets[id]; known {
return false
}
// New Ledger device, open it for communication
return true
})
// Start tracking all wallets which newly appeared // Start tracking all wallets which newly appeared
var err error var err error
for _, device := range devices { for _, device := range devices {