#!/bin/bash # # Copyright 2016 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # set -euo pipefail function set_route { local mode="${1}" local ip="${2}" local host="${3}" local gw="${4}" if [ "${mode}" = "ethernet" ]; then ${ip} route del "${host}" via "${gw}" else # FIXME: This *should* work for IPv6, but it returns EINVAL. ${ip} route add "${host}" via "${gw}" dbus-send --system --dest=org.chromium.flimflam --print-reply / \ org.chromium.flimflam.Manager.SetServiceOrder \ string:"vpn,wifi,ethernet,wimax,cellular" fi } function find_route { local mode="${1}" local host="${2}" local ip="ip -4" if [[ "${host}" = *:* ]]; then ip="ip -6" fi set_route "${mode}" "${ip}" "${host}" 192.168.231.254 exit 0 } function parse_netstat { local mode="${1}" while read -r proto recv_q send_q local foreign state; do if [[ "${proto}" = tcp* && \ ("${local}" = *:22 || "${local}" = *:2222) && \ "${state}" == ESTABLISHED ]]; then find_route "${mode}" "${foreign%:*}" exit 0 fi done echo "Could not find ssh connection in netstat" exit 1 } mode="${1:-}" if [ "${mode}" != "wifi" -a "${mode}" != "ethernet" ]; then echo "Tells shill to prioritize ethernet or wifi, and adds a route" echo "back to the ssh/adb host so that the device can still be controlled" echo "remotely." echo "" echo "usage: ${0} { ethernet | wifi }" exit 1 fi if [ "${mode}" = "ethernet" ]; then # Switch the service order first, because the IP lookup might fail. dbus-send --system --dest=org.chromium.flimflam --print-reply / \ org.chromium.flimflam.Manager.SetServiceOrder \ string:"vpn,ethernet,wifi,wimax,cellular" fi # Find the first connection to our local port 22 (ssh), then use it to # set a static route via eth0. # This should ideally use $SSH_CLIENT instead, but that will require enabling # transparent mode in sslh because $SSH_CLIENT currently points to # 127.0.0.1. netstat --tcp --numeric --wide | parse_netstat "${mode}" exit 0