# Copyright (c) 2013 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
import collections
import logging
import os.path
import time
import uuid
from autotest_lib.client.bin import site_utils
from autotest_lib.client.common_lib import error
from autotest_lib.client.common_lib.cros import path_utils
class PacketCapturesDisabledError(Exception):
"""Signifies that this remote host does not support packet captures."""
pass
# local_pcap_path refers to the path of the result on the local host.
# local_log_path refers to the tcpdump log file path on the local host.
CaptureResult = collections.namedtuple('CaptureResult',
['local_pcap_path', 'local_log_path'])
# The number of bytes needed for a probe request is hard to define,
# because the frame contents are variable (e.g. radiotap header may
# contain different fields, maybe SSID isn't the first tagged
# parameter?). The value here is 2x the largest frame size observed in
# a quick sample.
SNAPLEN_WIFI_PROBE_REQUEST = 600
TCPDUMP_START_TIMEOUT_SECONDS = 5
TCPDUMP_START_POLL_SECONDS = 0.1
def get_packet_capturer(host, host_description=None, cmd_ifconfig=None,
cmd_ip=None, cmd_iw=None, cmd_netdump=None,
ignore_failures=False):
cmd_ifconfig = (cmd_ifconfig or
path_utils.get_install_path('ifconfig', host=host))
cmd_iw = cmd_iw or path_utils.get_install_path('iw', host=host)
cmd_ip = cmd_ip or path_utils.get_install_path('ip', host=host)
cmd_netdump = (cmd_netdump or
path_utils.get_install_path('tcpdump', host=host))
host_description = host_description or 'cap_%s' % uuid.uuid4().hex
if None in [cmd_ifconfig, cmd_iw, cmd_ip, cmd_netdump, host_description]:
if ignore_failures:
logging.warning('Creating a disabled packet capturer for %s.',
host_description)
return DisabledPacketCapturer()
else:
raise error.TestFail('Missing commands needed for '
'capturing packets')
return PacketCapturer(host, host_description, cmd_ifconfig, cmd_ip, cmd_iw,
cmd_netdump)
class DisabledPacketCapturer(object):
"""Delegate meant to look like it could take packet captures."""
@property
def capture_running(self):
"""@return False"""
return False
def __init__(self):
pass
def __enter__(self):
return self
def __exit__(self):
pass
def close(self):
"""No-op"""
def create_raw_monitor(self, phy, frequency, ht_type=None,
monitor_device=None):
"""Appears to fail while creating a raw monitor device.
@param phy string ignored.
@param frequency int ignored.
@param ht_type string ignored.
@param monitor_device string ignored.
@return None.
"""
return None
def configure_raw_monitor(self, monitor_device, frequency, ht_type=None):
"""Fails to configure a raw monitor.
@param monitor_device string ignored.
@param frequency int ignored.
@param ht_type string ignored.
"""
def create_managed_monitor(self, existing_dev, monitor_device=None):
"""Fails to create a managed monitor device.
@param existing_device string ignored.
@param monitor_device string ignored.
@return None
"""
return None
def start_capture(self, interface, local_save_dir,
remote_file=None, snaplen=None):
"""Fails to start a packet capture.
@param interface string ignored.
@param local_save_dir string ignored.
@param remote_file string ignored.
@param snaplen int ignored.
@raises PacketCapturesDisabledError.
"""
raise PacketCapturesDisabledError()
def stop_capture(self, capture_pid=None):
"""Stops all ongoing packet captures.
@param capture_pid int ignored.
"""
class PacketCapturer(object):
"""Delegate with capability to initiate packet captures on a remote host."""
LIBPCAP_POLL_FREQ_SECS = 1
@property
def capture_running(self):
"""@return True iff we have at least one ongoing packet capture."""
if self._ongoing_captures:
return True
return False
def __init__(self, host, host_description, cmd_ifconfig, cmd_ip,
cmd_iw, cmd_netdump, disable_captures=False):
self._cmd_netdump = cmd_netdump
self._cmd_iw = cmd_iw
self._cmd_ip = cmd_ip
self._cmd_ifconfig = cmd_ifconfig
self._host = host
self._ongoing_captures = {}
self._cap_num = 0
self._if_num = 0
self._created_managed_devices = []
self._created_raw_devices = []
self._host_description = host_description
def __enter__(self):
return self
def __exit__(self):
self.close()
def close(self):
"""Stop ongoing captures and destroy all created devices."""
self.stop_capture()
for device in self._created_managed_devices:
self._host.run("%s dev %s del" % (self._cmd_iw, device))
self._created_managed_devices = []
for device in self._created_raw_devices:
self._host.run("%s link set %s down" % (self._cmd_ip, device))
self._host.run("%s dev %s del" % (self._cmd_iw, device))
self._created_raw_devices = []
def create_raw_monitor(self, phy, frequency, ht_type=None,
monitor_device=None):
"""Create and configure a monitor type WiFi interface on a phy.
If a device called |monitor_device| already exists, it is first removed.
@param phy string phy name for created monitor (e.g. phy0).
@param frequency int frequency for created monitor to watch.
@param ht_type string optional HT type ('HT20', 'HT40+', or 'HT40-').
@param monitor_device string name of monitor interface to create.
@return string monitor device name created or None on failure.
"""
if not monitor_device:
monitor_device = 'mon%d' % self._if_num
self._if_num += 1
self._host.run('%s dev %s del' % (self._cmd_iw, monitor_device),
ignore_status=True)
result = self._host.run('%s phy %s interface add %s type monitor' %
(self._cmd_iw,
phy,
monitor_device),
ignore_status=True)
if result.exit_status:
logging.error('Failed creating raw monitor.')
return None
self.configure_raw_monitor(monitor_device, frequency, ht_type)
self._created_raw_devices.append(monitor_device)
return monitor_device
def configure_raw_monitor(self, monitor_device, frequency, ht_type=None):
"""Configure a raw monitor with frequency and HT params.
Note that this will stomp on earlier device settings.
@param monitor_device string name of device to configure.
@param frequency int WiFi frequency to dwell on.
@param ht_type string optional HT type ('HT20', 'HT40+', or 'HT40-').
"""
channel_args = str(frequency)
if ht_type:
ht_type = ht_type.upper()
channel_args = '%s %s' % (channel_args, ht_type)
if ht_type not in ('HT20', 'HT40+', 'HT40-'):
raise error.TestError('Cannot set HT mode: %s', ht_type)
self._host.run("%s link set %s up" % (self._cmd_ip, monitor_device))
self._host.run("%s dev %s set freq %s" % (self._cmd_iw,
monitor_device,
channel_args))
def create_managed_monitor(self, existing_dev, monitor_device=None):
"""Create a monitor type WiFi interface next to a managed interface.
If a device called |monitor_device| already exists, it is first removed.
@param existing_device string existing interface (e.g. mlan0).
@param monitor_device string name of monitor interface to create.
@return string monitor device name created or None on failure.
"""
if not monitor_device:
monitor_device = 'mon%d' % self._if_num
self._if_num += 1
self._host.run('%s dev %s del' % (self._cmd_iw, monitor_device),
ignore_status=True)
result = self._host.run('%s dev %s interface add %s type monitor' %
(self._cmd_iw,
existing_dev,
monitor_device),
ignore_status=True)
if result.exit_status:
logging.warning('Failed creating monitor.')
return None
self._host.run('%s %s up' % (self._cmd_ifconfig, monitor_device))
self._created_managed_devices.append(monitor_device)
return monitor_device
def _is_capture_active(self, remote_log_file):
"""Check if a packet capture has completed initialization.
@param remote_log_file string path to the capture's log file
@return True iff log file indicates that tcpdump is listening.
"""
return self._host.run(
'grep "listening on" "%s"' % remote_log_file, ignore_status=True
).exit_status == 0
def start_capture(self, interface, local_save_dir,
remote_file=None, snaplen=None):
"""Start a packet capture on an existing interface.
@param interface string existing interface to capture on.
@param local_save_dir string directory on local machine to hold results.
@param remote_file string full path on remote host to hold the capture.
@param snaplen int maximum captured frame length.
@return int pid of started packet capture.
"""
remote_file = (remote_file or
'/tmp/%s.%d.pcap' % (self._host_description,
self._cap_num))
self._cap_num += 1
remote_log_file = '%s.log' % remote_file
# Redirect output because SSH refuses to return until the child file
# descriptors are closed.
cmd = '%s -U -i %s -w %s -s %d >%s 2>&1 & echo $!' % (
self._cmd_netdump,
interface,
remote_file,
snaplen or 0,
remote_log_file)
logging.debug('Starting managed packet capture')
pid = int(self._host.run(cmd).stdout)
self._ongoing_captures[pid] = (remote_file,
remote_log_file,
local_save_dir)
is_capture_active = lambda: self._is_capture_active(remote_log_file)
site_utils.poll_for_condition(
is_capture_active,
timeout=TCPDUMP_START_TIMEOUT_SECONDS,
sleep_interval=TCPDUMP_START_POLL_SECONDS,
desc='Timeout waiting for tcpdump to start.')
return pid
def stop_capture(self, capture_pid=None, local_save_dir=None,
local_pcap_filename=None):
"""Stop an ongoing packet capture, or all ongoing packet captures.
If |capture_pid| is given, stops that capture, otherwise stops all
ongoing captures.
This method will sleep for a small amount of time, to ensure that
libpcap has completed its last poll(). The caller must ensure that
no unwanted traffic is received during this time.
@param capture_pid int pid of ongoing packet capture or None.
@param local_save_dir path to directory to save pcap file in locally.
@param local_pcap_filename name of file to store pcap in
(basename only).
@return list of RemoteCaptureResult tuples
"""
time.sleep(self.LIBPCAP_POLL_FREQ_SECS * 2)
if capture_pid:
pids_to_kill = [capture_pid]
else:
pids_to_kill = list(self._ongoing_captures.keys())
results = []
for pid in pids_to_kill:
self._host.run('kill -INT %d' % pid, ignore_status=True)
remote_pcap, remote_pcap_log, save_dir = self._ongoing_captures[pid]
pcap_filename = os.path.basename(remote_pcap)
pcap_log_filename = os.path.basename(remote_pcap_log)
if local_pcap_filename:
pcap_filename = os.path.join(local_save_dir or save_dir,
local_pcap_filename)
pcap_log_filename = os.path.join(local_save_dir or save_dir,
'%s.log' % local_pcap_filename)
pairs = [(remote_pcap, pcap_filename),
(remote_pcap_log, pcap_log_filename)]
for remote_file, local_file in pairs:
self._host.get_file(remote_file, local_file)
self._host.run('rm -f %s' % remote_file)
self._ongoing_captures.pop(pid)
results.append(CaptureResult(pcap_filename,
pcap_log_filename))
return results