icmp.sh 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. #!/bin/bash
  2. # SPDX-License-Identifier: GPL-2.0
  3. # Test for checking ICMP response with dummy address instead of 0.0.0.0.
  4. # Sets up two namespaces like:
  5. # +----------------------+ +--------------------+
  6. # | ns1 | v4-via-v6 routes: | ns2 |
  7. # | | ' | |
  8. # | +--------+ -> 172.16.1.0/24 -> +--------+ |
  9. # | | veth0 +--------------------------+ veth0 | |
  10. # | +--------+ <- 172.16.0.0/24 <- +--------+ |
  11. # | 172.16.0.1 | | 2001:db8:1::2/64 |
  12. # | 2001:db8:1::2/64 | | |
  13. # +----------------------+ +--------------------+
  14. #
  15. # And then tries to ping 172.16.1.1 from ns1. This results in a "net
  16. # unreachable" message being sent from ns2, but there is no IPv4 address set in
  17. # that address space, so the kernel should substitute the dummy address
  18. # 192.0.0.8 defined in RFC7600.
  19. NS1=ns1
  20. NS2=ns2
  21. H1_IP=172.16.0.1/32
  22. H1_IP6=2001:db8:1::1
  23. RT1=172.16.1.0/24
  24. PINGADDR=172.16.1.1
  25. RT2=172.16.0.0/24
  26. H2_IP6=2001:db8:1::2
  27. TMPFILE=$(mktemp)
  28. cleanup()
  29. {
  30. rm -f "$TMPFILE"
  31. ip netns del $NS1
  32. ip netns del $NS2
  33. }
  34. trap cleanup EXIT
  35. # Namespaces
  36. ip netns add $NS1
  37. ip netns add $NS2
  38. # Connectivity
  39. ip -netns $NS1 link add veth0 type veth peer name veth0 netns $NS2
  40. ip -netns $NS1 link set dev veth0 up
  41. ip -netns $NS2 link set dev veth0 up
  42. ip -netns $NS1 addr add $H1_IP dev veth0
  43. ip -netns $NS1 addr add $H1_IP6/64 dev veth0 nodad
  44. ip -netns $NS2 addr add $H2_IP6/64 dev veth0 nodad
  45. ip -netns $NS1 route add $RT1 via inet6 $H2_IP6
  46. ip -netns $NS2 route add $RT2 via inet6 $H1_IP6
  47. # Make sure ns2 will respond with ICMP unreachable
  48. ip netns exec $NS2 sysctl -qw net.ipv4.icmp_ratelimit=0 net.ipv4.ip_forward=1
  49. # Run the test - a ping runs in the background, and we capture ICMP responses
  50. # with tcpdump; -c 1 means it should exit on the first ping, but add a timeout
  51. # in case something goes wrong
  52. ip netns exec $NS1 ping -w 3 -i 0.5 $PINGADDR >/dev/null &
  53. ip netns exec $NS1 timeout 10 tcpdump -tpni veth0 -c 1 'icmp and icmp[icmptype] != icmp-echo' > $TMPFILE 2>/dev/null
  54. # Parse response and check for dummy address
  55. # tcpdump output looks like:
  56. # IP 192.0.0.8 > 172.16.0.1: ICMP net 172.16.1.1 unreachable, length 92
  57. RESP_IP=$(awk '{print $2}' < $TMPFILE)
  58. if [[ "$RESP_IP" != "192.0.0.8" ]]; then
  59. echo "FAIL - got ICMP response from $RESP_IP, should be 192.0.0.8"
  60. exit 1
  61. else
  62. echo "OK"
  63. exit 0
  64. fi