blob: a6d025afec57be9037f50ed493d330352f1f6a9f [file] [log] [blame]
// Copyright 2018 The gVisor Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package ports
import (
"math/rand"
"testing"
"github.com/google/netstack/tcpip"
)
const (
fakeTransNumber tcpip.TransportProtocolNumber = 1
fakeNetworkNumber tcpip.NetworkProtocolNumber = 2
fakeIPAddress = tcpip.Address("\x08\x08\x08\x08")
fakeIPAddress1 = tcpip.Address("\x08\x08\x08\x09")
)
type portReserveTestAction struct {
port uint16
ip tcpip.Address
want *tcpip.Error
reuse bool
release bool
device tcpip.NICID
}
func TestPortReservation(t *testing.T) {
for _, test := range []struct {
tname string
actions []portReserveTestAction
}{
{
tname: "bind to ip",
actions: []portReserveTestAction{
{port: 80, ip: fakeIPAddress, want: nil},
{port: 80, ip: fakeIPAddress1, want: nil},
/* N.B. Order of tests matters! */
{port: 80, ip: anyIPAddress, want: tcpip.ErrPortInUse},
{port: 80, ip: fakeIPAddress, want: tcpip.ErrPortInUse, reuse: true},
},
},
{
tname: "bind to inaddr any",
actions: []portReserveTestAction{
{port: 22, ip: anyIPAddress, want: nil},
{port: 22, ip: fakeIPAddress, want: tcpip.ErrPortInUse},
/* release fakeIPAddress, but anyIPAddress is still inuse */
{port: 22, ip: fakeIPAddress, release: true},
{port: 22, ip: fakeIPAddress, want: tcpip.ErrPortInUse},
{port: 22, ip: fakeIPAddress, want: tcpip.ErrPortInUse, reuse: true},
/* Release port 22 from any IP address, then try to reserve fake IP address on 22 */
{port: 22, ip: anyIPAddress, want: nil, release: true},
{port: 22, ip: fakeIPAddress, want: nil},
},
}, {
tname: "bind to zero port",
actions: []portReserveTestAction{
{port: 00, ip: fakeIPAddress, want: nil},
{port: 00, ip: fakeIPAddress, want: nil},
{port: 00, ip: fakeIPAddress, reuse: true, want: nil},
},
}, {
tname: "bind to ip with reuseport",
actions: []portReserveTestAction{
{port: 25, ip: fakeIPAddress, reuse: true, want: nil},
{port: 25, ip: fakeIPAddress, reuse: true, want: nil},
{port: 25, ip: fakeIPAddress, reuse: false, want: tcpip.ErrPortInUse},
{port: 25, ip: anyIPAddress, reuse: false, want: tcpip.ErrPortInUse},
{port: 25, ip: anyIPAddress, reuse: true, want: nil},
},
}, {
tname: "bind to inaddr any with reuseport",
actions: []portReserveTestAction{
{port: 24, ip: anyIPAddress, reuse: true, want: nil},
{port: 24, ip: anyIPAddress, reuse: true, want: nil},
{port: 24, ip: anyIPAddress, reuse: false, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, reuse: false, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, release: true, want: nil},
{port: 24, ip: anyIPAddress, release: true},
{port: 24, ip: anyIPAddress, reuse: false, want: tcpip.ErrPortInUse},
{port: 24, ip: anyIPAddress, release: true},
{port: 24, ip: anyIPAddress, reuse: false, want: nil},
},
}, {
tname: "bind twice with device fails",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 3, want: nil},
{port: 24, ip: fakeIPAddress, device: 3, want: tcpip.ErrPortInUse},
},
}, {
tname: "bind to device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 1, want: nil},
{port: 24, ip: fakeIPAddress, device: 2, want: nil},
},
}, {
tname: "bind to device and then without device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, want: tcpip.ErrPortInUse},
},
}, {
tname: "bind without device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, want: nil},
{port: 24, ip: fakeIPAddress, device: 123, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, reuse: true, want: tcpip.ErrPortInUse},
},
}, {
tname: "bind with device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, want: nil},
{port: 24, ip: fakeIPAddress, device: 123, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 0, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 456, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 789, want: nil},
{port: 24, ip: fakeIPAddress, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, reuse: true, want: tcpip.ErrPortInUse},
},
}, {
tname: "bind with reuse",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 123, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: nil},
},
}, {
tname: "binding with reuse and device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 123, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 456, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 789, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 999, want: tcpip.ErrPortInUse},
},
}, {
tname: "mixing reuse and not reuse by binding to device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 456, want: nil},
{port: 24, ip: fakeIPAddress, device: 789, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 999, want: nil},
},
}, {
tname: "can't bind to 0 after mixing reuse and not reuse",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 456, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: tcpip.ErrPortInUse},
},
}, {
tname: "bind and release",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: nil},
{port: 24, ip: fakeIPAddress, device: 345, reuse: false, want: tcpip.ErrPortInUse},
{port: 24, ip: fakeIPAddress, device: 789, reuse: true, want: nil},
// Release the bind to device 0 and try again.
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: nil, release: true},
{port: 24, ip: fakeIPAddress, device: 345, reuse: false, want: nil},
},
}, {
tname: "bind twice with reuse once",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: false, want: nil},
{port: 24, ip: fakeIPAddress, device: 0, reuse: true, want: tcpip.ErrPortInUse},
},
}, {
tname: "release an unreserved device",
actions: []portReserveTestAction{
{port: 24, ip: fakeIPAddress, device: 123, reuse: false, want: nil},
{port: 24, ip: fakeIPAddress, device: 456, reuse: false, want: nil},
// The below don't exist.
{port: 24, ip: fakeIPAddress, device: 345, reuse: false, want: nil, release: true},
{port: 9999, ip: fakeIPAddress, device: 123, reuse: false, want: nil, release: true},
// Release all.
{port: 24, ip: fakeIPAddress, device: 123, reuse: false, want: nil, release: true},
{port: 24, ip: fakeIPAddress, device: 456, reuse: false, want: nil, release: true},
},
},
} {
t.Run(test.tname, func(t *testing.T) {
pm := NewPortManager()
net := []tcpip.NetworkProtocolNumber{fakeNetworkNumber}
for _, test := range test.actions {
if test.release {
pm.ReleasePort(net, fakeTransNumber, test.ip, test.port, test.device)
continue
}
gotPort, err := pm.ReservePort(net, fakeTransNumber, test.ip, test.port, test.reuse, test.device)
if err != test.want {
t.Fatalf("ReservePort(.., .., %s, %d, %t, %d) = %v, want %v", test.ip, test.port, test.reuse, test.device, err, test.want)
}
if test.port == 0 && (gotPort == 0 || gotPort < FirstEphemeral) {
t.Fatalf("ReservePort(.., .., .., 0) = %d, want port number >= %d to be picked", gotPort, FirstEphemeral)
}
}
})
}
}
func TestPickEphemeralPort(t *testing.T) {
customErr := &tcpip.Error{}
for _, test := range []struct {
name string
f func(port uint16) (bool, *tcpip.Error)
wantErr *tcpip.Error
wantPort uint16
}{
{
name: "no-port-available",
f: func(port uint16) (bool, *tcpip.Error) {
return false, nil
},
wantErr: tcpip.ErrNoPortAvailable,
},
{
name: "port-tester-error",
f: func(port uint16) (bool, *tcpip.Error) {
return false, customErr
},
wantErr: customErr,
},
{
name: "only-port-16042-available",
f: func(port uint16) (bool, *tcpip.Error) {
if port == FirstEphemeral+42 {
return true, nil
}
return false, nil
},
wantPort: FirstEphemeral + 42,
},
{
name: "only-port-under-16000-available",
f: func(port uint16) (bool, *tcpip.Error) {
if port < FirstEphemeral {
return true, nil
}
return false, nil
},
wantErr: tcpip.ErrNoPortAvailable,
},
} {
t.Run(test.name, func(t *testing.T) {
pm := NewPortManager()
if port, err := pm.PickEphemeralPort(test.f); port != test.wantPort || err != test.wantErr {
t.Errorf("PickEphemeralPort(..) = (port %d, err %v); want (port %d, err %v)", port, err, test.wantPort, test.wantErr)
}
})
}
}
func TestPickEphemeralPortStable(t *testing.T) {
customErr := &tcpip.Error{}
for _, test := range []struct {
name string
f func(port uint16) (bool, *tcpip.Error)
wantErr *tcpip.Error
wantPort uint16
}{
{
name: "no-port-available",
f: func(port uint16) (bool, *tcpip.Error) {
return false, nil
},
wantErr: tcpip.ErrNoPortAvailable,
},
{
name: "port-tester-error",
f: func(port uint16) (bool, *tcpip.Error) {
return false, customErr
},
wantErr: customErr,
},
{
name: "only-port-16042-available",
f: func(port uint16) (bool, *tcpip.Error) {
if port == FirstEphemeral+42 {
return true, nil
}
return false, nil
},
wantPort: FirstEphemeral + 42,
},
{
name: "only-port-under-16000-available",
f: func(port uint16) (bool, *tcpip.Error) {
if port < FirstEphemeral {
return true, nil
}
return false, nil
},
wantErr: tcpip.ErrNoPortAvailable,
},
} {
t.Run(test.name, func(t *testing.T) {
pm := NewPortManager()
portOffset := uint32(rand.Int31n(int32(numEphemeralPorts)))
if port, err := pm.PickEphemeralPortStable(portOffset, test.f); port != test.wantPort || err != test.wantErr {
t.Errorf("PickEphemeralPort(..) = (port %d, err %v); want (port %d, err %v)", port, err, test.wantPort, test.wantErr)
}
})
}
}