зеркало из https://github.com/golang/sys.git
187 строки
4.6 KiB
Go
187 строки
4.6 KiB
Go
// Copyright 2021 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
//go:build linux
|
|
// +build linux
|
|
|
|
package unix
|
|
|
|
import (
|
|
"bytes"
|
|
"net"
|
|
"testing"
|
|
"unsafe"
|
|
)
|
|
|
|
// An ifreqUnion is shorthand for a byte array matching the
|
|
// architecture-dependent size of an ifreq's union field.
|
|
type ifreqUnion = [len(ifreq{}.Ifru)]byte
|
|
|
|
func TestNewIfreq(t *testing.T) {
|
|
// Interface name too long.
|
|
if _, err := NewIfreq("abcdefghijklmnop"); err != EINVAL {
|
|
t.Fatalf("expected error EINVAL, but got: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestIfreqSize(t *testing.T) {
|
|
// Ensure ifreq (generated) and Ifreq/ifreqData (hand-written to create a
|
|
// safe wrapper and store a pointer field) are identical in size.
|
|
want := unsafe.Sizeof(ifreq{})
|
|
if got := unsafe.Sizeof(Ifreq{}); want != got {
|
|
t.Fatalf("unexpected Ifreq size: got: %d, want: %d", got, want)
|
|
}
|
|
|
|
if got := unsafe.Sizeof(ifreqData{}); want != got {
|
|
t.Fatalf("unexpected IfreqData size: got: %d, want: %d", got, want)
|
|
}
|
|
}
|
|
|
|
func TestIfreqName(t *testing.T) {
|
|
// Invalid ifreq (no NULL terminator), so expect empty string.
|
|
var name [IFNAMSIZ]byte
|
|
for i := range name {
|
|
name[i] = 0xff
|
|
}
|
|
|
|
bad := &Ifreq{raw: ifreq{Ifrn: name}}
|
|
if got := bad.Name(); got != "" {
|
|
t.Fatalf("expected empty ifreq name, but got: %q", got)
|
|
}
|
|
|
|
// Valid ifreq, expect the hard-coded testIfreq name.
|
|
ifr := testIfreq(t)
|
|
if want, got := ifreqName, ifr.Name(); want != got {
|
|
t.Fatalf("unexpected ifreq name: got: %q, want: %q", got, want)
|
|
}
|
|
}
|
|
|
|
func TestIfreqWithData(t *testing.T) {
|
|
ifr := testIfreq(t)
|
|
|
|
// Store pointer data in the ifreq so we can retrieve it and cast back later
|
|
// for comparison.
|
|
want := [5]byte{'h', 'e', 'l', 'l', 'o'}
|
|
ifrd := ifr.withData(unsafe.Pointer(&want[0]))
|
|
|
|
// Ensure the memory of the original Ifreq was not modified by SetData.
|
|
if ifr.raw.Ifru != (ifreqUnion{}) {
|
|
t.Fatalf("ifreq was unexpectedly modified: % #x", ifr.raw.Ifru)
|
|
}
|
|
|
|
got := *(*[5]byte)(ifrd.data)
|
|
if want != got {
|
|
t.Fatalf("unexpected ifreq data bytes:\n got: % #x\nwant: % #x", got, want)
|
|
}
|
|
}
|
|
|
|
func TestIfreqInet4Addr(t *testing.T) {
|
|
ifr := testIfreq(t)
|
|
in := net.IPv4(192, 0, 2, 1).To4()
|
|
if err := ifr.SetInet4Addr(in); err != nil {
|
|
t.Fatalf("failed to set ifreq IPv4 address: %v", err)
|
|
}
|
|
|
|
// Store fixed offset data (AF_INET, IPv4 address) within underlying
|
|
// sockaddr bytes. Everything else should be zeroed.
|
|
want := ifreqUnion{4: 192, 5: 0, 6: 2, 7: 1}
|
|
if isBigEndian {
|
|
want[0] = 0x00
|
|
want[1] = 0x02
|
|
} else {
|
|
want[0] = 0x02
|
|
want[1] = 0x00
|
|
}
|
|
|
|
if got := ifr.raw.Ifru; want != got {
|
|
t.Fatalf("unexpected ifreq sockaddr bytes:\n got: % #x\nwant: % #x", got, want)
|
|
}
|
|
|
|
got, err := ifr.Inet4Addr()
|
|
if err != nil {
|
|
t.Fatalf("failed to get ifreq IPv4 address: %v", err)
|
|
}
|
|
if !bytes.Equal(in, got) {
|
|
t.Fatalf("unexpected ifreq IPv4 address:\n got: % #x\nwant: % #x", got, in)
|
|
}
|
|
|
|
// Invalid input, wrong length.
|
|
if err := ifr.SetInet4Addr([]byte{0xff}); err == nil {
|
|
t.Fatal("expected an error setting invalid IPv4 address, but none occurred")
|
|
}
|
|
|
|
// Invalid output, AF_INET is only set by SetInet4Addr input.
|
|
ifr.SetUint32(0xffffffff)
|
|
if _, err := ifr.Inet4Addr(); err == nil {
|
|
t.Fatal("expected an error getting invalid IPv4 address, but none occurred")
|
|
}
|
|
}
|
|
|
|
func TestIfreqUint16(t *testing.T) {
|
|
ifr := testIfreq(t)
|
|
const in = 0x0102
|
|
ifr.SetUint16(in)
|
|
|
|
// The layout of the bytes depends on the machine's endianness.
|
|
var want ifreqUnion
|
|
if isBigEndian {
|
|
want[0] = 0x01
|
|
want[1] = 0x02
|
|
} else {
|
|
want[0] = 0x02
|
|
want[1] = 0x01
|
|
}
|
|
|
|
if got := ifr.raw.Ifru; want != got {
|
|
t.Fatalf("unexpected ifreq uint16 bytes:\n got: % #x\nwant: % #x", got, want)
|
|
}
|
|
|
|
if got := ifr.Uint16(); in != got {
|
|
t.Fatalf("unexpected ifreq uint16: got: %d, want: %d", got, in)
|
|
}
|
|
}
|
|
|
|
func TestIfreqUint32(t *testing.T) {
|
|
ifr := testIfreq(t)
|
|
const in = 0x01020304
|
|
ifr.SetUint32(in)
|
|
|
|
// The layout of the bytes depends on the machine's endianness.
|
|
var want ifreqUnion
|
|
if isBigEndian {
|
|
want[0] = 0x01
|
|
want[1] = 0x02
|
|
want[2] = 0x03
|
|
want[3] = 0x04
|
|
} else {
|
|
want[0] = 0x04
|
|
want[1] = 0x03
|
|
want[2] = 0x02
|
|
want[3] = 0x01
|
|
}
|
|
|
|
if got := ifr.raw.Ifru; want != got {
|
|
t.Fatalf("unexpected ifreq uint32 bytes:\n got: % #x\nwant: % #x", got, want)
|
|
}
|
|
|
|
if got := ifr.Uint32(); in != got {
|
|
t.Fatalf("unexpected ifreq uint32: got: %d, want: %d", got, in)
|
|
}
|
|
}
|
|
|
|
// ifreqName is a hard-coded name for testIfreq.
|
|
const ifreqName = "eth0"
|
|
|
|
// testIfreq returns an Ifreq with a populated interface name.
|
|
func testIfreq(t *testing.T) *Ifreq {
|
|
t.Helper()
|
|
|
|
ifr, err := NewIfreq(ifreqName)
|
|
if err != nil {
|
|
t.Fatalf("failed to create ifreq: %v", err)
|
|
}
|
|
|
|
return ifr
|
|
}
|