| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192 |
- // Copyright 2012 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.
- // +build darwin freebsd linux netbsd openbsd
- package ipv4_test
- import (
- "code.google.com/p/go.net/ipv4"
- "net"
- "os"
- "testing"
- )
- func benchmarkUDPListener() (net.PacketConn, net.Addr, error) {
- c, err := net.ListenPacket("udp4", "127.0.0.1:0")
- if err != nil {
- return nil, nil, err
- }
- dst, err := net.ResolveUDPAddr("udp4", c.LocalAddr().String())
- if err != nil {
- c.Close()
- return nil, nil, err
- }
- return c, dst, nil
- }
- func BenchmarkReadWriteNetUDP(b *testing.B) {
- c, dst, err := benchmarkUDPListener()
- if err != nil {
- b.Fatalf("benchmarkUDPListener failed: %v", err)
- }
- defer c.Close()
- wb, rb := []byte("HELLO-R-U-THERE"), make([]byte, 128)
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- benchmarkReadWriteNetUDP(b, c, wb, rb, dst)
- }
- }
- func benchmarkReadWriteNetUDP(b *testing.B, c net.PacketConn, wb, rb []byte, dst net.Addr) {
- if _, err := c.WriteTo(wb, dst); err != nil {
- b.Fatalf("net.PacketConn.WriteTo failed: %v", err)
- }
- if _, _, err := c.ReadFrom(rb); err != nil {
- b.Fatalf("net.PacketConn.ReadFrom failed: %v", err)
- }
- }
- func BenchmarkReadWriteIPv4UDP(b *testing.B) {
- c, dst, err := benchmarkUDPListener()
- if err != nil {
- b.Fatalf("benchmarkUDPListener failed: %v", err)
- }
- defer c.Close()
- p := ipv4.NewPacketConn(c)
- cf := ipv4.FlagTTL | ipv4.FlagInterface
- if err := p.SetControlMessage(cf, true); err != nil {
- b.Fatalf("ipv4.PacketConn.SetControlMessage failed: %v", err)
- }
- ifi := loopbackInterface()
- wb, rb := []byte("HELLO-R-U-THERE"), make([]byte, 128)
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- benchmarkReadWriteIPv4UDP(b, p, wb, rb, dst, ifi)
- }
- }
- func benchmarkReadWriteIPv4UDP(b *testing.B, p *ipv4.PacketConn, wb, rb []byte, dst net.Addr, ifi *net.Interface) {
- cm := ipv4.ControlMessage{TTL: 1}
- if ifi != nil {
- cm.IfIndex = ifi.Index
- }
- if _, err := p.WriteTo(wb, &cm, dst); err != nil {
- b.Fatalf("ipv4.PacketConn.WriteTo failed: %v", err)
- }
- if _, _, _, err := p.ReadFrom(rb); err != nil {
- b.Fatalf("ipv4.PacketConn.ReadFrom failed: %v", err)
- }
- }
- func TestReadWriteUnicastIPPayloadUDP(t *testing.T) {
- c, err := net.ListenPacket("udp4", "127.0.0.1:0")
- if err != nil {
- t.Fatalf("net.ListenPacket failed: %v", err)
- }
- defer c.Close()
- dst, err := net.ResolveUDPAddr("udp4", c.LocalAddr().String())
- if err != nil {
- t.Fatalf("net.ResolveUDPAddr failed: %v", err)
- }
- p := ipv4.NewPacketConn(c)
- cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
- for i, toggle := range []bool{true, false, true} {
- if err := p.SetControlMessage(cf, toggle); err != nil {
- t.Fatalf("ipv4.PacketConn.SetControlMessage failed: %v", err)
- }
- writeThenReadPayload(t, i, p, []byte("HELLO-R-U-THERE"), dst)
- }
- }
- func TestReadWriteUnicastIPPayloadICMP(t *testing.T) {
- if os.Getuid() != 0 {
- t.Skip("must be root")
- }
- c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
- if err != nil {
- t.Fatalf("net.ListenPacket failed: %v", err)
- }
- defer c.Close()
- dst, err := net.ResolveIPAddr("ip4", "127.0.0.1")
- if err != nil {
- t.Fatalf("ResolveIPAddr failed: %v", err)
- }
- p := ipv4.NewPacketConn(c)
- cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
- for i, toggle := range []bool{true, false, true} {
- wb, err := (&icmpMessage{
- Type: ipv4.ICMPTypeEcho, Code: 0,
- Body: &icmpEcho{
- ID: os.Getpid() & 0xffff, Seq: i + 1,
- Data: []byte("HELLO-R-U-THERE"),
- },
- }).Marshal()
- if err != nil {
- t.Fatalf("icmpMessage.Marshal failed: %v", err)
- }
- if err := p.SetControlMessage(cf, toggle); err != nil {
- t.Fatalf("ipv4.PacketConn.SetControlMessage failed: %v", err)
- }
- rb := writeThenReadPayload(t, i, p, wb, dst)
- m, err := parseICMPMessage(rb)
- if err != nil {
- t.Fatalf("parseICMPMessage failed: %v", err)
- }
- if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
- t.Fatalf("got type=%v, code=%v; expected type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)
- }
- }
- }
- func TestReadWriteUnicastIPDatagram(t *testing.T) {
- if os.Getuid() != 0 {
- t.Skip("must be root")
- }
- c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
- if err != nil {
- t.Fatalf("net.ListenPacket failed: %v", err)
- }
- defer c.Close()
- dst, err := net.ResolveIPAddr("ip4", "127.0.0.1")
- if err != nil {
- t.Fatalf("ResolveIPAddr failed: %v", err)
- }
- r, err := ipv4.NewRawConn(c)
- if err != nil {
- t.Fatalf("ipv4.NewRawConn failed: %v", err)
- }
- cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
- for i, toggle := range []bool{true, false, true} {
- wb, err := (&icmpMessage{
- Type: ipv4.ICMPTypeEcho, Code: 0,
- Body: &icmpEcho{
- ID: os.Getpid() & 0xffff, Seq: i + 1,
- Data: []byte("HELLO-R-U-THERE"),
- },
- }).Marshal()
- if err != nil {
- t.Fatalf("icmpMessage.Marshal failed: %v", err)
- }
- if err := r.SetControlMessage(cf, toggle); err != nil {
- t.Fatalf("ipv4.RawConn.SetControlMessage failed: %v", err)
- }
- rb := writeThenReadDatagram(t, i, r, wb, nil, dst)
- m, err := parseICMPMessage(rb)
- if err != nil {
- t.Fatalf("parseICMPMessage failed: %v", err)
- }
- if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
- t.Fatalf("got type=%v, code=%v; expected type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)
- }
- }
- }
|