blob: 2cc5ea82e111c006c49f02dc159a0ac45d9470c6 [file] [log] [blame]
#
# Copyright 2017 The Cobalt Authors. All Rights Reserved.
#
# 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.
#
"""Abstraction for running Cobalt development tools."""
import abc
import os
import sys
from enum import IntEnum
from starboard.tools import build
from starboard.tools import paths
ARG_NOINSTALL = "noinstall"
ARG_SYSTOOLS = "systools"
ARG_DRYRUN = "dryrun"
IS_MODULAR_BUILD = os.getenv("MODULAR_BUILD", "0") == "1"
def _GetLauncherForPlatform(platform_name):
"""Gets the module containing a platform's concrete launcher implementation.
Args:
platform_name: Platform on which the app will be run, ex. "linux-x64x11".
Returns:
The module containing the platform's launcher implementation.
"""
gyp_config = build.GetPlatformConfig(platform_name)
if not hasattr(gyp_config, "GetLauncher"):
return None
else:
return gyp_config.GetLauncher()
def LauncherFactory(platform_name,
target_name,
config,
device_id=None,
target_params=None,
output_file=None,
out_directory=None,
env_variables=None,
test_result_xml_path=None,
**kwargs):
"""Creates the proper launcher based upon command line args.
Args:
platform_name: The platform on which the app will run.
target_name: The name of the executable target (ex. "cobalt").
config: Type of configuration used by the launcher (ex. "qa", "devel").
device_id: The identifier for the devkit being used. Can be None.
target_params: Command line arguments to the executable. Can be None.
output_file: Open file object used for storing the launcher's output. If
None, sys.stdout is used.
out_directory: Directory containing the executable target. If None is
provided, the path to the directory is dynamically generated.
env_variables: Environment variables for the executable.
**kwargs: Additional parameters to be passed to the launcher.
Returns:
An instance of the concrete launcher class for the desired platform.
Raises:
RuntimeError: The platform does not exist, or there is no project root.
"""
# Creates launcher for provided platform if the platform has a valid port.
launcher_module = _GetLauncherForPlatform(platform_name)
if not launcher_module:
raise RuntimeError("Cannot load launcher for given platform.")
return launcher_module.Launcher(
platform_name,
target_name,
config,
device_id,
target_params=target_params,
output_file=output_file,
out_directory=out_directory,
env_variables=env_variables,
test_result_xml_path=test_result_xml_path,
**kwargs)
class TargetStatus(IntEnum):
"""Represents status of the target run and its return code availability."""
# Target exited normally. Return code is available.
OK = 0
# Target exited normally. Return code is not available.
NA = 1
# Target crashed.
CRASH = 2
# Target not started.
NOT_STARTED = 3
@classmethod
def ToString(cls, status):
return [
"SUCCEEDED", "SUCCEEDED", "FAILED (CRASHED)", "FAILED (NOT STARTED)"
][status]
class AbstractLauncher(object):
"""
Class that specifies all required behavior for Cobalt app launchers.
The following is definition of extended interface. Implementation is optional.
Functions:
- InitDevice: a function to be called before any other performance
on the target device.
- RebootDevice: a function, used to reboot the target device.
- Run2: a function to run a target on device. Must be implemented
to support extended interface. The target must have been deployed
on the device, if required.
Returns:
a tuple of ReturnCodeStatus and, target return code if available,
else 0.
- Deploy: creates a package (if required) and deploys it on the target
device.
- CheckPackageIsDeployed: a function, which returns True, if the target
is deployed on the device, False otherwise. Must be implemented,
if Deploy is implemented.
- Kill: request the target OS to kill the target process. Must be
implemented.
- SendStop: sends stop signal to the launcher's executable.
"""
__metaclass__ = abc.ABCMeta
def __init__(self, platform_name, target_name, config, device_id, **kwargs):
self.platform_name = platform_name
self.target_name = target_name
self.config = config
self.device_id = device_id
# The following pattern makes sure that variables will be initialized
# properly whether a kwarg is passed in with a value of None or it
# is not passed in at all.
out_directory = kwargs.get("out_directory", None)
if not out_directory:
out_directory = paths.BuildOutputDirectory(platform_name, config)
self.out_directory = out_directory
self.coverage_file_path = kwargs.get("coverage_file_path", None)
output_file = kwargs.get("output_file", None)
if not output_file:
output_file = sys.stdout
self.output_file = output_file
target_command_line_params = kwargs.get("target_params", None)
if target_command_line_params is None:
target_command_line_params = []
self.target_command_line_params = target_command_line_params
env_variables = kwargs.get("env_variables", None)
if env_variables is None:
env_variables = {}
self.env_variables = env_variables
launcher_args = kwargs.get("launcher_args", None)
if launcher_args is None:
launcher_args = []
self.launcher_args = launcher_args
# Launchers that need different startup timeout times should reassign
# this variable during initialization.
self.startup_timeout_seconds = 2 * 60
self.test_result_xml_path = kwargs.get("test_result_xml_path", None)
def HasExtendedInterface(self) -> bool:
return hasattr(self, "Run2")
@abc.abstractmethod
def Run(self):
"""Runs an underlying application. Supports launching target
executable on target device, or target executable directly.
Implementation is platform specific.
Must be implemented in subclasses.
Returns:
The return code from the underlying application, if available,
else, 0 in case of normal completion of the underlying application,
otherwise 1.
"""
pass
@abc.abstractmethod
def GetDeviceIp(self):
"""Gets the device IP. Must be implemented in subclasses."""
pass
@abc.abstractmethod
def GetDeviceOutputPath(self):
"""Writable path where test targets can output files."""
pass
def SupportsSuspendResume(self):
return False
# The Send*() functions are guaranteed to resolve sequences of calls (e.g.
# SendSuspend() followed immediately by SendResume()) in the order that they
# are sent.
def SendResume(self):
"""Sends resume signal to the launcher's executable.
Raises:
RuntimeError: Resume signal not supported on platform.
"""
raise RuntimeError("Resume not supported for this platform.")
def SendSuspend(self):
"""sends suspend signal to the launcher's executable.
When implementing this function, please coordinate with other functions
sending platform signals(e.g. SendResume()) to ensure Cobalt receives these
signals in the same order they are sent.
Raises:
RuntimeError: Suspend signal not supported on platform.
"""
raise RuntimeError("Suspend not supported for this platform.")
def SendConceal(self):
"""sends conceal signal to the launcher's executable.
Raises:
RuntimeError: Conceal signal not supported on platform.
"""
raise RuntimeError("Conceal not supported for this platform.")
def SendFocus(self):
"""sends focus signal to the launcher's executable.
Raises:
RuntimeError: focus signal not supported on platform.
"""
raise RuntimeError("Focus not supported for this platform.")
def SendFreeze(self):
"""sends freeze signal to the launcher's executable.
Raises:
RuntimeError: Freeze signal not supported on platform.
"""
raise RuntimeError("Freeze not supported for this platform.")
def SupportsDeepLink(self):
return False
def SendDeepLink(self, link):
"""Sends deep link to the launcher's executable.
Args:
link: Link to send to the executable.
Returns:
True if the link was sent, False if it wasn't.
Raises:
RuntimeError: Deep link not supported on platform.
"""
raise RuntimeError(
f"Deep link not supported for this platform (link {link} sent).")
# Not like SendSuspendResume sending signals to cobalt, system suspend and
# resume send system signals to suspend and resume cobalt process.
def SupportsSystemSuspendResume(self):
return False
def SendSystemResume(self):
"""sends a system signal to the resume the launcher's executable.
Raises:
RuntimeError: System resume signal not supported on platform.
"""
raise RuntimeError("System resume signal not supported for this platform.")
def SendSystemSuspend(self):
"""sends a system signal to suspend the current running executable.
Raises:
RuntimeError: System suspend signal not supported on platform.
"""
raise RuntimeError("System suspend signal not supported for this platform.")
def GetStartupTimeout(self):
"""Gets the number of seconds to wait before assuming a launcher timeout."""
return self.startup_timeout_seconds
def GetHostAndPortGivenPort(self, port):
"""Creates a host/port tuple for use on the target device.
This is used to gain access to a service on the target device, and
can be overridden when the host/port of that service is only accessible
at runtime and/or via forwarding.
Args:
port: Port number for the desired service on the target device.
Returns:
(Host, port) tuple for use in connecting to the target device.
"""
return self.device_id, port
def CreateDeviceToHostTunnel(self, host_port, device_port):
"""Creates a tunnel that transfers requests from device to host.
This is used by on-device processes to connect to services on the host.
Args:
host_port: The host_port to receive requests from the device.
device_port: The port on device to proxy on-device requests.
Returns:
True if succeed and false otherwise.
"""
del host_port
del device_port
return False
def RemoveDeviceToHostTunnel(self, host_port):
"""Removes the tunnel created from CreateDeviceToHostTunnel.
Args:
host_port: The host_port to receive requests from the device.
Returns:
True if succeed and false otherwise.
"""
del host_port
return False
def GetTargetPath(self):
"""Constructs the path to an executable target.
The default path returned by this method takes the form of:
"/path/to/out/<platform>_<config>/target_name".
Returns:
The path to an executable target.
"""
return os.path.abspath(os.path.join(self.out_directory, self.target_name))
def GetInstallTargetPath(self):
"""Constructs the path to an executable target in install dir.
The default path returned by this method takes the form of:
"/path/to/out/<platform>_<config>/install/target_name".
Returns:
The path to an executable target.
"""
return os.path.abspath(
os.path.join(self.out_directory, "install", self.target_name))