From dae3ad1d57cd166f6921d24ae0adccc37c226e9d Mon Sep 17 00:00:00 2001 From: Peter Harris Date: Mon, 6 Jan 2025 22:09:30 +0000 Subject: [PATCH] Start multi-use installer --- .pylintrc | 3 +- lgl_android_install.py | 220 +++++++++++++++++++++++++++++++++++++++++ lgl_host_server.py | 76 ++++++++++---- lglpy/android/adb.py | 2 +- lglpy/android/utils.py | 18 ++-- lglpy/ui/console.py | 1 - pylint.sh | 45 +++++++++ 7 files changed, 332 insertions(+), 33 deletions(-) create mode 100644 lgl_android_install.py create mode 100644 pylint.sh diff --git a/.pylintrc b/.pylintrc index d87589b..98fa115 100644 --- a/.pylintrc +++ b/.pylintrc @@ -435,7 +435,8 @@ disable=raw-checker-failed, deprecated-pragma, use-implicit-booleaness-not-comparison-to-string, use-implicit-booleaness-not-comparison-to-zero, - use-symbolic-message-instead + use-symbolic-message-instead, + duplicate-code # Enable the message, report, category or checker with the given id(s). You can # either give multiple identifier separated by comma (,) or put this option diff --git a/lgl_android_install.py b/lgl_android_install.py new file mode 100644 index 0000000..032cda6 --- /dev/null +++ b/lgl_android_install.py @@ -0,0 +1,220 @@ +#!/bin/env python3 +# SPDX-License-Identifier: MIT +# ----------------------------------------------------------------------------- +# Copyright (c) 2019-2025 Arm Limited +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the 'Software'), to +# deal in the Software without restriction, including without limitation the +# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +# sell copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +# SOFTWARE. +# ----------------------------------------------------------------------------- + +''' +This script is a helper utility to install one or more Vulkan layers on to an +Android device for a given debuggable application package. +''' + +import argparse +import sys +from typing import Optional + +from lglpy.android.adb import ADBConnect +from lglpy.android.utils import AndroidUtils +from lglpy.ui import console + + +def get_device_name( + conn: ADBConnect, device_param: Optional[str]) -> Optional[str]: + ''' + Determine which connected device to use. + + If multiple devices are connected, and the user does not provide an + unambiguous selection, then the user will be prompted to select. + + Args: + conn: The adb connection. + device_param: The user specified device name from the command line, + or None for menu-driven selection. + + Returns: + The selected device, or None if no device was selected. + ''' + good_devices, bad_devices = AndroidUtils.get_devices() + + # Log bad devices + if bad_devices: + print('\nSearching for devices:') + for device in bad_devices: + print(f' Device {device} is connected, but is not debuggable') + + # No devices found so early out + if not good_devices: + print('ERROR: No debuggable device is connected') + return None + + # If user specified a name check it exists and is non-ambiguous + if device_param: + search = device_param.lower() + match = [x for x in good_devices if x.lower().startswith(search)] + + # User device not found ... + if not match: + print(f'ERROR: Device {device_param} is not connected') + return None + + # User device found too many times ... + if len(match) > 1: + print(f'ERROR: Device {device_param} is ambiguous') + return None + + # Unambiguous match + return match[0] + + # Build a more literate option list for the menu + options = [] + for device in good_devices: + conn.set_device(device) + meta = AndroidUtils.get_device_model(conn) + + if meta: + vendor = meta[0][0].upper() + meta[0][1:] + model = meta[1][0].upper() + meta[1][1:] + options.append(f'{vendor} {model} ({device})') + + else: + options.append(f'Unknown device ({device})') + + conn.set_device(None) + + # Else match via the menu (will auto-select if only one option) + selection = console.select_from_menu('device', options) + if selection is None: + return None + + return good_devices[selection] + + +def get_package_name( + conn: ADBConnect, package_param: Optional[str], + debuggable_only: bool = True) -> Optional[str]: + ''' + Determine which application package to use. + + Currently only supports selecting launchable packages with a MAIN intent. + + Args: + conn: The adb connection. + package_param: The user specified package name from the command line. + - May be the full package name (case-insensitive). + - May be a package name prefix (case-insensitive). + - May be auto-select from menu (set as None) + debuggable_only: Show only debuggable packages if True. + + Returns: + The selected package, or None if no package was selected. + ''' + packages = AndroidUtils.get_packages(conn, debuggable_only, True) + + # No packages found so early out + if not packages: + print('ERROR: No packages detected') + return None + + # If user specified a name check it exists and is non-ambiguous + if package_param: + search = package_param.lower() + match = [x for x in packages if x.lower().startswith(search)] + + # User device not found ... + if not match: + print(f'ERROR: Package {package_param} not found') + return None + + # User device found too many times ... + if len(match) > 1: + print(f'ERROR: Package {package_param} is ambiguous') + return None + + # Unambiguous match + return match[0] + + # Else match via the menu (will auto-select if only one option) + title = 'debuggable package' if debuggable_only else 'package' + selection = console.select_from_menu(title, packages) + + if selection is None: + return None + + return packages[selection] + + +def parse_cli() -> argparse.Namespace: + ''' + Parse the command line. + + Returns: + An argparse results object. + ''' + parser = argparse.ArgumentParser() + + parser.add_argument( + '--device', '-D', default=None, + help='target device name or name prefix (default=auto-detected)') + + parser.add_argument( + '--package', '-P', default=None, + help='target package name or regex pattern (default=auto-detected)') + + parser.add_argument( + '--layer', '-L', action='append', required=True, + help='layer name to install (can be repeated)') + + return parser.parse_args() + + +def main() -> int: + ''' + The script main function. + + Returns: + The process exit code. + ''' + args = parse_cli() + + conn = ADBConnect() + + # Select a device to connect to + device = get_device_name(conn, args.device) + if not device: + return 1 + + conn.set_device(device) + + # Select a package to instrument + package = get_package_name(conn, args.package) + if not package: + return 2 + + conn.set_package(package) + + return 0 + + +if __name__ == "__main__": + try: + sys.exit(main()) + except KeyboardInterrupt: + print("\n\nERROR: User interrupted execution") diff --git a/lgl_host_server.py b/lgl_host_server.py index 962117d..5154d11 100644 --- a/lgl_host_server.py +++ b/lgl_host_server.py @@ -1,3 +1,4 @@ +#!/bin/env python3 # SPDX-License-Identifier: MIT # ----------------------------------------------------------------------------- # Copyright (c) 2024 Arm Limited @@ -21,55 +22,88 @@ # SOFTWARE. # ----------------------------------------------------------------------------- -# This module implements a host server that provides services over the network -# to a layer running on a remote device. -# -# Run with ... -# adb reverse localabstract:lglcomms tcp:63412 -# +''' +This script implements a simple network server that provides services to +layer drivers running on the target device over a simple network protocol. + +Android devices using layers can tunnel their connection using adb reverse +to forward a Unix domain socket on the device to a TCP socket on the host. + + adb reverse localabstract:lglcomms tcp:63412 +''' + +import argparse import sys import threading +from typing import Any + +from lglpy.comms import server +from lglpy.comms import service_gpu_timeline +from lglpy.comms import service_test +from lglpy.comms import service_log + + +def parse_cli() -> argparse.Namespace: + ''' + Parse the command line. -import lglpy.comms.server as server -import lglpy.comms.service_gpu_timeline as service_gpu_timeline -import lglpy.comms.service_test as service_test -import lglpy.comms.service_log as service_log + Returns: + An argparse results object. + ''' + parser = argparse.ArgumentParser() + + parser.add_argument( + '--test', '-T', action='store_true', default=False, + help='enable the communications unit test helper service') + + return parser.parse_args() + + +def main() -> int: + ''' + The script main function. + + Returns: + The process exit code. + ''' + args = parse_cli() -def main(): # Create a server instance - server = server.CommsServer(63412) + svr = server.CommsServer(63412) # Register all the services with it - print(f'Registering host services:') + print('Registering host services:') + + service: Any - if 0: + if args.test: service = service_test.TestService() - endpoint_id = server.register_endpoint(service) + endpoint_id = svr.register_endpoint(service) print(f' - [{endpoint_id}] = {service.get_service_name()}') service = service_log.LogService() - endpoint_id = server.register_endpoint(service) + endpoint_id = svr.register_endpoint(service) print(f' - [{endpoint_id}] = {service.get_service_name()}') service = service_gpu_timeline.GPUTimelineService() - endpoint_id = server.register_endpoint(service) + endpoint_id = svr.register_endpoint(service) print(f' - [{endpoint_id}] = {service.get_service_name()}') - print() # Start it running - serverThread = threading.Thread(target=server.run, daemon=True) - serverThread.start() + svr_thread = threading.Thread(target=svr.run, daemon=True) + svr_thread.start() # Press to exit try: input('Press any key to exit ...\n\n') except KeyboardInterrupt: print('Exiting ...') - sys.exit(0) + return 0 return 0 + if __name__ == '__main__': sys.exit(main()) diff --git a/lglpy/android/adb.py b/lglpy/android/adb.py index a41fc2d..f893fe3 100644 --- a/lglpy/android/adb.py +++ b/lglpy/android/adb.py @@ -57,7 +57,7 @@ def __init__(self, device: Optional[str] = None, self.device = device self.package = package - def set_device(self, device: str) -> None: + def set_device(self, device: Optional[str]) -> None: ''' Set the device for this connection. diff --git a/lglpy/android/utils.py b/lglpy/android/utils.py index db7349b..60153f8 100644 --- a/lglpy/android/utils.py +++ b/lglpy/android/utils.py @@ -257,59 +257,59 @@ def get_package_data_dir(conn: ADBConnect): return None @staticmethod - def set_property(conn: ADBConnect, property: str, value: str) -> bool: + def set_property(conn: ADBConnect, prop: str, value: str) -> bool: ''' Set an Android system property to a value. Args: conn: The adb connection. - property: The name of the property to set. + prop: The name of the property to set. value: The desired value of the property. Returns: True on success, False otherwise. ''' try: - conn.adb('shell', 'setprop', property, value) + conn.adb('shell', 'setprop', prop, value) return True except sp.CalledProcessError: return False @staticmethod - def get_property(conn: ADBConnect, property: str) -> Optional[str]: + def get_property(conn: ADBConnect, prop: str) -> Optional[str]: ''' Get an Android system property value. Args: conn: The adb connection. - property: The name of the property to get. + prop: The name of the property to get. Returns: The value of the property on success, None otherwise. Note that deleted settings that do not exist will also return None. ''' try: - value = conn.adb('shell', 'getprop', property) + value = conn.adb('shell', 'getprop', prop) return value.strip() except sp.CalledProcessError: return None @staticmethod - def clear_property(conn: ADBConnect, property: str) -> bool: + def clear_property(conn: ADBConnect, prop: str) -> bool: ''' Set an Android system property to an empty value. Args: conn: The adb connection. - property: The name of the property to set. + prop: The name of the property to clear. Returns: True on success, False otherwise. ''' try: - conn.adb('shell', 'setprop', property, '""') + conn.adb('shell', 'setprop', prop, '""') return True except sp.CalledProcessError: diff --git a/lglpy/ui/console.py b/lglpy/ui/console.py index 9820771..42390d4 100644 --- a/lglpy/ui/console.py +++ b/lglpy/ui/console.py @@ -85,5 +85,4 @@ def select_from_menu(title: str, options: list[str]) -> Optional[int]: except ValueError: print(f'\n Please enter a value between 0 and {len(options)}') - print(f'\n Selected {options[selection]}') return selection diff --git a/pylint.sh b/pylint.sh new file mode 100644 index 0000000..bc92af7 --- /dev/null +++ b/pylint.sh @@ -0,0 +1,45 @@ +# SPDX-License-Identifier: MIT +# ----------------------------------------------------------------------------- +# Copyright (c)2025 Arm Limited +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the 'Software'), to +# deal in the Software without restriction, including without limitation the +# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +# sell copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +# SOFTWARE. +# ----------------------------------------------------------------------------- + +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "Running pycodestyle\n" +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "\n" +python3 -m pycodestyle ./generator ./lglpy *.py +if [ $? -eq 0 ]; then + echo "Success: no issues found" +fi + +printf "\n" +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "Running mypy\n" +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "\n" +python3 -m mypy ./generator ./lglpy *.py + +printf "\n" +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "Running pylint\n" +printf "= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = \n" +printf "\n" +python3 -m pylint ./generator ./lglpy *.py