|  | #!/bin/bash | 
|  | # | 
|  | # Send data between two processes across namespaces | 
|  | # Run twice: once without and once with zerocopy | 
|  |  | 
|  | set -e | 
|  |  | 
|  | readonly DEV="veth0" | 
|  | readonly DEV_MTU=65535 | 
|  | readonly BIN="./msg_zerocopy" | 
|  |  | 
|  | readonly RAND="$(mktemp -u XXXXXX)" | 
|  | readonly NSPREFIX="ns-${RAND}" | 
|  | readonly NS1="${NSPREFIX}1" | 
|  | readonly NS2="${NSPREFIX}2" | 
|  |  | 
|  | readonly SADDR4='192.168.1.1' | 
|  | readonly DADDR4='192.168.1.2' | 
|  | readonly SADDR6='fd::1' | 
|  | readonly DADDR6='fd::2' | 
|  |  | 
|  | readonly path_sysctl_mem="net.core.optmem_max" | 
|  |  | 
|  | # No arguments: automated test | 
|  | if [[ "$#" -eq "0" ]]; then | 
|  | $0 4 tcp -t 1 | 
|  | $0 6 tcp -t 1 | 
|  | $0 4 udp -t 1 | 
|  | $0 6 udp -t 1 | 
|  | echo "OK. All tests passed" | 
|  | exit 0 | 
|  | fi | 
|  |  | 
|  | # Argument parsing | 
|  | if [[ "$#" -lt "2" ]]; then | 
|  | echo "Usage: $0 [4|6] [tcp|udp|raw|raw_hdrincl|packet|packet_dgram] <args>" | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | readonly IP="$1" | 
|  | shift | 
|  | readonly TXMODE="$1" | 
|  | shift | 
|  | readonly EXTRA_ARGS="$@" | 
|  |  | 
|  | # Argument parsing: configure addresses | 
|  | if [[ "${IP}" == "4" ]]; then | 
|  | readonly SADDR="${SADDR4}" | 
|  | readonly DADDR="${DADDR4}" | 
|  | elif [[ "${IP}" == "6" ]]; then | 
|  | readonly SADDR="${SADDR6}" | 
|  | readonly DADDR="${DADDR6}" | 
|  | else | 
|  | echo "Invalid IP version ${IP}" | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | # Argument parsing: select receive mode | 
|  | # | 
|  | # This differs from send mode for | 
|  | # - packet:	use raw recv, because packet receives skb clones | 
|  | # - raw_hdrinc: use raw recv, because hdrincl is a tx-only option | 
|  | case "${TXMODE}" in | 
|  | 'packet' | 'packet_dgram' | 'raw_hdrincl') | 
|  | RXMODE='raw' | 
|  | ;; | 
|  | *) | 
|  | RXMODE="${TXMODE}" | 
|  | ;; | 
|  | esac | 
|  |  | 
|  | # Start of state changes: install cleanup handler | 
|  |  | 
|  | cleanup() { | 
|  | ip netns del "${NS2}" | 
|  | ip netns del "${NS1}" | 
|  | } | 
|  |  | 
|  | trap cleanup EXIT | 
|  |  | 
|  | # Create virtual ethernet pair between network namespaces | 
|  | ip netns add "${NS1}" | 
|  | ip netns add "${NS2}" | 
|  |  | 
|  | # Configure system settings | 
|  | ip netns exec "${NS1}" sysctl -w -q "${path_sysctl_mem}=1000000" | 
|  | ip netns exec "${NS2}" sysctl -w -q "${path_sysctl_mem}=1000000" | 
|  |  | 
|  | ip link add "${DEV}" mtu "${DEV_MTU}" netns "${NS1}" type veth \ | 
|  | peer name "${DEV}" mtu "${DEV_MTU}" netns "${NS2}" | 
|  |  | 
|  | # Bring the devices up | 
|  | ip -netns "${NS1}" link set "${DEV}" up | 
|  | ip -netns "${NS2}" link set "${DEV}" up | 
|  |  | 
|  | # Set fixed MAC addresses on the devices | 
|  | ip -netns "${NS1}" link set dev "${DEV}" address 02:02:02:02:02:02 | 
|  | ip -netns "${NS2}" link set dev "${DEV}" address 06:06:06:06:06:06 | 
|  |  | 
|  | # Add fixed IP addresses to the devices | 
|  | ip -netns "${NS1}" addr add 192.168.1.1/24 dev "${DEV}" | 
|  | ip -netns "${NS2}" addr add 192.168.1.2/24 dev "${DEV}" | 
|  | ip -netns "${NS1}" addr add       fd::1/64 dev "${DEV}" nodad | 
|  | ip -netns "${NS2}" addr add       fd::2/64 dev "${DEV}" nodad | 
|  |  | 
|  | # Optionally disable sg or csum offload to test edge cases | 
|  | # ip netns exec "${NS1}" ethtool -K "${DEV}" sg off | 
|  |  | 
|  | do_test() { | 
|  | local readonly ARGS="$1" | 
|  |  | 
|  | echo "ipv${IP} ${TXMODE} ${ARGS}" | 
|  | ip netns exec "${NS2}" "${BIN}" "-${IP}" -i "${DEV}" -t 2 -C 2 -S "${SADDR}" -D "${DADDR}" ${ARGS} -r "${RXMODE}" & | 
|  | sleep 0.2 | 
|  | ip netns exec "${NS1}" "${BIN}" "-${IP}" -i "${DEV}" -t 1 -C 3 -S "${SADDR}" -D "${DADDR}" ${ARGS} "${TXMODE}" | 
|  | wait | 
|  | } | 
|  |  | 
|  | do_test "${EXTRA_ARGS}" | 
|  | do_test "-z ${EXTRA_ARGS}" | 
|  | echo ok |