mirror of
				https://git.zx2c4.com/wireguard-go
				synced 2025-10-31 20:02:37 +08:00 
			
		
		
		
	 0ad14a89f5
			
		
	
	0ad14a89f5
	
	
	
		
			
			This always struck me as kind of weird and non-standard. Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
		
			
				
	
	
		
			130 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			130 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /* SPDX-License-Identifier: MIT
 | |
|  *
 | |
|  * Copyright (C) 2017-2023 WireGuard LLC. All Rights Reserved.
 | |
|  */
 | |
| 
 | |
| package ipc
 | |
| 
 | |
| import (
 | |
| 	"net"
 | |
| 	"os"
 | |
| 
 | |
| 	"golang.org/x/sys/unix"
 | |
| 	"golang.zx2c4.com/wireguard/rwcancel"
 | |
| )
 | |
| 
 | |
| type UAPIListener struct {
 | |
| 	listener        net.Listener // unix socket listener
 | |
| 	connNew         chan net.Conn
 | |
| 	connErr         chan error
 | |
| 	inotifyFd       int
 | |
| 	inotifyRWCancel *rwcancel.RWCancel
 | |
| }
 | |
| 
 | |
| func (l *UAPIListener) Accept() (net.Conn, error) {
 | |
| 	for {
 | |
| 		select {
 | |
| 		case conn := <-l.connNew:
 | |
| 			return conn, nil
 | |
| 
 | |
| 		case err := <-l.connErr:
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l *UAPIListener) Close() error {
 | |
| 	err1 := unix.Close(l.inotifyFd)
 | |
| 	err2 := l.inotifyRWCancel.Cancel()
 | |
| 	err3 := l.listener.Close()
 | |
| 	if err1 != nil {
 | |
| 		return err1
 | |
| 	}
 | |
| 	if err2 != nil {
 | |
| 		return err2
 | |
| 	}
 | |
| 	return err3
 | |
| }
 | |
| 
 | |
| func (l *UAPIListener) Addr() net.Addr {
 | |
| 	return l.listener.Addr()
 | |
| }
 | |
| 
 | |
| func UAPIListen(name string, file *os.File) (net.Listener, error) {
 | |
| 	// wrap file in listener
 | |
| 
 | |
| 	listener, err := net.FileListener(file)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	if unixListener, ok := listener.(*net.UnixListener); ok {
 | |
| 		unixListener.SetUnlinkOnClose(true)
 | |
| 	}
 | |
| 
 | |
| 	uapi := &UAPIListener{
 | |
| 		listener: listener,
 | |
| 		connNew:  make(chan net.Conn, 1),
 | |
| 		connErr:  make(chan error, 1),
 | |
| 	}
 | |
| 
 | |
| 	// watch for deletion of socket
 | |
| 
 | |
| 	socketPath := sockPath(name)
 | |
| 
 | |
| 	uapi.inotifyFd, err = unix.InotifyInit()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	_, err = unix.InotifyAddWatch(
 | |
| 		uapi.inotifyFd,
 | |
| 		socketPath,
 | |
| 		unix.IN_ATTRIB|
 | |
| 			unix.IN_DELETE|
 | |
| 			unix.IN_DELETE_SELF,
 | |
| 	)
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	uapi.inotifyRWCancel, err = rwcancel.NewRWCancel(uapi.inotifyFd)
 | |
| 	if err != nil {
 | |
| 		unix.Close(uapi.inotifyFd)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	go func(l *UAPIListener) {
 | |
| 		var buf [0]byte
 | |
| 		for {
 | |
| 			defer uapi.inotifyRWCancel.Close()
 | |
| 			// start with lstat to avoid race condition
 | |
| 			if _, err := os.Lstat(socketPath); os.IsNotExist(err) {
 | |
| 				l.connErr <- err
 | |
| 				return
 | |
| 			}
 | |
| 			_, err := uapi.inotifyRWCancel.Read(buf[:])
 | |
| 			if err != nil {
 | |
| 				l.connErr <- err
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 	}(uapi)
 | |
| 
 | |
| 	// watch for new connections
 | |
| 
 | |
| 	go func(l *UAPIListener) {
 | |
| 		for {
 | |
| 			conn, err := l.listener.Accept()
 | |
| 			if err != nil {
 | |
| 				l.connErr <- err
 | |
| 				break
 | |
| 			}
 | |
| 			l.connNew <- conn
 | |
| 		}
 | |
| 	}(uapi)
 | |
| 
 | |
| 	return uapi, nil
 | |
| }
 |