2023-04-18 14:04:09 +08:00
|
|
|
package libbox
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
|
|
|
"net/netip"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/sagernet/sing-tun"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
|
|
E "github.com/sagernet/sing/common/exceptions"
|
2023-07-23 14:42:19 +08:00
|
|
|
"github.com/sagernet/sing/common/logger"
|
2023-04-18 14:04:09 +08:00
|
|
|
M "github.com/sagernet/sing/common/metadata"
|
|
|
|
"github.com/sagernet/sing/common/x/list"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
_ tun.DefaultInterfaceMonitor = (*platformDefaultInterfaceMonitor)(nil)
|
|
|
|
_ InterfaceUpdateListener = (*platformDefaultInterfaceMonitor)(nil)
|
|
|
|
)
|
|
|
|
|
|
|
|
type platformDefaultInterfaceMonitor struct {
|
|
|
|
*platformInterfaceWrapper
|
|
|
|
networkAddresses []networkAddress
|
|
|
|
defaultInterfaceName string
|
|
|
|
defaultInterfaceIndex int
|
|
|
|
element *list.Element[tun.NetworkUpdateCallback]
|
|
|
|
access sync.Mutex
|
|
|
|
callbacks list.List[tun.DefaultInterfaceUpdateCallback]
|
2023-07-23 14:42:19 +08:00
|
|
|
logger logger.Logger
|
2023-04-18 14:04:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type networkAddress struct {
|
|
|
|
interfaceName string
|
|
|
|
interfaceIndex int
|
|
|
|
addresses []netip.Prefix
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) Start() error {
|
|
|
|
return m.iif.StartDefaultInterfaceMonitor(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) Close() error {
|
|
|
|
return m.iif.CloseDefaultInterfaceMonitor(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) DefaultInterfaceName(destination netip.Addr) string {
|
|
|
|
for _, address := range m.networkAddresses {
|
|
|
|
for _, prefix := range address.addresses {
|
|
|
|
if prefix.Contains(destination) {
|
|
|
|
return address.interfaceName
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return m.defaultInterfaceName
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) DefaultInterfaceIndex(destination netip.Addr) int {
|
|
|
|
for _, address := range m.networkAddresses {
|
|
|
|
for _, prefix := range address.addresses {
|
|
|
|
if prefix.Contains(destination) {
|
|
|
|
return address.interfaceIndex
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return m.defaultInterfaceIndex
|
|
|
|
}
|
|
|
|
|
2023-10-21 12:00:00 +08:00
|
|
|
func (m *platformDefaultInterfaceMonitor) DefaultInterface(destination netip.Addr) (string, int) {
|
|
|
|
for _, address := range m.networkAddresses {
|
|
|
|
for _, prefix := range address.addresses {
|
|
|
|
if prefix.Contains(destination) {
|
|
|
|
return address.interfaceName, address.interfaceIndex
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return m.defaultInterfaceName, m.defaultInterfaceIndex
|
|
|
|
}
|
|
|
|
|
2023-04-18 14:04:09 +08:00
|
|
|
func (m *platformDefaultInterfaceMonitor) OverrideAndroidVPN() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) AndroidVPNEnabled() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) RegisterCallback(callback tun.DefaultInterfaceUpdateCallback) *list.Element[tun.DefaultInterfaceUpdateCallback] {
|
|
|
|
m.access.Lock()
|
|
|
|
defer m.access.Unlock()
|
|
|
|
return m.callbacks.PushBack(callback)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) UnregisterCallback(element *list.Element[tun.DefaultInterfaceUpdateCallback]) {
|
|
|
|
m.access.Lock()
|
|
|
|
defer m.access.Unlock()
|
|
|
|
m.callbacks.Remove(element)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) UpdateDefaultInterface(interfaceName string, interfaceIndex32 int32) {
|
2024-02-02 17:51:33 +08:00
|
|
|
if interfaceName == "" || interfaceIndex32 == -1 {
|
|
|
|
m.defaultInterfaceName = ""
|
|
|
|
m.defaultInterfaceIndex = -1
|
|
|
|
m.access.Lock()
|
|
|
|
callbacks := m.callbacks.Array()
|
|
|
|
m.access.Unlock()
|
|
|
|
for _, callback := range callbacks {
|
|
|
|
callback(tun.EventNoRoute)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2023-04-18 14:04:09 +08:00
|
|
|
var err error
|
|
|
|
if m.iif.UsePlatformInterfaceGetter() {
|
|
|
|
err = m.updateInterfacesPlatform()
|
|
|
|
} else {
|
|
|
|
err = m.updateInterfaces()
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
err = m.router.UpdateInterfaces()
|
|
|
|
}
|
|
|
|
if err != nil {
|
2023-07-23 14:42:19 +08:00
|
|
|
m.logger.Error(E.Cause(err, "update interfaces"))
|
2023-04-18 14:04:09 +08:00
|
|
|
}
|
|
|
|
interfaceIndex := int(interfaceIndex32)
|
|
|
|
if m.defaultInterfaceName == interfaceName && m.defaultInterfaceIndex == interfaceIndex {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
m.defaultInterfaceName = interfaceName
|
|
|
|
m.defaultInterfaceIndex = interfaceIndex
|
|
|
|
m.access.Lock()
|
|
|
|
callbacks := m.callbacks.Array()
|
|
|
|
m.access.Unlock()
|
|
|
|
for _, callback := range callbacks {
|
2023-07-23 14:42:19 +08:00
|
|
|
callback(tun.EventInterfaceUpdate)
|
2023-04-18 14:04:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) updateInterfaces() error {
|
|
|
|
interfaces, err := net.Interfaces()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var addresses []networkAddress
|
|
|
|
for _, iif := range interfaces {
|
|
|
|
var netAddresses []net.Addr
|
|
|
|
netAddresses, err = iif.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var address networkAddress
|
|
|
|
address.interfaceName = iif.Name
|
|
|
|
address.interfaceIndex = iif.Index
|
|
|
|
address.addresses = common.Map(common.FilterIsInstance(netAddresses, func(it net.Addr) (*net.IPNet, bool) {
|
|
|
|
value, loaded := it.(*net.IPNet)
|
|
|
|
return value, loaded
|
|
|
|
}), func(it *net.IPNet) netip.Prefix {
|
|
|
|
bits, _ := it.Mask.Size()
|
|
|
|
return netip.PrefixFrom(M.AddrFromIP(it.IP), bits)
|
|
|
|
})
|
|
|
|
addresses = append(addresses, address)
|
|
|
|
}
|
|
|
|
m.networkAddresses = addresses
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *platformDefaultInterfaceMonitor) updateInterfacesPlatform() error {
|
|
|
|
interfaces, err := m.Interfaces()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var addresses []networkAddress
|
|
|
|
for _, iif := range interfaces {
|
|
|
|
var address networkAddress
|
|
|
|
address.interfaceName = iif.Name
|
|
|
|
address.interfaceIndex = iif.Index
|
|
|
|
// address.addresses = common.Map(iif.Addresses, netip.MustParsePrefix)
|
|
|
|
addresses = append(addresses, address)
|
|
|
|
}
|
|
|
|
m.networkAddresses = addresses
|
|
|
|
return nil
|
|
|
|
}
|