Files
wireguard-go/device/device.go
T

521 lines
13 KiB
Go
Raw Normal View History

2019-01-02 01:55:51 +01:00
/* SPDX-License-Identifier: MIT
2018-05-03 15:04:00 +02:00
*
2021-01-28 17:52:15 +01:00
* Copyright (C) 2017-2021 WireGuard LLC. All Rights Reserved.
2018-05-03 15:04:00 +02:00
*/
2019-03-03 04:04:41 +01:00
package device
2017-05-30 22:36:49 +02:00
import (
2017-06-28 23:45:45 +02:00
"runtime"
2017-05-30 22:36:49 +02:00
"sync"
2017-07-15 16:27:59 +02:00
"sync/atomic"
2017-08-11 16:18:20 +02:00
"time"
2019-05-14 09:09:52 +02:00
"golang.org/x/net/ipv4"
"golang.org/x/net/ipv6"
2021-01-29 18:24:45 +01:00
"golang.zx2c4.com/wireguard/conn"
2019-05-14 09:09:52 +02:00
"golang.zx2c4.com/wireguard/ratelimiter"
"golang.zx2c4.com/wireguard/rwcancel"
2019-05-14 09:09:52 +02:00
"golang.zx2c4.com/wireguard/tun"
2017-05-30 22:36:49 +02:00
)
type Device struct {
2018-01-26 22:52:32 +01:00
state struct {
2021-01-19 09:02:16 -08:00
// state holds the device's state. It is accessed atomically.
// Use the device.deviceState method to read it.
// If state is not locked, state is the current state of the device.
// If state is locked, state is the current state or intended future state of the device.
2021-01-19 09:02:16 -08:00
// For example, while executing a call to Up, state will be deviceStateUp.
// There is no guarantee that that intended future state of the device
// will become the actual state; Up can fail.
// The device can also change state multiple times between time of check and time of use.
// Unsynchronized uses of state must therefore be advisory/best-effort only.
2021-02-09 15:32:55 +01:00
state uint32 // actually a deviceState, but typed uint32 for convenience
2021-01-19 09:02:16 -08:00
// stopping blocks until all inputs to Device have been closed.
2018-05-05 06:00:38 +02:00
stopping sync.WaitGroup
2021-01-19 09:02:16 -08:00
// mu protects state changes.
sync.Mutex
2018-01-26 22:52:32 +01:00
}
2018-02-02 16:40:14 +01:00
2017-07-14 14:25:18 +02:00
net struct {
2018-05-20 06:19:29 +02:00
stopping sync.WaitGroup
sync.RWMutex
bind conn.Bind // bind interface
netlinkCancel *rwcancel.RWCancel
port uint16 // listening port
fwmark uint32 // mark value (0 = disabled)
2017-06-30 14:41:08 +02:00
}
2018-02-02 16:40:14 +01:00
2018-05-13 23:14:43 +02:00
staticIdentity struct {
sync.RWMutex
2018-02-02 16:40:14 +01:00
privateKey NoisePrivateKey
publicKey NoisePublicKey
}
peers struct {
2020-12-15 17:44:21 -08:00
empty AtomicBool // empty reports whether len(keyMap) == 0
sync.RWMutex // protects keyMap
keyMap map[NoisePublicKey]*Peer
2018-02-02 16:40:14 +01:00
}
2018-05-13 23:14:43 +02:00
allowedips AllowedIPs
indexTable IndexTable
cookieChecker CookieChecker
2018-02-02 16:40:14 +01:00
rate struct {
underLoadUntil int64
2018-02-12 22:29:11 +01:00
limiter ratelimiter.Ratelimiter
2018-02-02 16:40:14 +01:00
}
pool struct {
messageBuffers *WaitPool
inboundElements *WaitPool
outboundElements *WaitPool
2018-02-02 16:40:14 +01:00
}
queue struct {
2021-01-29 14:54:11 +01:00
encryption *outboundQueue
decryption *inboundQueue
2021-01-29 18:24:45 +01:00
handshake *handshakeQueue
2017-06-28 23:45:45 +02:00
}
2018-02-02 16:40:14 +01:00
tun struct {
device tun.Device
2018-02-02 16:40:14 +01:00
mtu int32
}
ipcMutex sync.RWMutex
2021-01-29 18:24:45 +01:00
closed chan struct{}
log *Logger
2018-02-02 16:40:14 +01:00
}
2021-01-19 09:02:16 -08:00
// deviceState represents the state of a Device.
2021-02-09 15:39:19 +01:00
// There are three states: down, up, closed.
2021-01-19 09:02:16 -08:00
// Transitions:
//
2021-02-09 15:39:19 +01:00
// down -----+
// ↑↓ ↓
// up -> closed
2021-01-19 09:02:16 -08:00
//
type deviceState uint32
2021-02-09 15:39:19 +01:00
//go:generate go run golang.org/x/tools/cmd/stringer -type deviceState -trimprefix=deviceState
2021-01-19 09:02:16 -08:00
const (
2021-02-09 15:39:19 +01:00
deviceStateDown deviceState = iota
2021-01-19 09:02:16 -08:00
deviceStateUp
deviceStateClosed
)
// deviceState returns device.state.state as a deviceState
// See those docs for how to interpret this value.
func (device *Device) deviceState() deviceState {
return deviceState(atomic.LoadUint32(&device.state.state))
}
// isClosed reports whether the device is closed (or is closing).
// See device.state.state comments for how to interpret this value.
func (device *Device) isClosed() bool {
return device.deviceState() == deviceStateClosed
}
// isUp reports whether the device is up (or is attempting to come up).
// See device.state.state comments for how to interpret this value.
func (device *Device) isUp() bool {
return device.deviceState() == deviceStateUp
}
2018-02-02 16:40:14 +01:00
/* Converts the peer into a "zombie", which remains in the peer map,
* but processes no packets and does not exists in the routing table.
*
* Must hold device.peers.Mutex
2018-02-02 16:40:14 +01:00
*/
func unsafeRemovePeer(device *Device, peer *Peer, key NoisePublicKey) {
// stop routing and processing of packets
2018-05-13 23:14:43 +02:00
device.allowedips.RemoveByPeer(peer)
2018-02-02 16:40:14 +01:00
peer.Stop()
// remove from peer map
delete(device.peers.keyMap, key)
2020-12-15 17:44:21 -08:00
device.peers.empty.Set(len(device.peers.keyMap) == 0)
2017-06-01 21:31:30 +02:00
}
2021-01-19 09:02:16 -08:00
// changeState attempts to change the device state to match want.
func (device *Device) changeState(want deviceState) {
device.state.Lock()
defer device.state.Unlock()
2021-01-19 09:02:16 -08:00
old := device.deviceState()
if old == deviceStateClosed {
// once closed, always closed
device.log.Verbosef("Interface closed, ignored requested state %s", want)
2018-01-13 09:00:37 +01:00
return
}
2021-01-19 09:02:16 -08:00
switch want {
case old:
2018-02-04 16:46:24 +01:00
return
2021-01-19 09:02:16 -08:00
case deviceStateUp:
atomic.StoreUint32(&device.state.state, uint32(deviceStateUp))
if ok := device.upLocked(); ok {
2018-02-04 16:46:24 +01:00
break
}
2021-01-19 09:02:16 -08:00
fallthrough // up failed; bring the device all the way back down
case deviceStateDown:
atomic.StoreUint32(&device.state.state, uint32(deviceStateDown))
device.downLocked()
}
device.log.Verbosef("Interface state was %s, requested %s, now %s", old, want, device.deviceState())
}
2018-02-04 16:46:24 +01:00
2021-01-19 09:02:16 -08:00
// upLocked attempts to bring the device up and reports whether it succeeded.
// The caller must hold device.state.mu and is responsible for updating device.state.state.
func (device *Device) upLocked() bool {
if err := device.BindUpdate(); err != nil {
device.log.Errorf("Unable to update bind: %v", err)
return false
2018-02-04 16:46:24 +01:00
}
2021-01-19 09:02:16 -08:00
device.peers.RLock()
for _, peer := range device.peers.keyMap {
peer.Start()
if atomic.LoadUint32(&peer.persistentKeepaliveInterval) > 0 {
peer.SendKeepalive()
}
}
device.peers.RUnlock()
return true
}
2018-02-04 16:46:24 +01:00
2021-01-19 09:02:16 -08:00
// downLocked attempts to bring the device down.
// The caller must hold device.state.mu and is responsible for updating device.state.state.
func (device *Device) downLocked() {
err := device.BindClose()
if err != nil {
device.log.Errorf("Bind close failed: %v", err)
}
2018-01-26 22:52:32 +01:00
2021-01-19 09:02:16 -08:00
device.peers.RLock()
for _, peer := range device.peers.keyMap {
peer.Stop()
}
device.peers.RUnlock()
2018-01-26 22:52:32 +01:00
}
func (device *Device) Up() {
2021-01-19 09:02:16 -08:00
device.changeState(deviceStateUp)
2017-12-29 17:42:09 +01:00
}
func (device *Device) Down() {
2021-01-19 09:02:16 -08:00
device.changeState(deviceStateDown)
2017-12-29 17:42:09 +01:00
}
2017-08-11 16:18:20 +02:00
func (device *Device) IsUnderLoad() bool {
// check if currently under load
now := time.Now()
2021-01-29 18:24:45 +01:00
underLoad := len(device.queue.handshake.c) >= UnderLoadQueueSize
2017-08-11 16:18:20 +02:00
if underLoad {
atomic.StoreInt64(&device.rate.underLoadUntil, now.Add(UnderLoadAfterTime).UnixNano())
2017-08-11 16:18:20 +02:00
return true
}
// check if recently under load
return atomic.LoadInt64(&device.rate.underLoadUntil) > now.UnixNano()
2017-08-11 16:18:20 +02:00
}
2017-08-04 16:15:53 +02:00
func (device *Device) SetPrivateKey(sk NoisePrivateKey) error {
2018-02-02 16:40:14 +01:00
// lock required resources
device.staticIdentity.Lock()
defer device.staticIdentity.Unlock()
2018-02-02 16:40:14 +01:00
if sk.Equals(device.staticIdentity.privateKey) {
return nil
}
device.peers.Lock()
defer device.peers.Unlock()
2018-02-02 16:40:14 +01:00
2019-08-05 17:46:34 +02:00
lockedPeers := make([]*Peer, 0, len(device.peers.keyMap))
2018-02-02 16:40:14 +01:00
for _, peer := range device.peers.keyMap {
peer.handshake.mutex.RLock()
2019-08-05 17:46:34 +02:00
lockedPeers = append(lockedPeers, peer)
2018-02-02 16:40:14 +01:00
}
2017-06-24 15:34:17 +02:00
2017-08-07 15:25:04 +02:00
// remove peers with matching public keys
2017-08-04 16:15:53 +02:00
publicKey := sk.publicKey()
2018-02-02 16:40:14 +01:00
for key, peer := range device.peers.keyMap {
if peer.handshake.remoteStatic.Equals(publicKey) {
peer.handshake.mutex.RUnlock()
2018-02-02 16:40:14 +01:00
unsafeRemovePeer(device, peer, key)
peer.handshake.mutex.RLock()
2017-08-04 16:15:53 +02:00
}
}
2017-06-24 15:34:17 +02:00
// update key material
2018-05-13 23:14:43 +02:00
device.staticIdentity.privateKey = sk
device.staticIdentity.publicKey = publicKey
device.cookieChecker.Init(publicKey)
2017-06-24 15:34:17 +02:00
2018-02-02 16:40:14 +01:00
// do static-static DH pre-computations
2017-06-24 15:34:17 +02:00
2019-08-05 17:46:34 +02:00
expiredPeers := make([]*Peer, 0, len(device.peers.keyMap))
2020-02-04 18:08:51 +01:00
for _, peer := range device.peers.keyMap {
2018-05-14 12:27:29 +02:00
handshake := &peer.handshake
2020-02-04 18:08:51 +01:00
handshake.precomputedStaticStatic = device.staticIdentity.privateKey.sharedSecret(handshake.remoteStatic)
expiredPeers = append(expiredPeers, peer)
2017-06-23 13:41:59 +02:00
}
2017-08-04 16:15:53 +02:00
2019-08-05 17:46:34 +02:00
for _, peer := range lockedPeers {
peer.handshake.mutex.RUnlock()
}
for _, peer := range expiredPeers {
peer.ExpireCurrentKeypairs()
}
2017-08-04 16:15:53 +02:00
return nil
2017-06-23 13:41:59 +02:00
}
func NewDevice(tunDevice tun.Device, logger *Logger) *Device {
2017-06-28 23:45:45 +02:00
device := new(Device)
2021-01-19 09:02:16 -08:00
device.state.state = uint32(deviceStateDown)
2021-01-29 18:24:45 +01:00
device.closed = make(chan struct{})
2017-11-14 18:26:28 +01:00
device.log = logger
2018-05-23 02:10:54 +02:00
device.tun.device = tunDevice
2018-04-18 16:39:14 +02:00
mtu, err := device.tun.device.MTU()
if err != nil {
device.log.Errorf("Trouble determining MTU, assuming default: %v", err)
2018-04-19 15:52:59 +02:00
mtu = DefaultMTU
2018-04-18 16:39:14 +02:00
}
device.tun.mtu = int32(mtu)
2018-02-02 16:40:14 +01:00
device.peers.keyMap = make(map[NoisePublicKey]*Peer)
device.rate.limiter.Init()
2018-05-13 18:23:40 +02:00
device.indexTable.Init()
2018-09-22 06:29:02 +02:00
device.PopulatePools()
2017-07-14 14:25:18 +02:00
2017-06-30 14:41:08 +02:00
// create queues
2021-01-29 18:24:45 +01:00
device.queue.handshake = newHandshakeQueue()
2021-01-29 14:54:11 +01:00
device.queue.encryption = newOutboundQueue()
device.queue.decryption = newInboundQueue()
2017-07-01 23:29:22 +02:00
2017-11-11 15:43:55 +01:00
// prepare net
device.net.port = 0
device.net.bind = nil
2017-06-28 23:45:45 +02:00
// start workers
2018-05-05 06:00:38 +02:00
cpus := runtime.NumCPU()
2018-05-16 22:20:15 +02:00
device.state.stopping.Wait()
2020-12-22 11:38:24 -08:00
for i := 0; i < cpus; i++ {
2017-06-28 23:45:45 +02:00
go device.RoutineEncryption()
2017-07-01 23:29:22 +02:00
go device.RoutineDecryption()
go device.RoutineHandshake()
2017-06-28 23:45:45 +02:00
}
2017-12-01 23:37:26 +01:00
device.state.stopping.Add(1) // read from TUN
go device.RoutineReadFromTUN()
2017-08-07 15:25:04 +02:00
go device.RoutineTUNEventReader()
2017-12-01 23:37:26 +01:00
2017-06-28 23:45:45 +02:00
return device
2017-06-24 15:34:17 +02:00
}
func (device *Device) LookupPeer(pk NoisePublicKey) *Peer {
device.peers.RLock()
defer device.peers.RUnlock()
2018-02-02 16:40:14 +01:00
return device.peers.keyMap[pk]
2017-06-24 15:34:17 +02:00
}
func (device *Device) RemovePeer(key NoisePublicKey) {
device.peers.Lock()
defer device.peers.Unlock()
2018-02-02 16:40:14 +01:00
// stop peer and remove from routing
peer, ok := device.peers.keyMap[key]
if ok {
unsafeRemovePeer(device, peer, key)
}
2017-06-01 21:31:30 +02:00
}
2017-06-24 15:34:17 +02:00
func (device *Device) RemoveAllPeers() {
device.peers.Lock()
defer device.peers.Unlock()
2018-02-02 16:40:14 +01:00
for key, peer := range device.peers.keyMap {
unsafeRemovePeer(device, peer, key)
2017-06-01 21:31:30 +02:00
}
2018-02-02 16:40:14 +01:00
device.peers.keyMap = make(map[NoisePublicKey]*Peer)
2017-05-30 22:36:49 +02:00
}
2017-06-30 14:41:08 +02:00
func (device *Device) Close() {
device.state.Lock()
defer device.state.Unlock()
2021-01-19 09:02:16 -08:00
if device.isClosed() {
2017-11-17 17:25:45 +01:00
return
}
2021-01-19 09:02:16 -08:00
atomic.StoreUint32(&device.state.state, uint32(deviceStateClosed))
device.log.Verbosef("Device closing")
2017-11-11 23:26:44 +01:00
device.tun.device.Close()
2021-01-19 09:02:16 -08:00
device.downLocked()
// Remove peers before closing queues,
// because peers assume that queues are active.
device.RemoveAllPeers()
// We kept a reference to the encryption and decryption queues,
// in case we started any new peers that might write to them.
// No new peers are coming; we are done with these queues.
device.queue.encryption.wg.Done()
device.queue.decryption.wg.Done()
2021-01-29 18:24:45 +01:00
device.queue.handshake.wg.Done()
device.state.stopping.Wait()
2018-02-11 22:53:39 +01:00
device.rate.limiter.Close()
2021-01-19 09:02:16 -08:00
device.log.Verbosef("Device closed")
2021-01-29 18:24:45 +01:00
close(device.closed)
2017-07-13 14:32:40 +02:00
}
2017-12-01 23:37:26 +01:00
func (device *Device) Wait() chan struct{} {
2021-01-29 18:24:45 +01:00
return device.closed
2017-06-30 14:41:08 +02:00
}
func (device *Device) SendKeepalivesToPeersWithCurrentKeypair() {
2021-01-19 09:02:16 -08:00
if !device.isUp() {
return
}
device.peers.RLock()
for _, peer := range device.peers.keyMap {
peer.keypairs.RLock()
sendKeepalive := peer.keypairs.current != nil && !peer.keypairs.current.created.Add(RejectAfterTime).Before(time.Now())
peer.keypairs.RUnlock()
if sendKeepalive {
peer.SendKeepalive()
}
}
device.peers.RUnlock()
}
func unsafeCloseBind(device *Device) error {
var err error
netc := &device.net
if netc.netlinkCancel != nil {
netc.netlinkCancel.Cancel()
}
if netc.bind != nil {
err = netc.bind.Close()
netc.bind = nil
}
netc.stopping.Wait()
return err
}
func (device *Device) Bind() conn.Bind {
device.net.Lock()
defer device.net.Unlock()
return device.net.bind
}
func (device *Device) BindSetMark(mark uint32) error {
device.net.Lock()
defer device.net.Unlock()
// check if modified
if device.net.fwmark == mark {
return nil
}
// update fwmark on existing bind
device.net.fwmark = mark
2021-01-19 09:02:16 -08:00
if device.isUp() && device.net.bind != nil {
if err := device.net.bind.SetMark(mark); err != nil {
return err
}
}
// clear cached source addresses
device.peers.RLock()
for _, peer := range device.peers.keyMap {
peer.Lock()
defer peer.Unlock()
if peer.endpoint != nil {
peer.endpoint.ClearSrc()
}
}
device.peers.RUnlock()
return nil
}
func (device *Device) BindUpdate() error {
device.net.Lock()
defer device.net.Unlock()
// close existing sockets
if err := unsafeCloseBind(device); err != nil {
return err
}
// open new sockets
2021-01-19 09:02:16 -08:00
if !device.isUp() {
return nil
}
2021-01-19 09:02:16 -08:00
// bind to new port
var err error
netc := &device.net
netc.bind, netc.port, err = conn.CreateBind(netc.port)
if err != nil {
netc.bind = nil
netc.port = 0
return err
}
netc.netlinkCancel, err = device.startRouteListener(netc.bind)
if err != nil {
netc.bind.Close()
netc.bind = nil
netc.port = 0
return err
}
// set fwmark
if netc.fwmark != 0 {
err = netc.bind.SetMark(netc.fwmark)
if err != nil {
return err
}
}
// clear cached source addresses
device.peers.RLock()
for _, peer := range device.peers.keyMap {
peer.Lock()
defer peer.Unlock()
if peer.endpoint != nil {
peer.endpoint.ClearSrc()
}
}
device.peers.RUnlock()
// start receiving routines
device.net.stopping.Add(2)
device.queue.decryption.wg.Add(2) // each RoutineReceiveIncoming goroutine writes to device.queue.decryption
device.queue.handshake.wg.Add(2) // each RoutineReceiveIncoming goroutine writes to device.queue.handshake
go device.RoutineReceiveIncoming(ipv4.Version, netc.bind)
go device.RoutineReceiveIncoming(ipv6.Version, netc.bind)
device.log.Verbosef("UDP bind has been updated")
return nil
}
func (device *Device) BindClose() error {
device.net.Lock()
err := unsafeCloseBind(device)
device.net.Unlock()
return err
}