1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889 |
- package quic
- import (
- "bytes"
- "fmt"
- "net"
- "sync"
- "github.com/lucas-clemente/quic-go/internal/utils"
- )
- var (
- connMuxerOnce sync.Once
- connMuxer multiplexer
- )
- type multiplexer interface {
- AddConn(c net.PacketConn, connIDLen int, statelessResetKey []byte) (packetHandlerManager, error)
- RemoveConn(net.PacketConn) error
- }
- type connManager struct {
- connIDLen int
- statelessResetKey []byte
- manager packetHandlerManager
- }
- // The connMultiplexer listens on multiple net.PacketConns and dispatches
- // incoming packets to the session handler.
- type connMultiplexer struct {
- mutex sync.Mutex
- conns map[net.PacketConn]connManager
- newPacketHandlerManager func(net.PacketConn, int, []byte, utils.Logger) packetHandlerManager // so it can be replaced in the tests
- logger utils.Logger
- }
- var _ multiplexer = &connMultiplexer{}
- func getMultiplexer() multiplexer {
- connMuxerOnce.Do(func() {
- connMuxer = &connMultiplexer{
- conns: make(map[net.PacketConn]connManager),
- logger: utils.DefaultLogger.WithPrefix("muxer"),
- newPacketHandlerManager: newPacketHandlerMap,
- }
- })
- return connMuxer
- }
- func (m *connMultiplexer) AddConn(
- c net.PacketConn,
- connIDLen int,
- statelessResetKey []byte,
- ) (packetHandlerManager, error) {
- m.mutex.Lock()
- defer m.mutex.Unlock()
- p, ok := m.conns[c]
- if !ok {
- manager := m.newPacketHandlerManager(c, connIDLen, statelessResetKey, m.logger)
- p = connManager{
- connIDLen: connIDLen,
- statelessResetKey: statelessResetKey,
- manager: manager,
- }
- m.conns[c] = p
- }
- if p.connIDLen != connIDLen {
- return nil, fmt.Errorf("cannot use %d byte connection IDs on a connection that is already using %d byte connction IDs", connIDLen, p.connIDLen)
- }
- if statelessResetKey != nil && !bytes.Equal(p.statelessResetKey, statelessResetKey) {
- return nil, fmt.Errorf("cannot use different stateless reset keys on the same packet conn")
- }
- return p.manager, nil
- }
- func (m *connMultiplexer) RemoveConn(c net.PacketConn) error {
- m.mutex.Lock()
- defer m.mutex.Unlock()
- if _, ok := m.conns[c]; !ok {
- return fmt.Errorf("cannote remove connection, connection is unknown")
- }
- delete(m.conns, c)
- return nil
- }
|