golanggohlsrtmpwebrtcmedia-serverobs-studiortcprtmp-proxyrtmp-serverrtprtsprtsp-proxyrtsp-relayrtsp-serversrtstreamingwebrtc-proxy
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
3.4 KiB
175 lines
3.4 KiB
package serverudp |
|
|
|
import ( |
|
"net" |
|
"sync" |
|
"time" |
|
|
|
"github.com/aler9/gortsplib" |
|
"github.com/aler9/gortsplib/pkg/base" |
|
"github.com/aler9/gortsplib/pkg/multibuffer" |
|
) |
|
|
|
const ( |
|
// use the same buffer size as gstreamer's rtspsrc |
|
kernelReadBufferSize = 0x80000 |
|
|
|
readBufferSize = 2048 |
|
) |
|
|
|
// Publisher is implemented by client.Client. |
|
type Publisher interface { |
|
OnUdpPublisherFrame(int, base.StreamType, []byte) |
|
} |
|
|
|
type publisherData struct { |
|
publisher Publisher |
|
trackId int |
|
} |
|
|
|
// Parent is implemented by program. |
|
type Parent interface { |
|
Log(string, ...interface{}) |
|
} |
|
|
|
type publisherAddr struct { |
|
ip [net.IPv6len]byte // use a fixed-size array to enable the equality operator |
|
port int |
|
} |
|
|
|
func (p *publisherAddr) fill(ip net.IP, port int) { |
|
p.port = port |
|
|
|
if len(ip) == net.IPv4len { |
|
copy(p.ip[0:], []byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff}) // v4InV6Prefix |
|
copy(p.ip[12:], ip) |
|
} else { |
|
copy(p.ip[:], ip) |
|
} |
|
} |
|
|
|
// Server is a RTSP UDP server. |
|
type Server struct { |
|
writeTimeout time.Duration |
|
streamType gortsplib.StreamType |
|
|
|
pc *net.UDPConn |
|
readBuf *multibuffer.MultiBuffer |
|
publishersMutex sync.RWMutex |
|
publishers map[publisherAddr]*publisherData |
|
writeMutex sync.Mutex |
|
|
|
// out |
|
done chan struct{} |
|
} |
|
|
|
// New allocates a Server. |
|
func New(writeTimeout time.Duration, |
|
port int, |
|
streamType gortsplib.StreamType, |
|
parent Parent) (*Server, error) { |
|
|
|
pc, err := net.ListenUDP("udp", &net.UDPAddr{ |
|
Port: port, |
|
}) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
err = pc.SetReadBuffer(kernelReadBufferSize) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
s := &Server{ |
|
writeTimeout: writeTimeout, |
|
streamType: streamType, |
|
pc: pc, |
|
readBuf: multibuffer.New(1, readBufferSize), |
|
publishers: make(map[publisherAddr]*publisherData), |
|
done: make(chan struct{}), |
|
} |
|
|
|
var label string |
|
if s.streamType == gortsplib.StreamTypeRtp { |
|
label = "RTP" |
|
} else { |
|
label = "RTCP" |
|
} |
|
parent.Log("[UDP/"+label+" server] opened on :%d", port) |
|
|
|
go s.run() |
|
return s, nil |
|
} |
|
|
|
// Close closes a Server. |
|
func (s *Server) Close() { |
|
s.pc.Close() |
|
<-s.done |
|
} |
|
|
|
func (s *Server) run() { |
|
defer close(s.done) |
|
|
|
for { |
|
buf := s.readBuf.Next() |
|
n, addr, err := s.pc.ReadFromUDP(buf) |
|
if err != nil { |
|
break |
|
} |
|
|
|
func() { |
|
s.publishersMutex.RLock() |
|
defer s.publishersMutex.RUnlock() |
|
|
|
// find publisher data |
|
var pubAddr publisherAddr |
|
pubAddr.fill(addr.IP, addr.Port) |
|
pubData, ok := s.publishers[pubAddr] |
|
if !ok { |
|
return |
|
} |
|
|
|
pubData.publisher.OnUdpPublisherFrame(pubData.trackId, s.streamType, buf[:n]) |
|
}() |
|
} |
|
} |
|
|
|
// Port returns the server local port. |
|
func (s *Server) Port() int { |
|
return s.pc.LocalAddr().(*net.UDPAddr).Port |
|
} |
|
|
|
// Write writes a UDP packet. |
|
func (s *Server) Write(buf []byte, addr *net.UDPAddr) { |
|
s.writeMutex.Lock() |
|
defer s.writeMutex.Unlock() |
|
|
|
s.pc.SetWriteDeadline(time.Now().Add(s.writeTimeout)) |
|
s.pc.WriteTo(buf, addr) |
|
} |
|
|
|
// AddPublisher adds a publisher. |
|
func (s *Server) AddPublisher(ip net.IP, port int, publisher Publisher, trackId int) { |
|
s.publishersMutex.Lock() |
|
defer s.publishersMutex.Unlock() |
|
|
|
var addr publisherAddr |
|
addr.fill(ip, port) |
|
|
|
s.publishers[addr] = &publisherData{ |
|
publisher: publisher, |
|
trackId: trackId, |
|
} |
|
} |
|
|
|
// RemovePublisher removes a publisher. |
|
func (s *Server) RemovePublisher(ip net.IP, port int, publisher Publisher) { |
|
s.publishersMutex.Lock() |
|
defer s.publishersMutex.Unlock() |
|
|
|
var addr publisherAddr |
|
addr.fill(ip, port) |
|
|
|
delete(s.publishers, addr) |
|
}
|
|
|