mirror of
https://github.com/tailscale/tailscale.git
synced 2025-10-24 09:39:39 +00:00
fix port mapping (w/ maisem + andrew)
Change-Id: I703b39f05af2e3e1a979be8e77091586cb9ec3eb Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
@@ -91,6 +91,13 @@ func easy(c *vnet.Config) *vnet.Node {
|
|||||||
fmt.Sprintf("192.168.%d.1/24", n), vnet.EasyNAT))
|
fmt.Sprintf("192.168.%d.1/24", n), vnet.EasyNAT))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func easyPMP(c *vnet.Config) *vnet.Node {
|
||||||
|
n := c.NumNodes() + 1
|
||||||
|
return c.AddNode(c.AddNetwork(
|
||||||
|
fmt.Sprintf("2.%d.%d.%d", n, n, n), // public IP
|
||||||
|
fmt.Sprintf("192.168.%d.1/24", n), vnet.EasyNAT, vnet.NATPMP))
|
||||||
|
}
|
||||||
|
|
||||||
func hard(c *vnet.Config) *vnet.Node {
|
func hard(c *vnet.Config) *vnet.Node {
|
||||||
n := c.NumNodes() + 1
|
n := c.NumNodes() + 1
|
||||||
return c.AddNode(c.AddNetwork(
|
return c.AddNode(c.AddNetwork(
|
||||||
@@ -158,7 +165,7 @@ func (nt *natTest) runTest(node1, node2 addNodeFunc) {
|
|||||||
|
|
||||||
cmd := exec.Command("qemu-system-x86_64",
|
cmd := exec.Command("qemu-system-x86_64",
|
||||||
"-M", "microvm,isa-serial=off",
|
"-M", "microvm,isa-serial=off",
|
||||||
"-m", "1G",
|
"-m", "384M",
|
||||||
"-nodefaults", "-no-user-config", "-nographic",
|
"-nodefaults", "-no-user-config", "-nographic",
|
||||||
"-kernel", nt.kernel,
|
"-kernel", nt.kernel,
|
||||||
"-append", "console=hvc0 root=PARTUUID=60c24cc1-f3f9-427a-8199-dd02023b0001/PARTNROFF=1 ro init=/gokrazy/init panic=10 oops=panic pci=off nousb tsc=unstable clocksource=hpet tailscale-tta=1",
|
"-append", "console=hvc0 root=PARTUUID=60c24cc1-f3f9-427a-8199-dd02023b0001/PARTNROFF=1 ro init=/gokrazy/init panic=10 oops=panic pci=off nousb tsc=unstable clocksource=hpet tailscale-tta=1",
|
||||||
@@ -249,7 +256,7 @@ func streamDaemonLogs(ctx context.Context, t testing.TB, c *vnet.NodeAgentClient
|
|||||||
Text string `json:"text"`
|
Text string `json:"text"`
|
||||||
}
|
}
|
||||||
if err := dec.Decode(&logEntry); err != nil {
|
if err := dec.Decode(&logEntry); err != nil {
|
||||||
if err == io.EOF {
|
if err == io.EOF || errors.Is(err, context.Canceled) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
t.Errorf("log entry: %v", err)
|
t.Errorf("log entry: %v", err)
|
||||||
@@ -321,3 +328,8 @@ func TestEasyHardPMP(t *testing.T) {
|
|||||||
nt := newNatTest(t)
|
nt := newNatTest(t)
|
||||||
nt.runTest(easy, hardPMP)
|
nt.runTest(easy, hardPMP)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestEasyPMPHard(t *testing.T) {
|
||||||
|
nt := newNatTest(t)
|
||||||
|
nt.runTest(easyPMP, hard)
|
||||||
|
}
|
||||||
|
@@ -412,10 +412,11 @@ type network struct {
|
|||||||
ns *stack.Stack
|
ns *stack.Stack
|
||||||
linkEP *channel.Endpoint
|
linkEP *channel.Endpoint
|
||||||
|
|
||||||
natStyle syncs.AtomicValue[NAT]
|
natStyle syncs.AtomicValue[NAT]
|
||||||
natMu sync.Mutex // held while using + changing natTable
|
natMu sync.Mutex // held while using + changing natTable
|
||||||
natTable NATTable
|
natTable NATTable
|
||||||
portMap map[netip.AddrPort]portMapping // WAN ip:port -> LAN ip:port
|
portMap map[netip.AddrPort]portMapping // WAN ip:port -> LAN ip:port
|
||||||
|
portMapFlow map[portmapFlowKey]netip.AddrPort // (lanAP, peerWANAP) -> portmapped wanAP
|
||||||
|
|
||||||
// writeFunc is a map of MAC -> func to write to that MAC.
|
// writeFunc is a map of MAC -> func to write to that MAC.
|
||||||
// It contains entries for connected nodes only.
|
// It contains entries for connected nodes only.
|
||||||
@@ -1197,13 +1198,27 @@ func (s *Server) createDNSResponse(pkt gopacket.Packet) ([]byte, error) {
|
|||||||
// doNATOut performs NAT on an outgoing packet from src to dst, where
|
// doNATOut performs NAT on an outgoing packet from src to dst, where
|
||||||
// src is a LAN IP and dst is a WAN IP.
|
// src is a LAN IP and dst is a WAN IP.
|
||||||
//
|
//
|
||||||
// It returns the souce WAN ip:port to use.
|
// It returns the source WAN ip:port to use.
|
||||||
func (n *network) doNATOut(src, dst netip.AddrPort) (newSrc netip.AddrPort) {
|
func (n *network) doNATOut(src, dst netip.AddrPort) (newSrc netip.AddrPort) {
|
||||||
n.natMu.Lock()
|
n.natMu.Lock()
|
||||||
defer n.natMu.Unlock()
|
defer n.natMu.Unlock()
|
||||||
|
|
||||||
|
// First see if there's a port mapping, before doing NAT.
|
||||||
|
if wanAP, ok := n.portMapFlow[portmapFlowKey{
|
||||||
|
peerWAN: dst,
|
||||||
|
lanAP: src,
|
||||||
|
}]; ok {
|
||||||
|
return wanAP
|
||||||
|
}
|
||||||
|
|
||||||
return n.natTable.PickOutgoingSrc(src, dst, time.Now())
|
return n.natTable.PickOutgoingSrc(src, dst, time.Now())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type portmapFlowKey struct {
|
||||||
|
peerWAN netip.AddrPort // the peer's WAN ip:port
|
||||||
|
lanAP netip.AddrPort
|
||||||
|
}
|
||||||
|
|
||||||
// doNATIn performs NAT on an incoming packet from WAN src to WAN dst, returning
|
// doNATIn performs NAT on an incoming packet from WAN src to WAN dst, returning
|
||||||
// a new destination LAN ip:port to use.
|
// a new destination LAN ip:port to use.
|
||||||
func (n *network) doNATIn(src, dst netip.AddrPort) (newDst netip.AddrPort) {
|
func (n *network) doNATIn(src, dst netip.AddrPort) (newDst netip.AddrPort) {
|
||||||
@@ -1215,6 +1230,10 @@ func (n *network) doNATIn(src, dst netip.AddrPort) (newDst netip.AddrPort) {
|
|||||||
// First see if there's a port mapping, before doing NAT.
|
// First see if there's a port mapping, before doing NAT.
|
||||||
if lanAP, ok := n.portMap[dst]; ok {
|
if lanAP, ok := n.portMap[dst]; ok {
|
||||||
if now.Before(lanAP.expiry) {
|
if now.Before(lanAP.expiry) {
|
||||||
|
mak.Set(&n.portMapFlow, portmapFlowKey{
|
||||||
|
peerWAN: src,
|
||||||
|
lanAP: lanAP.dst,
|
||||||
|
}, dst)
|
||||||
n.logf("XXX NAT: doNatIn: port mapping %v=>%v", dst, lanAP.dst)
|
n.logf("XXX NAT: doNatIn: port mapping %v=>%v", dst, lanAP.dst)
|
||||||
return lanAP.dst
|
return lanAP.dst
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user