You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

273 lines
5.7 KiB

  1. #!/bin/sh /etc/rc.common
  2. # Copyright (C) 2013 Julius Schulz-Zander <julius@net.t-labs.tu-berlin.de>
  3. # Copyright (C) 2014-2017 OpenWrt.org
  4. # Copyright (C) 2018 Yousong Zhou <yszhou4tech@gmail.com>
  5. # Copyright (C) 2021 Felix Fietkau <nbd@nbd.name>
  6. . /lib/functions/procd.sh
  7. START=15
  8. basescript=$(readlink "$initscript")
  9. ovs_ctl="/usr/share/openvswitch/scripts/ovs-ctl"; [ -x "$ovs_ctl" ] || ovs_ctl=:
  10. ovn_ctl="/usr/share/ovn/scripts/ovn-ctl"; [ -x "$ovn_ctl" ] || ovn_ctl=:
  11. extra_command "status" "Get status information"
  12. service_triggers() {
  13. procd_add_reload_trigger openvswitch
  14. }
  15. init_triggers() {
  16. procd_open_service "$(basename ${basescript:-$initscript})" "$initscript"
  17. procd_close_service set
  18. }
  19. start() {
  20. init_triggers
  21. ovs_action start "$@"
  22. }
  23. reload() {
  24. start
  25. }
  26. running() {
  27. return 0
  28. }
  29. stop() {
  30. procd_kill "$(basename ${basescript:-$initscript})"
  31. ovs_action stop "$@"
  32. }
  33. restart() {
  34. init_triggers
  35. ovs_action restart "$@"
  36. }
  37. status() {
  38. ovs_action status "$@"
  39. }
  40. ovs_action_cfgs=
  41. ovs_action() {
  42. local action="$1"; shift
  43. local cfgtype
  44. ovs_action_cfgs="$*"
  45. config_load openvswitch
  46. for cfgtype in ovs ovn_northd ovn_controller; do
  47. config_foreach "ovs_xx" "$cfgtype" "$action" "$cfgtype"
  48. done
  49. case "$action" in
  50. restart|start)
  51. config_foreach ovs_bridge_init "ovs_bridge"
  52. ;;
  53. esac
  54. }
  55. ovs_xx() {
  56. local cfg="$1"
  57. local action="$2"
  58. local cfgtype="$3"
  59. local disabled
  60. if [ -n "$ovs_action_cfgs" ] && ! list_contains "ovs_action_cfgs" "$cfg"; then
  61. return
  62. fi
  63. case "$action" in
  64. status|stop) ;;
  65. *)
  66. config_get_bool disabled "$cfg" disabled 0
  67. [ "$disabled" == "0" ] || return
  68. ;;
  69. esac
  70. case "$cfgtype" in
  71. ovs)
  72. "$ovs_ctl" "$action" \
  73. --system-id=random 1000>&-
  74. ovs_set_ssl
  75. ;;
  76. ovn_*)
  77. "$ovn_ctl" "${action}_${cfgtype#ovn_}"
  78. ;;
  79. esac
  80. }
  81. ovs_bridge_parse_port() {
  82. case "$1" in
  83. *:*)
  84. port="${1%%:*}"
  85. type="${1#*:}"
  86. ;;
  87. *)
  88. port="$1"
  89. type=""
  90. ;;
  91. esac
  92. }
  93. ovs_bridge_port_add() {
  94. [ -n "$1" ] || return
  95. ovs_bridge_parse_port "$1"
  96. cur_type="$(ovs-vsctl get interface "$port" type 2>/dev/null)"
  97. [ "$?" = 0 ] && {
  98. [ "$type" = "$cur_type" ] || ovs-vsctl del-port "$port"
  99. }
  100. ovs-vsctl --may-exist add-port "$name" "$port" ${type:+ -- set interface "$port" type="$type"}
  101. __port_list="$__port_list ${port} "
  102. }
  103. ovs_bridge_port_add_complex() {
  104. local cfg="$1"
  105. local cur_bridge="$2"
  106. local bridge disabled ofport port tag type
  107. local cur_tag cur_type del_port
  108. config_get_bool disabled "$cfg" disabled 0
  109. [ "$disabled" = "0" ] || return
  110. config_get bridge "$cfg" bridge
  111. [ "$bridge" = "$cur_bridge" ] || return
  112. ovs-vsctl br-exists "$bridge" || return
  113. config_get port "$cfg" port
  114. [ -n "$port" ] || return
  115. config_get ofport "$cfg" ofport
  116. config_get tag "$cfg" tag
  117. if [ -n "$tag" ]; then
  118. if cur_tag="$(ovs-vsctl get port "$port" tag 2>/dev/null)"; then
  119. [ "$tag" = "$cur_tag" ] || del_port=1
  120. fi
  121. fi
  122. config_get type "$cfg" type
  123. if [ -n "$type" ]; then
  124. if cur_type="$(ovs-vsctl get interface "$port" type 2>/dev/null)"; then
  125. [ "$type" = "$cur_type" ] || del_port=1
  126. fi
  127. fi
  128. [ "${del_port:-0}" -eq 1 ] && ovs-vsctl --if-exists del-port "$bridge" "$port"
  129. ovs-vsctl --may-exist add-port "$bridge" "$port" ${tag:+tag="$tag"} \
  130. ${ofport:+ -- set interface "$port" ofport_request="$ofport"} \
  131. ${type:+ -- set interface "$port" type="$type"}
  132. __port_list="$__port_list ${port} "
  133. }
  134. ovs_bridge_port_cleanup() {
  135. for port in `ovs-vsctl list-ports "$name"`; do
  136. case "$__port_list" in
  137. *" $port "*);;
  138. *) ovs-vsctl del-port "$port";;
  139. esac
  140. done
  141. }
  142. ovs_bridge_validate_datapath_id() {
  143. local dpid="$1"
  144. if expr "$dpid" : '[[:xdigit:]]\{16\}$' > /dev/null; then
  145. return 0
  146. elif expr "$dpid" : '0x[[:xdigit:]]\{1,16\}$' > /dev/null; then
  147. return 0
  148. else
  149. logger -t openvswitch "invalid datapath_id: $dpid"
  150. return 1
  151. fi
  152. }
  153. ovs_bridge_validate_datapath_desc() {
  154. local dpdesc="$1"
  155. if [ "$(echo $dpdesc | wc -c)" -le 255 ]; then
  156. return 0
  157. else
  158. logger -t openvswitch "invalid datapath_desc: $dpdesc"
  159. return 1
  160. fi
  161. }
  162. ovs_bridge_validate_fail_mode() {
  163. local fail_mode="$1"
  164. case "$fail_mode" in
  165. secure|standalone)
  166. return 0
  167. ;;
  168. *)
  169. logger -t openvswitch "invalid fail_mode: $fail_mode"
  170. return 1
  171. ;;
  172. esac
  173. }
  174. ovs_bridge_init() {
  175. local cfg="$1"
  176. local disabled
  177. local name
  178. local controller
  179. local datapath_id
  180. config_get_bool disabled "$cfg" disabled 0
  181. [ "$disabled" == "0" ] || return
  182. config_get name "$cfg" name $cfg
  183. ovs-vsctl --may-exist add-br "$name"
  184. config_get datapath_id "$cfg" datapath_id
  185. [ -n "$datapath_id" ] && {
  186. ovs_bridge_validate_datapath_id "$datapath_id" && {
  187. ovs-vsctl --if-exists set bridge "$name" other-config:datapath-id="$datapath_id"
  188. }
  189. }
  190. config_get datapath_desc "$cfg" datapath_desc
  191. [ -n "$datapath_desc" ] && {
  192. ovs_bridge_validate_datapath_desc "$datapath_desc" && {
  193. ovs-vsctl --if-exists set bridge "$name" other-config:dp-desc="$datapath_desc"
  194. }
  195. }
  196. config_get fail_mode "$cfg" fail_mode
  197. [ -n "$fail_mode" ] && {
  198. ovs_bridge_validate_fail_mode "$fail_mode" && {
  199. ovs-vsctl set-fail-mode "$name" "$fail_mode" 2> /dev/null
  200. } || {
  201. ovs-vsctl del-fail-mode "$name" 2> /dev/null
  202. }
  203. } || {
  204. ovs-vsctl del-fail-mode "$name" 2> /dev/null
  205. }
  206. config_list_foreach "$cfg" "ports" ovs_bridge_port_add
  207. config_foreach ovs_bridge_port_add_complex ovs_port "$name"
  208. config_get_bool drop "$cfg" "drop_unknown_ports" 0
  209. [ "$drop" == 1 ] && ovs_bridge_port_cleanup
  210. config_get controller "$cfg" controller
  211. [ -n "$controller" ] && \
  212. ovs-vsctl set-controller "$name" "$controller"
  213. }
  214. ovs_set_ssl() {
  215. local ca="$(uci -q get openvswitch.ovs.ca)"
  216. [ -f "$ca" ] || return
  217. local cert="$(uci get openvswitch.ovs.cert)"
  218. [ -f "$cert" ] || return
  219. local key="$(uci get openvswitch.ovs.key)"
  220. [ -f "$key" ] || return
  221. ovs-vsctl set-ssl "$key" "$cert" "$ca"
  222. }