From 5789fd018b80a460f45f3a58005db4442fa2175b Mon Sep 17 00:00:00 2001 From: cyrus Date: Thu, 29 Nov 2012 20:14:28 +0000 Subject: ipv6-support: Add new IPv6-support meta-package git-svn-id: svn://svn.openwrt.org/openwrt/trunk@34422 3c298f89-4303-0410-b956-a3cf2f4a3e73 --- package/network/ipv6/ipv6-support/Makefile | 46 ++++ package/network/ipv6/ipv6-support/files/dhcpv6.sh | 52 ++++ .../network/ipv6/ipv6-support/files/ipv6.hotplug | 18 ++ .../ipv6/ipv6-support/files/network6.config | 17 ++ package/network/ipv6/ipv6-support/files/support.sh | 304 +++++++++++++++++++++ 5 files changed, 437 insertions(+) create mode 100644 package/network/ipv6/ipv6-support/Makefile create mode 100755 package/network/ipv6/ipv6-support/files/dhcpv6.sh create mode 100644 package/network/ipv6/ipv6-support/files/ipv6.hotplug create mode 100644 package/network/ipv6/ipv6-support/files/network6.config create mode 100644 package/network/ipv6/ipv6-support/files/support.sh diff --git a/package/network/ipv6/ipv6-support/Makefile b/package/network/ipv6/ipv6-support/Makefile new file mode 100644 index 000000000..8397bc462 --- /dev/null +++ b/package/network/ipv6/ipv6-support/Makefile @@ -0,0 +1,46 @@ +# +# Copyright (C) 2010-2012 OpenWrt.org +# +# This is free software, licensed under the GNU General Public License v2. +# See /LICENSE for more information. +# + +include $(TOPDIR)/rules.mk + +PKG_NAME:=ipv6-support +PKG_VERSION:=2012-11-29 +PKG_RELEASE:=1 + +include $(INCLUDE_DIR)/package.mk + +define Package/ipv6-support + SECTION:=ipv6 + CATEGORY:=IPv6 + DEPENDS:=+kmod-ipv6 +6relayd +odhcp6c +6distributed +ip6tables +ubus + TITLE:=Basic IPv6-support for Customer Edge Routers + MAINTAINER:=Steven Barth + PKGARCH:=all +endef + +define Package/ipv6-support/description +This package provides basic IPv6 support including Router Discovery, +DHCPv6 (client & server), prefix delegation and distribution. +endef + +define Build/Compile +endef + +define Build/Configure +endef + +define Package/ipv6-support/install + $(INSTALL_DIR) $(1)/etc/hotplug.d/iface + $(INSTALL_DATA) ./files/ipv6.hotplug $(1)/etc/hotplug.d/iface/20-ipv6 + $(INSTALL_DIR) $(1)/lib/ipv6 + $(INSTALL_DATA) ./files/support.sh $(1)/lib/ipv6/support.sh + $(INSTALL_BIN) ./files/dhcpv6.sh $(1)/lib/ipv6/dhcpv6.sh + $(INSTALL_DIR) $(1)/etc/config + $(INSTALL_DATA) ./files/network6.config $(1)/etc/config/network6 +endef + +$(eval $(call BuildPackage,ipv6-support)) diff --git a/package/network/ipv6/ipv6-support/files/dhcpv6.sh b/package/network/ipv6/ipv6-support/files/dhcpv6.sh new file mode 100755 index 000000000..67fa174d1 --- /dev/null +++ b/package/network/ipv6/ipv6-support/files/dhcpv6.sh @@ -0,0 +1,52 @@ +#!/bin/sh +# Copyright (c) 2012 OpenWrt.org +. /lib/ipv6/support.sh +. /lib/netifd/netifd-proto.sh + +local device="$1" +local state="$2" +local network="" + +resolve_network network "$device" + +# Unknown network +[ -z "$network" ] && exit 0 + + +# Announce prefixes +for prefix in $PREFIXES; do + announce_prefix "$prefix" "$network" +done + +for prefix in $PREFIXES_LOST; do + announce_prefix "$prefix" "$network" delprefix +done + + +# Enable relaying if requested +local prefix_fallback +config_get prefix_fallback "$network" prefix_fallback +[ "$prefix_fallback" == "relay" -a -z "$PREFIXES" -a "$state" != "unbound" ] && + restart_relay "$network" 1 + +# Disable relay if requested +[ "$prefix_fallback" != "relay" -o -n "$PREFIXES" -o "$state" == "unbound" ] && + stop_relay "$network" + + +# Operations in case of success +[ "$state" == "timeout" || "$state" == "unbound" ] && exit 0 + +local peerdns +config_get_bool peerdns "$network" peerdns 0 +[ "peerdns" -eq "1" ] && { + proto_init_update "*" 1 + for server in $RDNSS; do + proto_add_dns_server "$server" + done + for domain in $DOMAINS; do + proto_add_dns_search "$domain" + done + proto_send_update "$network" +} + diff --git a/package/network/ipv6/ipv6-support/files/ipv6.hotplug b/package/network/ipv6/ipv6-support/files/ipv6.hotplug new file mode 100644 index 000000000..e3379b6b2 --- /dev/null +++ b/package/network/ipv6/ipv6-support/files/ipv6.hotplug @@ -0,0 +1,18 @@ +#!/bin/sh +# Copyright (c) 2012 OpenWrt.org +[ "$DEVICE" == "lo" ] && exit 0 +. /lib/ipv6/support.sh + +local mode +config_get mode "$INTERFACE" mode + +case "$ACTION" in + ifup) + [ "$mode" != "downstream" ] && enable_static $INTERFACE $DEVICE + [ "$mode" == "upstream" ] && enable_upstream $INTERFACE $DEVICE + [ "$mode" == "downstream" ] && enable_downstream $INTERFACE $DEVICE + ;; + ifdown) + disable_interface $INTERFACE $DEVICE + ;; +esac diff --git a/package/network/ipv6/ipv6-support/files/network6.config b/package/network/ipv6/ipv6-support/files/network6.config new file mode 100644 index 000000000..4632bba0b --- /dev/null +++ b/package/network/ipv6/ipv6-support/files/network6.config @@ -0,0 +1,17 @@ +config interface wan + option mode upstream + option ula_prefix auto + option request_prefix auto + option prefix_fallback relay + option peerdns 1 + + +config interface lan + option mode downstream + option advertise_prefix 64 + option relay_master wan + + +config interface 6in4 + option mode static + list static_prefix 2001:DB8::/48 diff --git a/package/network/ipv6/ipv6-support/files/support.sh b/package/network/ipv6/ipv6-support/files/support.sh new file mode 100644 index 000000000..5525a3a56 --- /dev/null +++ b/package/network/ipv6/ipv6-support/files/support.sh @@ -0,0 +1,304 @@ +#!/bin/sh +# Copyright (c) 2012 OpenWrt.org +. /lib/functions.sh +. /lib/functions/service.sh +. /lib/functions/network.sh + +config_load network6 + + +conf_get() { + local __return="$1" + local __device="$2" + local __option="$3" + local __value=$(cat "/proc/sys/net/ipv6/conf/$device/$option") + eval "$__return=$__value" +} + + +conf_set() { + local device="$1" + local option="$2" + local value="$3" + echo "$value" > "/proc/sys/net/ipv6/conf/$device/$option" +} + + +stop_service() { + local __exe="$1" + SERVICE_PID_FILE="$2" + local __return="$3" + + service_check "$__exe" && { + service_stop "$__exe" + [ -n "$__return" ] && eval "$__return=1" + } + rm -f "$SERVICE_PID_FILE" +} + + +start_service() { + local cmd="$1" + local pidfile="$2" + + SERVICE_DAEMONIZE=1 + SERVICE_WRITE_PID=1 + SERVICE_PID_FILE="$pidfile" + service_start $cmd +} + + +resolve_network_add() { + local __section="$1" + local __device="$2" + local __return="$3" + + local __cdevice + network_get_device __cdevice "$__section" + [ "$__cdevice" != "$__device" ] && return + + eval "$__return"'="'"$__section"'"' +} + + +resolve_network() { + local __return="$1" + local __device="$2" + config_foreach resolve_network_add interface "$__device" "$__return" +} + + +announce_prefix() { + local prefix="$1" + local network="$2" + local cmd="$3" + + local addr=$(echo "$prefix" | cut -d/ -f1) + local rem=$(echo "$prefix" | cut -d/ -f2) + local length=$(echo "$rem" | cut -d, -f1) + local prefer="" + local valid="" + + # If preferred / valid provided + [ "$rem" != "$length" ] && { + prefer=$(echo "$rem" | cut -d, -f2) + valid=$(echo "$rem" | cut -d, -f3) + } + + local msg='{"network": "'"$network"'", "prefix": "'"$addr"'", "length": '"$length" + [ -n "$valid" ] && msg="$msg"', "valid": '"$valid"', "preferred": '"$prefer" + [ -z "$cmd" ] && cmd=newprefix + + ubus call 6distributed "$cmd" "$msg}" +} + + +disable_downstream() { + local network="$1" + + # Notify the address distribution daemon + ubus call 6distributed deliface '{"network": "'"$network"'"}' + + # Disable advertisement daemon + stop_service /usr/sbin/6relayd "/var/run/ipv6-downstream-$network.pid" +} + + +restart_relay_add() { + local __section="$1" + local __return="$2" + local __master="$3" + local __disable="$4" + + network_is_up "$__section" || return + + # Match master network + local __cmaster="" + config_get __cmaster "$__section" relay_master + [ "$__master" != "$__cmaster" ] && return + + # Disable any active distribution + disable_downstream "$__section" + + local __device="" + network_get_device __device "$__section" + + # Coming from stop relay, reenable distribution + [ "$__disable" == "disable" ] && { + enable_downstream "$__section" "$__device" + return + } + + + eval "$__return"'="$'"$__return"' '"$__device"'"' +} + + +stop_relay() { + local network="$1" + local pid="/var/run/ipv6-relay-$network.pid" + local was_running="" + + stop_service /usr/sbin/6relayd "$pid" was_running + + # Reenable normal distribution on slave interfaces + [ -n "$was_running" ] && config_foreach restart_relay_add interface dummy "$network" disable +} + + +restart_relay() { + local network="$1" + local force="$2" + local pid="/var/run/ipv6-relay-$network.pid" + + local not_running=0 + [ -f "$pid" ] || not_running=1 + + # Don't start if not desired + [ "$force" != "1" ] && [ "$not_running" == "1" ] && return + + # Kill current relay and distribution daemon + stop_relay "$network" + + # Detect master device + local device="" + network_get_device device $network + + # Generate command string + local cmd="/usr/sbin/6relayd -A $device " + config_foreach restart_relay_add interface cmd "$network" + + # Start relay + start_service "$cmd" "$pid" +} + + +restart_master_relay() { + local network="$1" + + # Disable active relaying to this interface + local relay_master + config_get relay_master "$network" relay_master + [ -n "$relay_master" ] && restart_relay "$relay_master" +} + + +disable_interface() { + local network="$1" + + # Delete all prefixes routed to this interface + ubus call 6distributed delprefix '{"network": "'"$network"'"}' + + # Restart Relay + restart_master_relay "$network" + + # Disable distribution + disable_downstream "$network" + + # Disable relay + stop_relay "$network" + + # Disable DHCPv6 client if enabled, state script will take care + stop_service /usr/sbin/odhcp6c "/var/run/ipv6-upstream-$network.pid" +} + + +enable_static() { + local network="$1" + local device="$2" + + # Enable global forwarding + local global_forward + conf_get global_forward all forwarding + [ "$global_forward" != "1" ] && conf_set all forwarding 1 + + # Configure device + conf_set "$device" accept_ra 1 + conf_set "$device" forwarding 1 + + # ULA-integration + local ula_prefix="" + config_get ula_prefix "$network" ula_prefix + + # ULA auto configuration (first init) + [ "$ula_prefix" == "auto" ] && { + local r1="" + local r2="" + local r3="" + + # Sometimes results are empty, therefore try until it works... + while [ -z "$r1" -o -z "$r2" -o -z "$r3" ]; do + r1=$(printf "%02x" $(($(