2021-08-03 15:28:13 -07:00
// Copyright (c) 2021 Tailscale Inc & AUTHORS All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package portmapper
import (
2021-08-03 15:29:53 -07:00
"context"
2021-08-03 15:28:13 -07:00
"crypto/rand"
"encoding/binary"
2021-08-03 15:29:53 -07:00
"fmt"
"time"
2021-08-03 15:28:13 -07:00
"inet.af/netaddr"
)
// References:
//
// https://www.rfc-editor.org/rfc/pdfrfc/rfc6887.txt.pdf
// https://tools.ietf.org/html/rfc6887
2021-11-23 14:43:04 -08:00
//go:generate go run tailscale.com/cmd/addlicense -year 2021 -file pcpresultcode_string.go go run golang.org/x/tools/cmd/stringer -type=pcpResultCode -trimprefix=pcpCode
type pcpResultCode uint8
2021-08-03 15:28:13 -07:00
// PCP constants
const (
2021-08-09 12:52:15 -07:00
pcpVersion = 2
pcpDefaultPort = 5351
2021-08-03 15:28:13 -07:00
pcpMapLifetimeSec = 7200 // TODO does the RFC recommend anything? This is taken from PMP.
2021-11-23 14:43:04 -08:00
pcpCodeOK pcpResultCode = 0
pcpCodeNotAuthorized pcpResultCode = 2
2021-11-23 13:49:47 -08:00
// From RFC 6887:
// ADDRESS_MISMATCH: The source IP address of the request packet does
// not match the contents of the PCP Client's IP Address field, due
// to an unexpected NAT on the path between the PCP client and the
// PCP-controlled NAT or firewall.
2021-11-23 14:43:04 -08:00
pcpCodeAddressMismatch pcpResultCode = 12
2021-08-03 15:28:13 -07:00
pcpOpReply = 0x80 // OR'd into request's op code on response
pcpOpAnnounce = 0
pcpOpMap = 1
pcpUDPMapping = 17 // portmap UDP
pcpTCPMapping = 6 // portmap TCP
)
2021-08-03 15:29:53 -07:00
type pcpMapping struct {
2021-08-09 12:52:15 -07:00
c * Client
gw netaddr . IPPort
2021-08-03 15:29:53 -07:00
internal netaddr . IPPort
external netaddr . IPPort
renewAfter time . Time
goodUntil time . Time
2021-08-04 16:51:10 -07:00
// TODO should this also contain an epoch?
// Doesn't seem to be used elsewhere, but can use it for validation at some point.
2021-08-03 15:29:53 -07:00
}
2021-08-03 15:28:13 -07:00
2021-08-03 15:29:53 -07:00
func ( p * pcpMapping ) GoodUntil ( ) time . Time { return p . goodUntil }
func ( p * pcpMapping ) RenewAfter ( ) time . Time { return p . renewAfter }
func ( p * pcpMapping ) External ( ) netaddr . IPPort { return p . external }
func ( p * pcpMapping ) Release ( ctx context . Context ) {
2021-08-09 12:52:15 -07:00
uc , err := p . c . listenPacket ( ctx , "udp4" , ":0" )
2021-08-03 15:29:53 -07:00
if err != nil {
return
}
defer uc . Close ( )
2021-08-05 12:33:13 -07:00
pkt := buildPCPRequestMappingPacket ( p . internal . IP ( ) , p . internal . Port ( ) , p . external . Port ( ) , 0 , p . external . IP ( ) )
2021-08-09 12:52:15 -07:00
uc . WriteTo ( pkt , p . gw . UDPAddr ( ) )
2021-08-03 15:29:53 -07:00
}
2021-08-03 15:28:13 -07:00
2021-08-03 15:29:53 -07:00
// buildPCPRequestMappingPacket generates a PCP packet with a MAP opcode.
// To create a packet which deletes a mapping, lifetimeSec should be set to 0.
// If prevPort is not known, it should be set to 0.
2021-08-05 12:33:13 -07:00
// If prevExternalIP is not known, it should be set to 0.0.0.0.
func buildPCPRequestMappingPacket (
myIP netaddr . IP ,
localPort , prevPort uint16 ,
lifetimeSec uint32 ,
prevExternalIP netaddr . IP ,
) ( pkt [ ] byte ) {
2021-08-03 15:29:53 -07:00
// 24 byte common PCP header + 36 bytes of MAP-specific fields
pkt = make ( [ ] byte , 24 + 36 )
pkt [ 0 ] = pcpVersion
pkt [ 1 ] = pcpOpMap
binary . BigEndian . PutUint32 ( pkt [ 4 : 8 ] , lifetimeSec )
2021-08-03 15:28:13 -07:00
myIP16 := myIP . As16 ( )
2021-08-03 15:29:53 -07:00
copy ( pkt [ 8 : 24 ] , myIP16 [ : ] )
2021-08-03 15:28:13 -07:00
mapOp := pkt [ 24 : ]
2021-08-03 15:29:53 -07:00
rand . Read ( mapOp [ : 12 ] ) // 96 bit mapping nonce
2021-08-04 16:51:10 -07:00
// TODO: should this be a UDP mapping? It looks like it supports "all protocols" with 0, but
2021-08-03 15:29:53 -07:00
// also doesn't support a local port then.
mapOp [ 12 ] = pcpUDPMapping
binary . BigEndian . PutUint16 ( mapOp [ 16 : 18 ] , localPort )
binary . BigEndian . PutUint16 ( mapOp [ 18 : 20 ] , prevPort )
2021-08-05 12:33:13 -07:00
prevExternalIP16 := prevExternalIP . As16 ( )
copy ( mapOp [ 20 : ] , prevExternalIP16 [ : ] )
2021-08-03 15:28:13 -07:00
return pkt
}
2021-08-09 12:52:15 -07:00
// parsePCPMapResponse parses resp into a partially populated pcpMapping.
// In particular, its Client is not populated.
2021-08-03 15:29:53 -07:00
func parsePCPMapResponse ( resp [ ] byte ) ( * pcpMapping , error ) {
if len ( resp ) < 60 {
return nil , fmt . Errorf ( "Does not appear to be PCP MAP response" )
}
res , ok := parsePCPResponse ( resp [ : 24 ] )
if ! ok {
return nil , fmt . Errorf ( "Invalid PCP common header" )
}
if res . ResultCode != pcpCodeOK {
return nil , fmt . Errorf ( "PCP response not ok, code %d" , res . ResultCode )
}
2021-08-04 16:51:10 -07:00
// TODO: don't ignore the nonce and make sure it's the same?
2021-08-03 15:29:53 -07:00
externalPort := binary . BigEndian . Uint16 ( resp [ 42 : 44 ] )
externalIPBytes := [ 16 ] byte { }
copy ( externalIPBytes [ : ] , resp [ 44 : ] )
externalIP := netaddr . IPFrom16 ( externalIPBytes )
external := netaddr . IPPortFrom ( externalIP , externalPort )
lifetime := time . Second * time . Duration ( res . Lifetime )
now := time . Now ( )
mapping := & pcpMapping {
external : external ,
renewAfter : now . Add ( lifetime / 2 ) ,
goodUntil : now . Add ( lifetime ) ,
}
return mapping , nil
}
2021-08-03 15:28:13 -07:00
// pcpAnnounceRequest generates a PCP packet with an ANNOUNCE opcode.
func pcpAnnounceRequest ( myIP netaddr . IP ) [ ] byte {
// See https://tools.ietf.org/html/rfc6887#section-7.1
pkt := make ( [ ] byte , 24 )
2021-08-03 15:29:53 -07:00
pkt [ 0 ] = pcpVersion
2021-08-03 15:28:13 -07:00
pkt [ 1 ] = pcpOpAnnounce
myIP16 := myIP . As16 ( )
copy ( pkt [ 8 : ] , myIP16 [ : ] )
return pkt
}
type pcpResponse struct {
OpCode uint8
2021-11-23 14:43:04 -08:00
ResultCode pcpResultCode
2021-08-03 15:28:13 -07:00
Lifetime uint32
Epoch uint32
}
func parsePCPResponse ( b [ ] byte ) ( res pcpResponse , ok bool ) {
if len ( b ) < 24 || b [ 0 ] != pcpVersion {
return
}
res . OpCode = b [ 1 ]
2021-11-23 14:43:04 -08:00
res . ResultCode = pcpResultCode ( b [ 3 ] )
2021-08-03 15:28:13 -07:00
res . Lifetime = binary . BigEndian . Uint32 ( b [ 4 : ] )
res . Epoch = binary . BigEndian . Uint32 ( b [ 8 : ] )
return res , true
}