| 
									
										
										
										
											2023-01-27 13:37:20 -08:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Package wgint provides somewhat shady access to wireguard-go | 
					
						
							|  |  |  | // internals that don't (yet) have public APIs. | 
					
						
							|  |  |  | package wgint | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"reflect" | 
					
						
							|  |  |  | 	"sync/atomic" | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | 	"unsafe" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-09 15:12:20 -08:00
										 |  |  | 	"github.com/tailscale/wireguard-go/device" | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	offHandshake = getPeerStatsOffset("lastHandshakeNano") | 
					
						
							|  |  |  | 	offRxBytes   = getPeerStatsOffset("rxBytes") | 
					
						
							|  |  |  | 	offTxBytes   = getPeerStatsOffset("txBytes") | 
					
						
							| 
									
										
										
										
											2024-02-25 06:40:35 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	offHandshakeAttempts = getPeerHandshakeAttemptsOffset() | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func getPeerStatsOffset(name string) uintptr { | 
					
						
							| 
									
										
										
										
											2024-02-08 17:34:22 -08:00
										 |  |  | 	peerType := reflect.TypeFor[device.Peer]() | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	field, ok := peerType.FieldByName(name) | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | 	if !ok { | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 		panic("no " + name + " field in device.Peer") | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	if s := field.Type.String(); s != "atomic.Int64" && s != "atomic.Uint64" { | 
					
						
							|  |  |  | 		panic("unexpected type " + s + " of field " + name + " in device.Peer") | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	return field.Offset | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 06:40:35 -08:00
										 |  |  | func getPeerHandshakeAttemptsOffset() uintptr { | 
					
						
							|  |  |  | 	peerType := reflect.TypeFor[device.Peer]() | 
					
						
							|  |  |  | 	field, ok := peerType.FieldByName("timers") | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		panic("no timers field in device.Peer") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	field2, ok := field.Type.FieldByName("handshakeAttempts") | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		panic("no handshakeAttempts field in device.Peer.timers") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if g, w := field2.Type.String(), "atomic.Uint32"; g != w { | 
					
						
							|  |  |  | 		panic("unexpected type " + g + " of field handshakeAttempts in device.Peer.timers; want " + w) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return field.Offset + field2.Offset | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | // peerLastHandshakeNano returns the last handshake time in nanoseconds since the | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | // unix epoch. | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | func peerLastHandshakeNano(peer *device.Peer) int64 { | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	return (*atomic.Int64)(unsafe.Add(unsafe.Pointer(peer), offHandshake)).Load() | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | // peerRxBytes returns the number of bytes received from this peer. | 
					
						
							|  |  |  | func peerRxBytes(peer *device.Peer) uint64 { | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	return (*atomic.Uint64)(unsafe.Add(unsafe.Pointer(peer), offRxBytes)).Load() | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | // peerTxBytes returns the number of bytes sent to this peer. | 
					
						
							|  |  |  | func peerTxBytes(peer *device.Peer) uint64 { | 
					
						
							| 
									
										
										
										
											2022-09-06 11:21:30 -07:00
										 |  |  | 	return (*atomic.Uint64)(unsafe.Add(unsafe.Pointer(peer), offTxBytes)).Load() | 
					
						
							| 
									
										
										
										
											2022-08-29 09:46:26 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-02-25 06:40:35 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | // peerHandshakeAttempts returns the number of WireGuard handshake attempts | 
					
						
							| 
									
										
										
										
											2024-02-25 06:40:35 -08:00
										 |  |  | // made for the current handshake. It resets to zero before every new handshake. | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | func peerHandshakeAttempts(peer *device.Peer) uint32 { | 
					
						
							| 
									
										
										
										
											2024-02-25 06:40:35 -08:00
										 |  |  | 	return (*atomic.Uint32)(unsafe.Add(unsafe.Pointer(peer), offHandshakeAttempts)).Load() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-02-28 07:33:15 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Peer is a wrapper around a wireguard-go device.Peer pointer. | 
					
						
							|  |  |  | type Peer struct { | 
					
						
							|  |  |  | 	p *device.Peer | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // PeerOf returns a Peer wrapper around a wireguard-go device.Peer. | 
					
						
							|  |  |  | func PeerOf(p *device.Peer) Peer { | 
					
						
							|  |  |  | 	return Peer{p} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // LastHandshake returns the last handshake time. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // If the handshake has never happened, it returns the zero value. | 
					
						
							|  |  |  | func (p Peer) LastHandshake() time.Time { | 
					
						
							|  |  |  | 	if n := peerLastHandshakeNano(p.p); n != 0 { | 
					
						
							|  |  |  | 		return time.Unix(0, n) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return time.Time{} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p Peer) IsValid() bool { return p.p != nil } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TxBytes returns the number of bytes sent to this peer. | 
					
						
							|  |  |  | func (p Peer) TxBytes() uint64 { return peerTxBytes(p.p) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RxBytes returns the number of bytes received from this peer. | 
					
						
							|  |  |  | func (p Peer) RxBytes() uint64 { return peerRxBytes(p.p) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // HandshakeAttempts returns the number of failed WireGuard handshake attempts | 
					
						
							|  |  |  | // made for the current handshake. It resets to zero before every new handshake | 
					
						
							|  |  |  | // and after a successful handshake. | 
					
						
							|  |  |  | func (p Peer) HandshakeAttempts() uint32 { | 
					
						
							|  |  |  | 	return peerHandshakeAttempts(p.p) | 
					
						
							|  |  |  | } |