mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-02-21 20:38:57 +00:00
Merge pull request #376 from setrofim/next
Implementing UIAutomator-based Android workloads
This commit is contained in:
commit
fc45b94b4e
9
.gitignore
vendored
9
.gitignore
vendored
@ -14,17 +14,10 @@ wa_output/
|
|||||||
doc/source/api/
|
doc/source/api/
|
||||||
doc/source/plugins/
|
doc/source/plugins/
|
||||||
MANIFEST
|
MANIFEST
|
||||||
wlauto/external/uiautomator/bin/
|
|
||||||
wlauto/external/uiautomator/*.properties
|
|
||||||
wlauto/external/uiautomator/build.xml
|
|
||||||
*.orig
|
*.orig
|
||||||
local.properties
|
local.properties
|
||||||
wlauto/external/revent/libs/
|
|
||||||
wlauto/external/revent/obj/
|
|
||||||
wlauto/external/bbench_server/libs/
|
|
||||||
wlauto/external/bbench_server/obj/
|
|
||||||
pmu_logger.mod.c
|
pmu_logger.mod.c
|
||||||
.tmp_versions
|
.tmp_versions
|
||||||
obj/
|
obj/
|
||||||
libs/armeabi
|
libs/armeabi
|
||||||
wlauto/workloads/*/uiauto/bin/
|
uiauto/bin/
|
||||||
|
@ -14,4 +14,4 @@ from wa.framework.plugin import Plugin, Parameter
|
|||||||
from wa.framework.processor import ResultProcessor
|
from wa.framework.processor import ResultProcessor
|
||||||
from wa.framework.resource import (NO_ONE, JarFile, ApkFile, ReventFile, File,
|
from wa.framework.resource import (NO_ONE, JarFile, ApkFile, ReventFile, File,
|
||||||
Executable)
|
Executable)
|
||||||
from wa.framework.workload import Workload
|
from wa.framework.workload import Workload, ApkUiautoWorkload
|
||||||
|
@ -30,7 +30,7 @@ import wa.framework.signal as signal
|
|||||||
from wa.framework import instrumentation, pluginloader
|
from wa.framework import instrumentation, pluginloader
|
||||||
from wa.framework.configuration.core import settings, Status
|
from wa.framework.configuration.core import settings, Status
|
||||||
from wa.framework.exception import (WAError, ConfigError, TimeoutError,
|
from wa.framework.exception import (WAError, ConfigError, TimeoutError,
|
||||||
InstrumentError, TargetError,
|
InstrumentError, TargetError, HostError,
|
||||||
TargetNotRespondingError)
|
TargetNotRespondingError)
|
||||||
from wa.framework.output import init_job_output
|
from wa.framework.output import init_job_output
|
||||||
from wa.framework.plugin import Artifact
|
from wa.framework.plugin import Artifact
|
||||||
@ -138,16 +138,21 @@ class ExecutionContext(object):
|
|||||||
self.current_job = self.job_queue.pop(0)
|
self.current_job = self.job_queue.pop(0)
|
||||||
self.current_job.output = init_job_output(self.run_output, self.current_job)
|
self.current_job.output = init_job_output(self.run_output, self.current_job)
|
||||||
self.update_job_state(self.current_job)
|
self.update_job_state(self.current_job)
|
||||||
|
self.tm.start()
|
||||||
return self.current_job
|
return self.current_job
|
||||||
|
|
||||||
def end_job(self):
|
def end_job(self):
|
||||||
if not self.current_job:
|
if not self.current_job:
|
||||||
raise RuntimeError('No jobs in progress')
|
raise RuntimeError('No jobs in progress')
|
||||||
|
self.tm.stop()
|
||||||
self.completed_jobs.append(self.current_job)
|
self.completed_jobs.append(self.current_job)
|
||||||
self.update_job_state(self.current_job)
|
self.update_job_state(self.current_job)
|
||||||
self.output.write_result()
|
self.output.write_result()
|
||||||
self.current_job = None
|
self.current_job = None
|
||||||
|
|
||||||
|
def extract_results(self):
|
||||||
|
self.tm.extract_results(self)
|
||||||
|
|
||||||
def move_failed(self, job):
|
def move_failed(self, job):
|
||||||
self.run_output.move_failed(job.output)
|
self.run_output.move_failed(job.output)
|
||||||
|
|
||||||
@ -173,6 +178,22 @@ class ExecutionContext(object):
|
|||||||
classifiers)
|
classifiers)
|
||||||
self.output.add_metric(name, value, units, lower_is_better, classifiers)
|
self.output.add_metric(name, value, units, lower_is_better, classifiers)
|
||||||
|
|
||||||
|
def get_artifact(self, name):
|
||||||
|
try:
|
||||||
|
return self.output.get_artifact(name)
|
||||||
|
except HostError:
|
||||||
|
if not self.current_job:
|
||||||
|
raise
|
||||||
|
return self.run_output.get_artifact(name)
|
||||||
|
|
||||||
|
def get_artifact_path(self, name):
|
||||||
|
try:
|
||||||
|
return self.output.get_artifact_path(name)
|
||||||
|
except HostError:
|
||||||
|
if not self.current_job:
|
||||||
|
raise
|
||||||
|
return self.run_output.get_artifact_path(name)
|
||||||
|
|
||||||
def add_artifact(self, name, path, kind, description=None, classifiers=None):
|
def add_artifact(self, name, path, kind, description=None, classifiers=None):
|
||||||
self.output.add_artifact(name, path, kind, description, classifiers)
|
self.output.add_artifact(name, path, kind, description, classifiers)
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@ def get_by_extension(path, ext):
|
|||||||
|
|
||||||
found = []
|
found = []
|
||||||
for entry in os.listdir(path):
|
for entry in os.listdir(path):
|
||||||
entry_ext = os.path.splitext(entry)
|
entry_ext = os.path.splitext(entry)[1]
|
||||||
if entry_ext == ext:
|
if entry_ext == ext:
|
||||||
found.append(os.path.join(path, entry))
|
found.append(os.path.join(path, entry))
|
||||||
return found
|
return found
|
||||||
@ -112,7 +112,8 @@ class UserDirectory(ResourceGetter):
|
|||||||
|
|
||||||
def get(self, resource):
|
def get(self, resource):
|
||||||
basepath = settings.dependencies_directory
|
basepath = settings.dependencies_directory
|
||||||
return get_from_location(basepath, resource)
|
directory = _d(os.path.join(basepath, resource.owner.name))
|
||||||
|
return get_from_location(directory, resource)
|
||||||
|
|
||||||
|
|
||||||
class Http(ResourceGetter):
|
class Http(ResourceGetter):
|
||||||
@ -318,7 +319,9 @@ class Filer(ResourceGetter):
|
|||||||
result = get_from_location(local_path, resource)
|
result = get_from_location(local_path, resource)
|
||||||
if result:
|
if result:
|
||||||
return result
|
return result
|
||||||
if remote_path:
|
if not os.path.exists(local_path):
|
||||||
|
return None
|
||||||
|
if os.path.exists(remote_path):
|
||||||
# Didn't find it cached locally; now check the remoted
|
# Didn't find it cached locally; now check the remoted
|
||||||
result = get_from_location(remote_path, resource)
|
result = get_from_location(remote_path, resource)
|
||||||
if not result:
|
if not result:
|
||||||
|
@ -125,8 +125,8 @@ SIGNAL_MAP = OrderedDict([
|
|||||||
('setup', signal.BEFORE_WORKLOAD_SETUP),
|
('setup', signal.BEFORE_WORKLOAD_SETUP),
|
||||||
('start', signal.BEFORE_WORKLOAD_EXECUTION),
|
('start', signal.BEFORE_WORKLOAD_EXECUTION),
|
||||||
('stop', signal.AFTER_WORKLOAD_EXECUTION),
|
('stop', signal.AFTER_WORKLOAD_EXECUTION),
|
||||||
('process_workload_result', signal.SUCCESSFUL_WORKLOAD_RESULT_UPDATE),
|
('process_workload_output', signal.SUCCESSFUL_WORKLOAD_OUTPUT_UPDATE),
|
||||||
('update_result', signal.AFTER_WORKLOAD_RESULT_UPDATE),
|
('update_result', signal.AFTER_WORKLOAD_OUTPUT_UPDATE),
|
||||||
('teardown', signal.AFTER_WORKLOAD_TEARDOWN),
|
('teardown', signal.AFTER_WORKLOAD_TEARDOWN),
|
||||||
('finalize', signal.RUN_FINALIZED),
|
('finalize', signal.RUN_FINALIZED),
|
||||||
|
|
||||||
|
@ -68,8 +68,11 @@ class Job(object):
|
|||||||
|
|
||||||
def process_output(self, context):
|
def process_output(self, context):
|
||||||
self.logger.info('Processing output for job {}'.format(self.id))
|
self.logger.info('Processing output for job {}'.format(self.id))
|
||||||
with signal.wrap('WORKLOAD_RESULT_UPDATE', self, context):
|
with signal.wrap('WORKLOAD_RESULT_EXTRACTION', self, context):
|
||||||
self.workload.update_result(context)
|
self.workload.extract_results(context)
|
||||||
|
context.extract_results()
|
||||||
|
with signal.wrap('WORKLOAD_OUTPUT_UPDATE', self, context):
|
||||||
|
self.workload.update_output(context)
|
||||||
|
|
||||||
def teardown(self, context):
|
def teardown(self, context):
|
||||||
self.logger.info('Tearing down job {}'.format(self.id))
|
self.logger.info('Tearing down job {}'.format(self.id))
|
||||||
|
@ -84,6 +84,13 @@ class Output(object):
|
|||||||
def add_event(self, message):
|
def add_event(self, message):
|
||||||
self.result.add_event(message)
|
self.result.add_event(message)
|
||||||
|
|
||||||
|
def get_artifact(self, name):
|
||||||
|
return self.result.get_artifact(name)
|
||||||
|
|
||||||
|
def get_artifact_path(self, name):
|
||||||
|
artifact = self.get_artifact(name)
|
||||||
|
return self.get_path(artifact.path)
|
||||||
|
|
||||||
|
|
||||||
class RunOutput(Output):
|
class RunOutput(Output):
|
||||||
|
|
||||||
@ -234,6 +241,12 @@ class Result(object):
|
|||||||
def add_event(self, message):
|
def add_event(self, message):
|
||||||
self.events.append(Event(message))
|
self.events.append(Event(message))
|
||||||
|
|
||||||
|
def get_artifact(self, name):
|
||||||
|
for artifact in self.artifacts:
|
||||||
|
if artifact.name == name:
|
||||||
|
return artifact
|
||||||
|
raise HostError('Artifact "{}" not found'.format(name))
|
||||||
|
|
||||||
def to_pod(self):
|
def to_pod(self):
|
||||||
return dict(
|
return dict(
|
||||||
status=str(self.status),
|
status=str(self.status),
|
||||||
@ -465,7 +478,7 @@ def init_job_output(run_output, job):
|
|||||||
path = os.path.join(run_output.basepath, output_name)
|
path = os.path.join(run_output.basepath, output_name)
|
||||||
ensure_directory_exists(path)
|
ensure_directory_exists(path)
|
||||||
write_pod(Result().to_pod(), os.path.join(path, 'result.json'))
|
write_pod(Result().to_pod(), os.path.join(path, 'result.json'))
|
||||||
job_output = JobOutput(path, job.id, job.iteration, job.label, job.retries)
|
job_output = JobOutput(path, job.id, job.label, job.iteration, job.retries)
|
||||||
job_output.status = job.status
|
job_output.status = job.status
|
||||||
run_output.jobs.append(job_output)
|
run_output.jobs.append(job_output)
|
||||||
return job_output
|
return job_output
|
||||||
|
@ -75,7 +75,7 @@ class Resource(object):
|
|||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '<{}\'s {}>'.format(self.owner, self.name)
|
return '<{}\'s {}>'.format(self.owner, self.kind)
|
||||||
|
|
||||||
|
|
||||||
class File(Resource):
|
class File(Resource):
|
||||||
|
@ -97,7 +97,8 @@ JOB_FINALIZED = Signal('job-finalized')
|
|||||||
|
|
||||||
|
|
||||||
# Signals associated with particular stages of workload execution
|
# Signals associated with particular stages of workload execution
|
||||||
BEFORE_WORKLOAD_INITIALIZED = Signal('before-workload-initialized', invert_priority=True)
|
BEFORE_WORKLOAD_INITIALIZED = Signal('before-workload-initialized',
|
||||||
|
invert_priority=True)
|
||||||
SUCCESSFUL_WORKLOAD_INITIALIZED = Signal('successful-workload-initialized')
|
SUCCESSFUL_WORKLOAD_INITIALIZED = Signal('successful-workload-initialized')
|
||||||
AFTER_WORKLOAD_INITIALIZED = Signal('after-workload-initialized')
|
AFTER_WORKLOAD_INITIALIZED = Signal('after-workload-initialized')
|
||||||
|
|
||||||
@ -109,9 +110,15 @@ BEFORE_WORKLOAD_EXECUTION = Signal('before-workload-execution', invert_priority=
|
|||||||
SUCCESSFUL_WORKLOAD_EXECUTION = Signal('successful-workload-execution')
|
SUCCESSFUL_WORKLOAD_EXECUTION = Signal('successful-workload-execution')
|
||||||
AFTER_WORKLOAD_EXECUTION = Signal('after-workload-execution')
|
AFTER_WORKLOAD_EXECUTION = Signal('after-workload-execution')
|
||||||
|
|
||||||
BEFORE_WORKLOAD_RESULT_UPDATE = Signal('before-workload-result-update', invert_priority=True)
|
BEFORE_WORKLOAD_RESULT_EXTRACTION = Signal('before-workload-result-exptracton',
|
||||||
SUCCESSFUL_WORKLOAD_RESULT_UPDATE = Signal('successful-workload-result-update')
|
invert_priority=True)
|
||||||
AFTER_WORKLOAD_RESULT_UPDATE = Signal('after-workload-result-update')
|
SUCCESSFUL_WORKLOAD_RESULT_EXTRACTION = Signal('successful-workload-result-exptracton')
|
||||||
|
AFTER_WORKLOAD_RESULT_EXTRACTION = Signal('after-workload-result-exptracton')
|
||||||
|
|
||||||
|
BEFORE_WORKLOAD_OUTPUT_UPDATE = Signal('before-workload-output-update',
|
||||||
|
invert_priority=True)
|
||||||
|
SUCCESSFUL_WORKLOAD_OUTPUT_UPDATE = Signal('successful-workload-output-update')
|
||||||
|
AFTER_WORKLOAD_OUTPUT_UPDATE = Signal('after-workload-output-update')
|
||||||
|
|
||||||
BEFORE_WORKLOAD_TEARDOWN = Signal('before-workload-teardown', invert_priority=True)
|
BEFORE_WORKLOAD_TEARDOWN = Signal('before-workload-teardown', invert_priority=True)
|
||||||
SUCCESSFUL_WORKLOAD_TEARDOWN = Signal('successful-workload-teardown')
|
SUCCESSFUL_WORKLOAD_TEARDOWN = Signal('successful-workload-teardown')
|
||||||
|
144
wa/framework/target/assistant.py
Normal file
144
wa/framework/target/assistant.py
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import sys
|
||||||
|
import tempfile
|
||||||
|
import threading
|
||||||
|
import time
|
||||||
|
|
||||||
|
from wa import Parameter
|
||||||
|
from wa.framework.exception import WorkerThreadError
|
||||||
|
|
||||||
|
|
||||||
|
class LinuxAssistant(object):
|
||||||
|
|
||||||
|
parameters = []
|
||||||
|
|
||||||
|
def __init__(self, target):
|
||||||
|
self.target = target
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def extract_results(self, context):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class AndroidAssistant(object):
|
||||||
|
|
||||||
|
parameters = [
|
||||||
|
Parameter('logcat_poll_period', kind=int,
|
||||||
|
constraint=lambda x: x > 0,
|
||||||
|
description="""
|
||||||
|
Polling period for logcat in seconds. If not specified,
|
||||||
|
no polling will be used.
|
||||||
|
|
||||||
|
Logcat buffer on android is of limited size and it cannot be
|
||||||
|
adjusted at run time. Depending on the amount of logging activity,
|
||||||
|
the buffer may not be enought to capture comlete trace for a
|
||||||
|
workload execution. For those situations, logcat may be polled
|
||||||
|
periodically during the course of the run and stored in a
|
||||||
|
temporary locaiton on the host. Setting the value of the poll
|
||||||
|
period enables this behavior.
|
||||||
|
"""),
|
||||||
|
]
|
||||||
|
|
||||||
|
def __init__(self, target, logcat_poll_period=None):
|
||||||
|
self.target = target
|
||||||
|
if logcat_poll_period:
|
||||||
|
self.logcat_poller = LogcatPoller(target, logcat_poll_period)
|
||||||
|
else:
|
||||||
|
self.logcat_poller = None
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
if self.logcat_poller:
|
||||||
|
self.logcat_poller.start()
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
if self.logcat_poller:
|
||||||
|
self.logcat_poller.stop()
|
||||||
|
|
||||||
|
def extract_results(self, context):
|
||||||
|
logcat_file = os.path.join(context.output_directory, 'logcat.log')
|
||||||
|
self.dump_logcat(logcat_file)
|
||||||
|
context.add_artifact('logcat', logcat_file, kind='log')
|
||||||
|
self.clear_logcat()
|
||||||
|
|
||||||
|
def dump_logcat(self, outfile):
|
||||||
|
if self.logcat_poller:
|
||||||
|
self.logcat_poller.write_log(outfile)
|
||||||
|
else:
|
||||||
|
self.target.dump_logcat(outfile)
|
||||||
|
|
||||||
|
def clear_logcat(self):
|
||||||
|
if self.logcat_poller:
|
||||||
|
self.logcat_poller.clear_buffer()
|
||||||
|
|
||||||
|
|
||||||
|
class LogcatPoller(threading.Thread):
|
||||||
|
|
||||||
|
def __init__(self, target, period=60, timeout=30):
|
||||||
|
super(LogcatPoller, self).__init__()
|
||||||
|
self.target = target
|
||||||
|
self.logger = logging.getLogger('logcat')
|
||||||
|
self.period = period
|
||||||
|
self.timeout = timeout
|
||||||
|
self.stop_signal = threading.Event()
|
||||||
|
self.lock = threading.Lock()
|
||||||
|
self.buffer_file = tempfile.mktemp()
|
||||||
|
self.last_poll = 0
|
||||||
|
self.daemon = True
|
||||||
|
self.exc = None
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
self.logger.debug('starting polling')
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
if self.stop_signal.is_set():
|
||||||
|
break
|
||||||
|
with self.lock:
|
||||||
|
current_time = time.time()
|
||||||
|
if (current_time - self.last_poll) >= self.period:
|
||||||
|
self.poll()
|
||||||
|
time.sleep(0.5)
|
||||||
|
except Exception: # pylint: disable=W0703
|
||||||
|
self.exc = WorkerThreadError(self.name, sys.exc_info())
|
||||||
|
self.logger.debug('polling stopped')
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
self.logger.debug('Stopping logcat polling')
|
||||||
|
self.stop_signal.set()
|
||||||
|
self.join(self.timeout)
|
||||||
|
if self.is_alive():
|
||||||
|
self.logger.error('Could not join logcat poller thread.')
|
||||||
|
if self.exc:
|
||||||
|
raise self.exc # pylint: disable=E0702
|
||||||
|
|
||||||
|
def clear_buffer(self):
|
||||||
|
self.logger.debug('clearing logcat buffer')
|
||||||
|
with self.lock:
|
||||||
|
self.target.clear_logcat()
|
||||||
|
with open(self.buffer_file, 'w') as _: # NOQA
|
||||||
|
pass
|
||||||
|
|
||||||
|
def write_log(self, outfile):
|
||||||
|
with self.lock:
|
||||||
|
self.poll()
|
||||||
|
if os.path.isfile(self.buffer_file):
|
||||||
|
shutil.copy(self.buffer_file, outfile)
|
||||||
|
else: # there was no logcat trace at this time
|
||||||
|
with open(outfile, 'w') as _: # NOQA
|
||||||
|
pass
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
self.logger.debug('closing poller')
|
||||||
|
if os.path.isfile(self.buffer_file):
|
||||||
|
os.remove(self.buffer_file)
|
||||||
|
|
||||||
|
def poll(self):
|
||||||
|
self.last_poll = time.time()
|
||||||
|
self.target.dump_logcat(self.buffer_file, append=True, timeout=self.timeout)
|
||||||
|
self.target.clear_logcat()
|
@ -7,6 +7,7 @@ from devlib import (LinuxTarget, AndroidTarget, LocalLinuxTarget,
|
|||||||
from wa.framework import pluginloader
|
from wa.framework import pluginloader
|
||||||
from wa.framework.exception import PluginLoaderError
|
from wa.framework.exception import PluginLoaderError
|
||||||
from wa.framework.plugin import Plugin, Parameter
|
from wa.framework.plugin import Plugin, Parameter
|
||||||
|
from wa.framework.target.assistant import LinuxAssistant, AndroidAssistant
|
||||||
from wa.utils.types import list_of_strings, list_of_ints
|
from wa.utils.types import list_of_strings, list_of_ints
|
||||||
from wa.utils.misc import isiterable
|
from wa.utils.misc import isiterable
|
||||||
|
|
||||||
@ -28,6 +29,7 @@ def instantiate_target(tdesc, params, connect=None):
|
|||||||
target_params = {p.name: p for p in tdesc.target_params}
|
target_params = {p.name: p for p in tdesc.target_params}
|
||||||
platform_params = {p.name: p for p in tdesc.platform_params}
|
platform_params = {p.name: p for p in tdesc.platform_params}
|
||||||
conn_params = {p.name: p for p in tdesc.conn_params}
|
conn_params = {p.name: p for p in tdesc.conn_params}
|
||||||
|
assistant_params = {p.name: p for p in tdesc.assistant_params}
|
||||||
|
|
||||||
tp, pp, cp = {}, {}, {}
|
tp, pp, cp = {}, {}, {}
|
||||||
|
|
||||||
@ -38,6 +40,8 @@ def instantiate_target(tdesc, params, connect=None):
|
|||||||
pp[name] = value
|
pp[name] = value
|
||||||
elif name in conn_params:
|
elif name in conn_params:
|
||||||
cp[name] = value
|
cp[name] = value
|
||||||
|
elif name in assistant_params:
|
||||||
|
pass
|
||||||
else:
|
else:
|
||||||
msg = 'Unexpected parameter for {}: {}'
|
msg = 'Unexpected parameter for {}: {}'
|
||||||
raise ValueError(msg.format(tdesc.name, name))
|
raise ValueError(msg.format(tdesc.name, name))
|
||||||
@ -53,17 +57,27 @@ def instantiate_target(tdesc, params, connect=None):
|
|||||||
return tdesc.target(**tp)
|
return tdesc.target(**tp)
|
||||||
|
|
||||||
|
|
||||||
|
def instantiate_assistant(tdesc, params, target):
|
||||||
|
assistant_params = {}
|
||||||
|
for param in tdesc.assistant_params:
|
||||||
|
if param.name in params:
|
||||||
|
assistant_params[param.name] = params[param.name]
|
||||||
|
return tdesc.assistant(target, **assistant_params)
|
||||||
|
|
||||||
|
|
||||||
class TargetDescription(object):
|
class TargetDescription(object):
|
||||||
|
|
||||||
def __init__(self, name, source, description=None, target=None, platform=None,
|
def __init__(self, name, source, description=None, target=None, platform=None,
|
||||||
conn=None, target_params=None, platform_params=None,
|
conn=None, assistant=None, target_params=None, platform_params=None,
|
||||||
conn_params=None):
|
conn_params=None, assistant_params=None):
|
||||||
self.name = name
|
self.name = name
|
||||||
self.source = source
|
self.source = source
|
||||||
self.description = description
|
self.description = description
|
||||||
self.target = target
|
self.target = target
|
||||||
self.platform = platform
|
self.platform = platform
|
||||||
self.connection = conn
|
self.connection = conn
|
||||||
|
self.assistant = assistant
|
||||||
|
self.assistant_params = assistant_params
|
||||||
self._set('target_params', target_params)
|
self._set('target_params', target_params)
|
||||||
self._set('platform_params', platform_params)
|
self._set('platform_params', platform_params)
|
||||||
self._set('conn_params', conn_params)
|
self._set('conn_params', conn_params)
|
||||||
@ -218,7 +232,7 @@ GEM5_PLATFORM_PARAMS = [
|
|||||||
'''),
|
'''),
|
||||||
]
|
]
|
||||||
|
|
||||||
# name --> (target_class, params_list, defaults)
|
# name --> (target_class, params_list, defaults, assistant_class)
|
||||||
TARGETS = {
|
TARGETS = {
|
||||||
'linux': (LinuxTarget, COMMON_TARGET_PARAMS, None),
|
'linux': (LinuxTarget, COMMON_TARGET_PARAMS, None),
|
||||||
'android': (AndroidTarget, COMMON_TARGET_PARAMS +
|
'android': (AndroidTarget, COMMON_TARGET_PARAMS +
|
||||||
@ -230,6 +244,13 @@ TARGETS = {
|
|||||||
'local': (LocalLinuxTarget, COMMON_TARGET_PARAMS, None),
|
'local': (LocalLinuxTarget, COMMON_TARGET_PARAMS, None),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# name --> assistant
|
||||||
|
ASSISTANTS = {
|
||||||
|
'linux': LinuxAssistant,
|
||||||
|
'android': AndroidAssistant,
|
||||||
|
'local': LinuxAssistant,
|
||||||
|
}
|
||||||
|
|
||||||
# name --> (platform_class, params_list, defaults)
|
# name --> (platform_class, params_list, defaults)
|
||||||
PLATFORMS = {
|
PLATFORMS = {
|
||||||
'generic': (Platform, COMMON_PLATFORM_PARAMS, None),
|
'generic': (Platform, COMMON_PLATFORM_PARAMS, None),
|
||||||
@ -267,6 +288,7 @@ class DefaultTargetDescriptor(TargetDescriptor):
|
|||||||
result = []
|
result = []
|
||||||
for target_name, target_tuple in TARGETS.iteritems():
|
for target_name, target_tuple in TARGETS.iteritems():
|
||||||
target, target_params = self._get_item(target_tuple)
|
target, target_params = self._get_item(target_tuple)
|
||||||
|
assistant = ASSISTANTS[target_name]
|
||||||
for platform_name, platform_tuple in PLATFORMS.iteritems():
|
for platform_name, platform_tuple in PLATFORMS.iteritems():
|
||||||
platform, platform_params = self._get_item(platform_tuple)
|
platform, platform_params = self._get_item(platform_tuple)
|
||||||
|
|
||||||
@ -274,8 +296,10 @@ class DefaultTargetDescriptor(TargetDescriptor):
|
|||||||
td = TargetDescription(name, self)
|
td = TargetDescription(name, self)
|
||||||
td.target = target
|
td.target = target
|
||||||
td.platform = platform
|
td.platform = platform
|
||||||
|
td.assistant = assistant
|
||||||
td.target_params = target_params
|
td.target_params = target_params
|
||||||
td.platform_params = platform_params
|
td.platform_params = platform_params
|
||||||
|
td.assistant_params = assistant.parameters
|
||||||
result.append(td)
|
result.append(td)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@ -10,7 +10,8 @@ from wa.framework import signal
|
|||||||
from wa.framework.exception import WorkerThreadError, ConfigError
|
from wa.framework.exception import WorkerThreadError, ConfigError
|
||||||
from wa.framework.plugin import Parameter
|
from wa.framework.plugin import Parameter
|
||||||
from wa.framework.target.descriptor import (get_target_descriptions,
|
from wa.framework.target.descriptor import (get_target_descriptions,
|
||||||
instantiate_target)
|
instantiate_target,
|
||||||
|
instantiate_assistant)
|
||||||
from wa.framework.target.info import TargetInfo
|
from wa.framework.target.info import TargetInfo
|
||||||
from wa.framework.target.runtime_config import (SysfileValuesRuntimeConfig,
|
from wa.framework.target.runtime_config import (SysfileValuesRuntimeConfig,
|
||||||
HotplugRuntimeConfig,
|
HotplugRuntimeConfig,
|
||||||
@ -62,7 +63,6 @@ class TargetManager(object):
|
|||||||
self.info = TargetInfo()
|
self.info = TargetInfo()
|
||||||
|
|
||||||
self._init_target()
|
self._init_target()
|
||||||
self._init_assistant()
|
|
||||||
self.runtime_configs = [cls(self.target) for cls in self.runtime_config_cls]
|
self.runtime_configs = [cls(self.target) for cls in self.runtime_config_cls]
|
||||||
|
|
||||||
def finalize(self):
|
def finalize(self):
|
||||||
@ -83,6 +83,15 @@ class TargetManager(object):
|
|||||||
if any(parameter in name for parameter in cfg.supported_parameters):
|
if any(parameter in name for parameter in cfg.supported_parameters):
|
||||||
cfg.add(name, self.parameters.pop(name))
|
cfg.add(name, self.parameters.pop(name))
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
self.assistant.start()
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
self.assistant.stop()
|
||||||
|
|
||||||
|
def extract_results(self, context):
|
||||||
|
self.assistant.extract_results(context)
|
||||||
|
|
||||||
@memoized
|
@memoized
|
||||||
def get_target_info(self):
|
def get_target_info(self):
|
||||||
return TargetInfo(self.target)
|
return TargetInfo(self.target)
|
||||||
@ -107,238 +116,12 @@ class TargetManager(object):
|
|||||||
if self.target_name not in target_map:
|
if self.target_name not in target_map:
|
||||||
raise ValueError('Unknown Target: {}'.format(self.target_name))
|
raise ValueError('Unknown Target: {}'.format(self.target_name))
|
||||||
tdesc = target_map[self.target_name]
|
tdesc = target_map[self.target_name]
|
||||||
|
|
||||||
self.target = instantiate_target(tdesc, self.parameters, connect=False)
|
self.target = instantiate_target(tdesc, self.parameters, connect=False)
|
||||||
|
|
||||||
with signal.wrap('TARGET_CONNECT'):
|
with signal.wrap('TARGET_CONNECT'):
|
||||||
self.target.connect()
|
self.target.connect()
|
||||||
self.logger.info('Setting up target')
|
self.logger.info('Setting up target')
|
||||||
self.target.setup()
|
self.target.setup()
|
||||||
|
|
||||||
def _init_assistant(self):
|
self.assistant = instantiate_assistant(tdesc, self.parameters, self.target)
|
||||||
# Create a corresponding target and target-assistant to help with
|
|
||||||
# platformy stuff?
|
|
||||||
if self.target.os == 'android':
|
|
||||||
self.assistant = AndroidAssistant(self.target)
|
|
||||||
elif self.target.os == 'linux':
|
|
||||||
self.assistant = LinuxAssistant(self.target) # pylint: disable=redefined-variable-type
|
|
||||||
else:
|
|
||||||
raise ValueError('Unknown Target OS: {}'.format(self.target.os))
|
|
||||||
|
|
||||||
|
|
||||||
class LinuxAssistant(object):
|
|
||||||
|
|
||||||
name = 'linux-assistant'
|
|
||||||
|
|
||||||
description = """
|
|
||||||
Performs configuration, instrumentation, etc. during runs on Linux targets.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, target, **kwargs):
|
|
||||||
self.target = target
|
|
||||||
# parameters = [
|
|
||||||
|
|
||||||
# Parameter('disconnect', kind=bool, default=False,
|
|
||||||
# description="""
|
|
||||||
# Specifies whether the target should be disconnected from
|
|
||||||
# at the end of the run.
|
|
||||||
# """),
|
|
||||||
# ]
|
|
||||||
|
|
||||||
# runtime_config_cls = [
|
|
||||||
# # order matters
|
|
||||||
# SysfileValuesRuntimeConfig,
|
|
||||||
# HotplugRuntimeConfig,
|
|
||||||
# CpufreqRuntimeConfig,
|
|
||||||
# CpuidleRuntimeConfig,
|
|
||||||
# ]
|
|
||||||
|
|
||||||
# def __init__(self, target, context, **kwargs):
|
|
||||||
# # super(LinuxTargetManager, self).__init__(target, context, **kwargs)
|
|
||||||
# self.target = target
|
|
||||||
# self.context = context
|
|
||||||
# self.info = TargetInfo()
|
|
||||||
# self.runtime_configs = [cls(target) for cls in self.runtime_config_cls]
|
|
||||||
|
|
||||||
# def __init__(self):
|
|
||||||
# # super(LinuxTargetManager, self).__init__(target, context, **kwargs)
|
|
||||||
# self.target = target
|
|
||||||
# self.info = TargetInfo()
|
|
||||||
# self.parameters = parameters
|
|
||||||
|
|
||||||
# self.info = TargetInfo()
|
|
||||||
# self.runtime_configs = [cls(target) for cls in self.runtime_config_cls]
|
|
||||||
|
|
||||||
# def initialize(self):
|
|
||||||
# # self.runtime_configs = [cls(self.target) for cls in self.runtime_config_cls]
|
|
||||||
# # if self.parameters:
|
|
||||||
# self.logger.info('Connecting to the device')
|
|
||||||
# with signal.wrap('TARGET_CONNECT'):
|
|
||||||
# self.target.connect()
|
|
||||||
# self.info.load(self.target)
|
|
||||||
# # info_file = os.path.join(self.context.info_directory, 'target.json')
|
|
||||||
# # with open(info_file, 'w') as wfh:
|
|
||||||
# # json.dump(self.info.to_pod(), wfh)
|
|
||||||
|
|
||||||
# def finalize(self, runner):
|
|
||||||
# self.logger.info('Disconnecting from the device')
|
|
||||||
# if self.disconnect:
|
|
||||||
# with signal.wrap('TARGET_DISCONNECT'):
|
|
||||||
# self.target.disconnect()
|
|
||||||
|
|
||||||
# def _add_parameters(self):
|
|
||||||
# for name, value in self.parameters.iteritems():
|
|
||||||
# self.add_parameter(name, value)
|
|
||||||
|
|
||||||
# def validate_runtime_parameters(self, parameters):
|
|
||||||
# self.clear()
|
|
||||||
# for name, value in parameters.iteritems():
|
|
||||||
# self.add_parameter(name, value)
|
|
||||||
# self.validate_parameters()
|
|
||||||
|
|
||||||
# def set_runtime_parameters(self, parameters):
|
|
||||||
# self.clear()
|
|
||||||
# for name, value in parameters.iteritems():
|
|
||||||
# self.add_parameter(name, value)
|
|
||||||
# self.set_parameters()
|
|
||||||
|
|
||||||
# def clear_parameters(self):
|
|
||||||
# for cfg in self.runtime_configs:
|
|
||||||
# cfg.clear()
|
|
||||||
|
|
||||||
# def add_parameter(self, name, value):
|
|
||||||
# for cfg in self.runtime_configs:
|
|
||||||
# if name in cfg.supported_parameters:
|
|
||||||
# cfg.add(name, value)
|
|
||||||
# return
|
|
||||||
# raise ConfigError('Unexpected runtime parameter "{}".'.format(name))
|
|
||||||
|
|
||||||
# def validate_parameters(self):
|
|
||||||
# for cfg in self.runtime_configs:
|
|
||||||
# cfg.validate()
|
|
||||||
|
|
||||||
# def set_parameters(self):
|
|
||||||
# for cfg in self.runtime_configs:
|
|
||||||
# cfg.set()
|
|
||||||
|
|
||||||
|
|
||||||
class AndroidAssistant(LinuxAssistant):
|
|
||||||
|
|
||||||
name = 'android-assistant'
|
|
||||||
description = """
|
|
||||||
Extends ``LinuxTargetManager`` with Android-specific operations.
|
|
||||||
"""
|
|
||||||
|
|
||||||
parameters = [
|
|
||||||
Parameter('logcat_poll_period', kind=int,
|
|
||||||
description="""
|
|
||||||
If specified, logcat will cached in a temporary file on the
|
|
||||||
host every ``logcat_poll_period`` seconds. This is useful for
|
|
||||||
longer job executions, where on-device logcat buffer may not be
|
|
||||||
big enough to capture output for the entire execution.
|
|
||||||
"""),
|
|
||||||
]
|
|
||||||
|
|
||||||
def __init__(self, target, **kwargs):
|
|
||||||
super(AndroidAssistant, self).__init__(target)
|
|
||||||
self.logcat_poll_period = kwargs.get('logcat_poll_period', None)
|
|
||||||
if self.logcat_poll_period:
|
|
||||||
self.logcat_poller = LogcatPoller(target, self.logcat_poll_period)
|
|
||||||
else:
|
|
||||||
self.logcat_poller = None
|
|
||||||
|
|
||||||
# def __init__(self, target, context, **kwargs):
|
|
||||||
# super(AndroidAssistant, self).__init__(target, context, **kwargs)
|
|
||||||
# self.logcat_poll_period = kwargs.get('logcat_poll_period', None)
|
|
||||||
# if self.logcat_poll_period:
|
|
||||||
# self.logcat_poller = LogcatPoller(target, self.logcat_poll_period)
|
|
||||||
# else:
|
|
||||||
# self.logcat_poller = None
|
|
||||||
|
|
||||||
# def next_job(self, job):
|
|
||||||
# super(AndroidAssistant, self).next_job(job)
|
|
||||||
# if self.logcat_poller:
|
|
||||||
# self.logcat_poller.start()
|
|
||||||
|
|
||||||
# def job_done(self, job):
|
|
||||||
# super(AndroidAssistant, self).job_done(job)
|
|
||||||
# if self.logcat_poller:
|
|
||||||
# self.logcat_poller.stop()
|
|
||||||
# outfile = os.path.join(self.context.output_directory, 'logcat.log')
|
|
||||||
# self.logger.debug('Dumping logcat to {}'.format(outfile))
|
|
||||||
# self.dump_logcat(outfile)
|
|
||||||
# self.clear()
|
|
||||||
|
|
||||||
def dump_logcat(self, outfile):
|
|
||||||
if self.logcat_poller:
|
|
||||||
self.logcat_poller.write_log(outfile)
|
|
||||||
else:
|
|
||||||
self.target.dump_logcat(outfile)
|
|
||||||
|
|
||||||
def clear_logcat(self):
|
|
||||||
if self.logcat_poller:
|
|
||||||
self.logcat_poller.clear_buffer()
|
|
||||||
|
|
||||||
|
|
||||||
class LogcatPoller(threading.Thread):
|
|
||||||
|
|
||||||
def __init__(self, target, period=60, timeout=30):
|
|
||||||
super(LogcatPoller, self).__init__()
|
|
||||||
self.target = target
|
|
||||||
self.logger = logging.getLogger('logcat')
|
|
||||||
self.period = period
|
|
||||||
self.timeout = timeout
|
|
||||||
self.stop_signal = threading.Event()
|
|
||||||
self.lock = threading.Lock()
|
|
||||||
self.buffer_file = tempfile.mktemp()
|
|
||||||
self.last_poll = 0
|
|
||||||
self.daemon = True
|
|
||||||
self.exc = None
|
|
||||||
|
|
||||||
def start(self):
|
|
||||||
self.logger.debug('starting polling')
|
|
||||||
try:
|
|
||||||
while True:
|
|
||||||
if self.stop_signal.is_set():
|
|
||||||
break
|
|
||||||
with self.lock:
|
|
||||||
current_time = time.time()
|
|
||||||
if (current_time - self.last_poll) >= self.period:
|
|
||||||
self.poll()
|
|
||||||
time.sleep(0.5)
|
|
||||||
except Exception: # pylint: disable=W0703
|
|
||||||
self.exc = WorkerThreadError(self.name, sys.exc_info())
|
|
||||||
self.logger.debug('polling stopped')
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
self.logger.debug('Stopping logcat polling')
|
|
||||||
self.stop_signal.set()
|
|
||||||
self.join(self.timeout)
|
|
||||||
if self.is_alive():
|
|
||||||
self.logger.error('Could not join logcat poller thread.')
|
|
||||||
if self.exc:
|
|
||||||
raise self.exc # pylint: disable=E0702
|
|
||||||
|
|
||||||
def clear_buffer(self):
|
|
||||||
self.logger.debug('clearing logcat buffer')
|
|
||||||
with self.lock:
|
|
||||||
self.target.clear_logcat()
|
|
||||||
with open(self.buffer_file, 'w') as _: # NOQA
|
|
||||||
pass
|
|
||||||
|
|
||||||
def write_log(self, outfile):
|
|
||||||
with self.lock:
|
|
||||||
self.poll()
|
|
||||||
if os.path.isfile(self.buffer_file):
|
|
||||||
shutil.copy(self.buffer_file, outfile)
|
|
||||||
else: # there was no logcat trace at this time
|
|
||||||
with open(outfile, 'w') as _: # NOQA
|
|
||||||
pass
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
self.logger.debug('closing poller')
|
|
||||||
if os.path.isfile(self.buffer_file):
|
|
||||||
os.remove(self.buffer_file)
|
|
||||||
|
|
||||||
def poll(self):
|
|
||||||
self.last_poll = time.time()
|
|
||||||
self.target.dump_logcat(self.buffer_file, append=True, timeout=self.timeout)
|
|
||||||
self.target.clear_logcat()
|
|
||||||
|
BIN
wa/framework/uiauto/BaseUiAutomation.class
Normal file
BIN
wa/framework/uiauto/BaseUiAutomation.class
Normal file
Binary file not shown.
21
wa/framework/uiauto/build.sh
Executable file
21
wa/framework/uiauto/build.sh
Executable file
@ -0,0 +1,21 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Copyright 2013-2015 ARM Limited
|
||||||
|
#
|
||||||
|
# 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 -e
|
||||||
|
|
||||||
|
|
||||||
|
ant build
|
||||||
|
|
||||||
|
cp bin/classes/com/arm/wa/uiauto/BaseUiAutomation.class .
|
92
wa/framework/uiauto/build.xml
Normal file
92
wa/framework/uiauto/build.xml
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project name="com.arm.wa.uiauto" default="help">
|
||||||
|
|
||||||
|
<!-- The local.properties file is created and updated by the 'android' tool.
|
||||||
|
It contains the path to the SDK. It should *NOT* be checked into
|
||||||
|
Version Control Systems. -->
|
||||||
|
<property file="local.properties" />
|
||||||
|
|
||||||
|
<!-- The ant.properties file can be created by you. It is only edited by the
|
||||||
|
'android' tool to add properties to it.
|
||||||
|
This is the place to change some Ant specific build properties.
|
||||||
|
Here are some properties you may want to change/update:
|
||||||
|
|
||||||
|
source.dir
|
||||||
|
The name of the source directory. Default is 'src'.
|
||||||
|
out.dir
|
||||||
|
The name of the output directory. Default is 'bin'.
|
||||||
|
|
||||||
|
For other overridable properties, look at the beginning of the rules
|
||||||
|
files in the SDK, at tools/ant/build.xml
|
||||||
|
|
||||||
|
Properties related to the SDK location or the project target should
|
||||||
|
be updated using the 'android' tool with the 'update' action.
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems.
|
||||||
|
|
||||||
|
-->
|
||||||
|
<property file="ant.properties" />
|
||||||
|
|
||||||
|
<!-- if sdk.dir was not set from one of the property file, then
|
||||||
|
get it from the ANDROID_HOME env var.
|
||||||
|
This must be done before we load project.properties since
|
||||||
|
the proguard config can use sdk.dir -->
|
||||||
|
<property environment="env" />
|
||||||
|
<condition property="sdk.dir" value="${env.ANDROID_HOME}">
|
||||||
|
<isset property="env.ANDROID_HOME" />
|
||||||
|
</condition>
|
||||||
|
|
||||||
|
<!-- The project.properties file is created and updated by the 'android'
|
||||||
|
tool, as well as ADT.
|
||||||
|
|
||||||
|
This contains project specific properties such as project target, and library
|
||||||
|
dependencies. Lower level build properties are stored in ant.properties
|
||||||
|
(or in .classpath for Eclipse projects).
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems. -->
|
||||||
|
<loadproperties srcFile="project.properties" />
|
||||||
|
|
||||||
|
<!-- quick check on sdk.dir -->
|
||||||
|
<fail
|
||||||
|
message="sdk.dir is missing. Make sure to generate local.properties using 'android update project' or to inject it through the ANDROID_HOME environment variable."
|
||||||
|
unless="sdk.dir"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
Import per project custom build rules if present at the root of the project.
|
||||||
|
This is the place to put custom intermediary targets such as:
|
||||||
|
-pre-build
|
||||||
|
-pre-compile
|
||||||
|
-post-compile (This is typically used for code obfuscation.
|
||||||
|
Compiled code location: ${out.classes.absolute.dir}
|
||||||
|
If this is not done in place, override ${out.dex.input.absolute.dir})
|
||||||
|
-post-package
|
||||||
|
-post-build
|
||||||
|
-pre-clean
|
||||||
|
-->
|
||||||
|
<import file="custom_rules.xml" optional="true" />
|
||||||
|
|
||||||
|
<!-- Import the actual build file.
|
||||||
|
|
||||||
|
To customize existing targets, there are two options:
|
||||||
|
- Customize only one target:
|
||||||
|
- copy/paste the target into this file, *before* the
|
||||||
|
<import> task.
|
||||||
|
- customize it to your needs.
|
||||||
|
- Customize the whole content of build.xml
|
||||||
|
- copy/paste the content of the rules files (minus the top node)
|
||||||
|
into this file, replacing the <import> task.
|
||||||
|
- customize to your needs.
|
||||||
|
|
||||||
|
***********************
|
||||||
|
****** IMPORTANT ******
|
||||||
|
***********************
|
||||||
|
In all cases you must update the value of version-tag below to read 'custom' instead of an integer,
|
||||||
|
in order to avoid having your file be overridden by tools such as "android update project"
|
||||||
|
-->
|
||||||
|
<!-- version-tag: VERSION_TAG -->
|
||||||
|
<import file="${sdk.dir}/tools/ant/uibuild.xml" />
|
||||||
|
|
||||||
|
</project>
|
14
wa/framework/uiauto/project.properties
Normal file
14
wa/framework/uiauto/project.properties
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# This file is automatically generated by Android Tools.
|
||||||
|
# Do not modify this file -- YOUR CHANGES WILL BE ERASED!
|
||||||
|
#
|
||||||
|
# This file must be checked in Version Control Systems.
|
||||||
|
#
|
||||||
|
# To customize properties used by the Ant build system edit
|
||||||
|
# "ant.properties", and override values to adapt the script to your
|
||||||
|
# project structure.
|
||||||
|
#
|
||||||
|
# To enable ProGuard to shrink and obfuscate your code, uncomment this (available properties: sdk.dir, user.home):
|
||||||
|
#proguard.config=${sdk.dir}/tools/proguard/proguard-android.txt:proguard-project.txt
|
||||||
|
|
||||||
|
# Project target.
|
||||||
|
target=android-17
|
124
wa/framework/uiauto/src/com/arm/wa/uiauto/BaseUiAutomation.java
Normal file
124
wa/framework/uiauto/src/com/arm/wa/uiauto/BaseUiAutomation.java
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
/* Copyright 2013-2015 ARM Limited
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
package com.arm.wa.uiauto;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.util.concurrent.TimeoutException;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.os.Bundle;
|
||||||
|
|
||||||
|
// Import the uiautomator libraries
|
||||||
|
import com.android.uiautomator.core.UiObject;
|
||||||
|
import com.android.uiautomator.core.UiObjectNotFoundException;
|
||||||
|
import com.android.uiautomator.core.UiScrollable;
|
||||||
|
import com.android.uiautomator.core.UiSelector;
|
||||||
|
import com.android.uiautomator.testrunner.UiAutomatorTestCase;
|
||||||
|
|
||||||
|
public class BaseUiAutomation extends UiAutomatorTestCase {
|
||||||
|
|
||||||
|
public void setup() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void runWorkload() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void extractResults() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void teardown() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void sleep(int second) {
|
||||||
|
super.sleep(second * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean takeScreenshot(String name) {
|
||||||
|
Bundle params = getParams();
|
||||||
|
String png_dir = params.getString("workdir");
|
||||||
|
|
||||||
|
try {
|
||||||
|
return getUiDevice().takeScreenshot(new File(png_dir, name + ".png"));
|
||||||
|
} catch(NoSuchMethodError e) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitText(String text) throws UiObjectNotFoundException {
|
||||||
|
waitText(text, 600);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitText(String text, int second) throws UiObjectNotFoundException {
|
||||||
|
UiSelector selector = new UiSelector();
|
||||||
|
UiObject text_obj = new UiObject(selector.text(text)
|
||||||
|
.className("android.widget.TextView"));
|
||||||
|
waitObject(text_obj, second);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitObject(UiObject obj) throws UiObjectNotFoundException {
|
||||||
|
waitObject(obj, 600);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitObject(UiObject obj, int second) throws UiObjectNotFoundException {
|
||||||
|
if (! obj.waitForExists(second * 1000)){
|
||||||
|
throw new UiObjectNotFoundException("UiObject is not found: "
|
||||||
|
+ obj.getSelector().toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean waitUntilNoObject(UiObject obj, int second) {
|
||||||
|
return obj.waitUntilGone(second * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clearLogcat() throws Exception {
|
||||||
|
Runtime.getRuntime().exec("logcat -c");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitForLogcatText(String searchText, long timeout) throws Exception {
|
||||||
|
long startTime = System.currentTimeMillis();
|
||||||
|
Process process = Runtime.getRuntime().exec("logcat");
|
||||||
|
BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
|
||||||
|
String line;
|
||||||
|
|
||||||
|
long currentTime = System.currentTimeMillis();
|
||||||
|
boolean found = false;
|
||||||
|
while ((currentTime - startTime) < timeout){
|
||||||
|
sleep(2); // poll every two seconds
|
||||||
|
|
||||||
|
while((line=reader.readLine())!=null) {
|
||||||
|
if (line.contains(searchText)) {
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
currentTime = System.currentTimeMillis();
|
||||||
|
}
|
||||||
|
|
||||||
|
process.destroy();
|
||||||
|
|
||||||
|
if ((currentTime - startTime) >= timeout) {
|
||||||
|
throw new TimeoutException("Timed out waiting for Logcat text \"%s\"".format(searchText));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -13,9 +13,11 @@
|
|||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
#
|
#
|
||||||
import logging
|
import logging
|
||||||
|
import os
|
||||||
|
import time
|
||||||
|
|
||||||
from wa.framework.plugin import TargetedPlugin
|
from wa.framework.plugin import TargetedPlugin
|
||||||
from wa.framework.resource import JarFile, ReventFile, NO_ONE
|
from wa.framework.resource import ApkFile, JarFile, ReventFile, NO_ONE
|
||||||
from wa.framework.exception import WorkloadError
|
from wa.framework.exception import WorkloadError
|
||||||
|
|
||||||
from devlib.utils.android import ApkInfo
|
from devlib.utils.android import ApkInfo
|
||||||
@ -66,10 +68,16 @@ class Workload(TargetedPlugin):
|
|||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def update_result(self, context):
|
def extract_results(self, context):
|
||||||
"""
|
"""
|
||||||
Update the result within the specified execution context with the
|
Extract results on the target
|
||||||
metrics form this workload iteration.
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
def update_output(self, context):
|
||||||
|
"""
|
||||||
|
Update the output within the specified execution context with the
|
||||||
|
metrics and artifacts form this workload iteration.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
@ -85,53 +93,117 @@ class Workload(TargetedPlugin):
|
|||||||
return '<Workload {}>'.format(self.name)
|
return '<Workload {}>'.format(self.name)
|
||||||
|
|
||||||
|
|
||||||
class UiAutomatorGUI(object):
|
class ApkUiautoWorkload(Workload):
|
||||||
|
|
||||||
|
platform = 'android'
|
||||||
|
|
||||||
def __init__(self, target, package='', klass='UiAutomation',
|
def __init__(self, target, **kwargs):
|
||||||
method='runUiAutoamtion'):
|
super(ApkUiautoWorkload, self).__init__(target, **kwargs)
|
||||||
self.target = target
|
self.apk = ApkHander(self)
|
||||||
self.uiauto_package = package
|
self.gui = UiAutomatorGUI(self)
|
||||||
self.uiauto_class = klass
|
|
||||||
self.uiauto_method = method
|
|
||||||
self.uiauto_file = None
|
|
||||||
self.target_uiauto_file = None
|
|
||||||
self.command = None
|
|
||||||
self.uiauto_params = {}
|
|
||||||
|
|
||||||
def init_resources(self, context):
|
def init_resources(self, context):
|
||||||
self.uiauto_file = context.resolver.get(JarFile(self))
|
self.apk.init_resources(context.resolver)
|
||||||
self.target_uiauto_file = self.target.path.join(self.target.working_directory,
|
self.gui.init_resources(context.resolver)
|
||||||
os.path.basename(self.uiauto_file))
|
self.gui.init_commands()
|
||||||
if not self.uiauto_package:
|
|
||||||
self.uiauto_package = os.path.splitext(os.path.basename(self.uiauto_file))[0]
|
|
||||||
|
|
||||||
def validate(self):
|
def initialize(self, context):
|
||||||
if not self.uiauto_file:
|
self.gui.deploy()
|
||||||
raise WorkloadError('No UI automation JAR file found for workload {}.'.format(self.name))
|
|
||||||
if not self.uiauto_package:
|
|
||||||
raise WorkloadError('No UI automation package specified for workload {}.'.format(self.name))
|
|
||||||
|
|
||||||
def setup(self, context):
|
def setup(self, context):
|
||||||
method_string = '{}.{}#{}'.format(self.uiauto_package, self.uiauto_class, self.uiauto_method)
|
self.apk.setup(context)
|
||||||
|
self.gui.setup()
|
||||||
|
|
||||||
|
def run(self, context):
|
||||||
|
self.gui.run()
|
||||||
|
|
||||||
|
def extract_results(self, context):
|
||||||
|
self.gui.extract_results()
|
||||||
|
|
||||||
|
def teardown(self, context):
|
||||||
|
self.gui.teardown()
|
||||||
|
self.apk.teardown()
|
||||||
|
|
||||||
|
def finalize(self, context):
|
||||||
|
self.gui.remove()
|
||||||
|
|
||||||
|
|
||||||
|
class UiAutomatorGUI(object):
|
||||||
|
|
||||||
|
stages = ['setup', 'runWorkload', 'extractResults', 'teardown']
|
||||||
|
|
||||||
|
def __init__(self, owner, package=None, klass='UiAutomation', timeout=600):
|
||||||
|
self.owner = owner
|
||||||
|
self.target = self.owner.target
|
||||||
|
self.uiauto_package = package
|
||||||
|
self.uiauto_class = klass
|
||||||
|
self.timeout = timeout
|
||||||
|
self.logger = logging.getLogger('gui')
|
||||||
|
self.jar_file = None
|
||||||
|
self.target_jar_file = None
|
||||||
|
self.commands = {}
|
||||||
|
self.uiauto_params = {}
|
||||||
|
|
||||||
|
def init_resources(self, resolver):
|
||||||
|
self.jar_file = resolver.get(JarFile(self.owner))
|
||||||
|
jar_name = os.path.basename(self.jar_file)
|
||||||
|
self.target_jar_file = self.target.get_workpath(jar_name)
|
||||||
|
if not self.uiauto_package:
|
||||||
|
package = os.path.splitext(os.path.basename(self.jar_file))[0]
|
||||||
|
self.uiauto_package = package
|
||||||
|
|
||||||
|
def init_commands(self):
|
||||||
params_dict = self.uiauto_params
|
params_dict = self.uiauto_params
|
||||||
params_dict['workdir'] = self.target.working_directory
|
params_dict['workdir'] = self.target.working_directory
|
||||||
params = ''
|
params = ''
|
||||||
for k, v in self.uiauto_params.iteritems():
|
for k, v in self.uiauto_params.iteritems():
|
||||||
params += ' -e {} {}'.format(k, v)
|
params += ' -e {} {}'.format(k, v)
|
||||||
self.command = 'uiautomator runtest {}{} -c {}'.format(self.target_uiauto_file, params, method_string)
|
|
||||||
self.target.push_file(self.uiauto_file, self.target_uiauto_file)
|
|
||||||
self.target.killall('uiautomator')
|
|
||||||
|
|
||||||
def run(self, context):
|
for stage in self.stages:
|
||||||
result = self.target.execute(self.command, self.run_timeout)
|
method_string = '{}.{}#{}'.format(self.uiauto_package,
|
||||||
|
self.uiauto_class,
|
||||||
|
stage)
|
||||||
|
cmd_template = 'uiautomator runtest {}{} -c {}'
|
||||||
|
self.commands[stage] = cmd_template.format(self.target_jar_file,
|
||||||
|
params, method_string)
|
||||||
|
|
||||||
|
def deploy(self):
|
||||||
|
self.target.push(self.jar_file, self.target_jar_file)
|
||||||
|
|
||||||
|
def set(self, name, value):
|
||||||
|
self.uiauto_params[name] = value
|
||||||
|
|
||||||
|
def setup(self, timeout=None):
|
||||||
|
if not self.commands:
|
||||||
|
raise RuntimeError('Commands have not been initialized')
|
||||||
|
self.target.killall('uiautomator')
|
||||||
|
self._execute('setup', timeout or self.timeout)
|
||||||
|
|
||||||
|
def run(self, timeout=None):
|
||||||
|
if not self.commands:
|
||||||
|
raise RuntimeError('Commands have not been initialized')
|
||||||
|
self._execute('runWorkload', timeout or self.timeout)
|
||||||
|
|
||||||
|
def extract_results(self, timeout=None):
|
||||||
|
if not self.commands:
|
||||||
|
raise RuntimeError('Commands have not been initialized')
|
||||||
|
self._execute('extractResults', timeout or self.timeout)
|
||||||
|
|
||||||
|
def teardown(self, timeout=None):
|
||||||
|
if not self.commands:
|
||||||
|
raise RuntimeError('Commands have not been initialized')
|
||||||
|
self._execute('teardown', timeout or self.timeout)
|
||||||
|
|
||||||
|
def remove(self):
|
||||||
|
self.target.remove(self.target_jar_file)
|
||||||
|
|
||||||
|
def _execute(self, stage, timeout):
|
||||||
|
result = self.target.execute(self.commands[stage], timeout)
|
||||||
if 'FAILURE' in result:
|
if 'FAILURE' in result:
|
||||||
raise WorkloadError(result)
|
raise WorkloadError(result)
|
||||||
else:
|
else:
|
||||||
self.logger.debug(result)
|
self.logger.debug(result)
|
||||||
time.sleep(DELAY)
|
time.sleep(2)
|
||||||
|
|
||||||
def teardown(self, context):
|
|
||||||
self.target.delete_file(self.target_uiauto_file)
|
|
||||||
|
|
||||||
|
|
||||||
class ReventGUI(object):
|
class ReventGUI(object):
|
||||||
@ -191,21 +263,27 @@ class ReventGUI(object):
|
|||||||
|
|
||||||
class ApkHander(object):
|
class ApkHander(object):
|
||||||
|
|
||||||
def __init__(self, owner, target, view, install_timeout=300, version=None,
|
def __init__(self, owner, install_timeout=300, version=None, variant=None,
|
||||||
strict=True, force_install=False, uninstall=False):
|
strict=True, force_install=False, uninstall=False):
|
||||||
self.logger = logging.getLogger('apk')
|
self.logger = logging.getLogger('apk')
|
||||||
self.owner = owner
|
self.owner = owner
|
||||||
self.target = target
|
self.target = self.owner.target
|
||||||
|
self.install_timeout = install_timeout
|
||||||
self.version = version
|
self.version = version
|
||||||
|
self.variant = variant
|
||||||
|
self.strict = strict
|
||||||
|
self.force_install = force_install
|
||||||
|
self.uninstall = uninstall
|
||||||
self.apk_file = None
|
self.apk_file = None
|
||||||
self.apk_info = None
|
self.apk_info = None
|
||||||
self.apk_version = None
|
self.apk_version = None
|
||||||
self.logcat_log = None
|
self.logcat_log = None
|
||||||
|
|
||||||
def init_resources(self, context):
|
def init_resources(self, resolver):
|
||||||
self.apk_file = context.resolver.get(ApkFile(self.owner),
|
self.apk_file = resolver.get(ApkFile(self.owner,
|
||||||
version=self.version,
|
variant=self.variant,
|
||||||
strict=strict)
|
version=self.version),
|
||||||
|
strict=self.strict)
|
||||||
self.apk_info = ApkInfo(self.apk_file)
|
self.apk_info = ApkInfo(self.apk_file)
|
||||||
|
|
||||||
def setup(self, context):
|
def setup(self, context):
|
||||||
@ -220,16 +298,17 @@ class ApkHander(object):
|
|||||||
self.initialize_with_host_apk(context, installed_version)
|
self.initialize_with_host_apk(context, installed_version)
|
||||||
else:
|
else:
|
||||||
if not installed_version:
|
if not installed_version:
|
||||||
message = '''{} not found found on the device and check_apk is set to "False"
|
message = '{} not found found on the device and check_apk is set '\
|
||||||
so host version was not checked.'''
|
'to "False" so host version was not checked.'
|
||||||
raise WorkloadError(message.format(self.package))
|
raise WorkloadError(message.format(self.apk_info.package))
|
||||||
message = 'Version {} installed on device; skipping host APK check.'
|
message = 'Version {} installed on device; skipping host APK check.'
|
||||||
self.logger.debug(message.format(installed_version))
|
self.logger.debug(message.format(installed_version))
|
||||||
self.reset(context)
|
self.reset(context)
|
||||||
self.version = installed_version
|
self.version = installed_version
|
||||||
|
|
||||||
def initialize_with_host_apk(self, context, installed_version):
|
def initialize_with_host_apk(self, context, installed_version):
|
||||||
if installed_version != self.apk_file.version_name:
|
host_version = self.apk_info.version_name
|
||||||
|
if installed_version != host_version:
|
||||||
if installed_version:
|
if installed_version:
|
||||||
message = '{} host version: {}, device version: {}; re-installing...'
|
message = '{} host version: {}, device version: {}; re-installing...'
|
||||||
self.logger.debug(message.format(os.path.basename(self.apk_file),
|
self.logger.debug(message.format(os.path.basename(self.apk_file),
|
||||||
@ -245,42 +324,38 @@ class ApkHander(object):
|
|||||||
host_version))
|
host_version))
|
||||||
if self.force_install:
|
if self.force_install:
|
||||||
if installed_version:
|
if installed_version:
|
||||||
self.device.uninstall(self.package)
|
self.target.uninstall_package(self.apk_info.package)
|
||||||
self.install_apk(context)
|
self.install_apk(context)
|
||||||
else:
|
else:
|
||||||
self.reset(context)
|
self.reset(context)
|
||||||
self.apk_version = host_version
|
self.apk_version = host_version
|
||||||
|
|
||||||
def start_activity(self):
|
def start_activity(self):
|
||||||
output = self.device.execute('am start -W -n {}/{}'.format(self.package, self.activity))
|
cmd = 'am start -W -n {}/{}'
|
||||||
|
output = self.target.execute(cmd.format(self.apk_info.package,
|
||||||
|
self.apk_info.activity))
|
||||||
if 'Error:' in output:
|
if 'Error:' in output:
|
||||||
self.device.execute('am force-stop {}'.format(self.package)) # this will dismiss any erro dialogs
|
# this will dismiss any error dialogs
|
||||||
|
self.target.execute('am force-stop {}'.format(self.apk_info.package))
|
||||||
raise WorkloadError(output)
|
raise WorkloadError(output)
|
||||||
self.logger.debug(output)
|
self.logger.debug(output)
|
||||||
|
|
||||||
def reset(self, context): # pylint: disable=W0613
|
def reset(self, context): # pylint: disable=W0613
|
||||||
self.device.execute('am force-stop {}'.format(self.package))
|
self.target.execute('am force-stop {}'.format(self.apk_info.package))
|
||||||
self.device.execute('pm clear {}'.format(self.package))
|
self.target.execute('pm clear {}'.format(self.apk_info.package))
|
||||||
|
|
||||||
def install_apk(self, context):
|
def install_apk(self, context):
|
||||||
output = self.device.install(self.apk_file, self.install_timeout)
|
output = self.target.install_apk(self.apk_file, self.install_timeout)
|
||||||
if 'Failure' in output:
|
if 'Failure' in output:
|
||||||
if 'ALREADY_EXISTS' in output:
|
if 'ALREADY_EXISTS' in output:
|
||||||
self.logger.warn('Using already installed APK (did not unistall properly?)')
|
msg = 'Using already installed APK (did not unistall properly?)'
|
||||||
|
self.logger.warn(msg)
|
||||||
else:
|
else:
|
||||||
raise WorkloadError(output)
|
raise WorkloadError(output)
|
||||||
else:
|
else:
|
||||||
self.logger.debug(output)
|
self.logger.debug(output)
|
||||||
|
|
||||||
def update_result(self, context):
|
def teardown(self):
|
||||||
self.logcat_log = os.path.join(context.output_directory, 'logcat.log')
|
self.target.execute('am force-stop {}'.format(self.apk_info.package))
|
||||||
self.device.dump_logcat(self.logcat_log)
|
if self.uninstall:
|
||||||
context.add_iteration_artifact(name='logcat',
|
self.target.uninstall_package(self.apk_info.package)
|
||||||
path='logcat.log',
|
|
||||||
kind='log',
|
|
||||||
description='Logact dump for the run.')
|
|
||||||
|
|
||||||
def teardown(self, context):
|
|
||||||
self.device.execute('am force-stop {}'.format(self.package))
|
|
||||||
if self.uninstall_apk:
|
|
||||||
self.device.uninstall(self.package)
|
|
||||||
|
62
wa/workloads/benchmarkpi/__init__.py
Normal file
62
wa/workloads/benchmarkpi/__init__.py
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
# Copyright 2013-2015 ARM Limited
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
|
||||||
|
import re
|
||||||
|
|
||||||
|
from wa import ApkUiautoWorkload
|
||||||
|
|
||||||
|
|
||||||
|
class BenchmarkPi(ApkUiautoWorkload):
|
||||||
|
|
||||||
|
name = 'benchmarkpi'
|
||||||
|
description = """
|
||||||
|
Measures the time the target device takes to run and complete the Pi
|
||||||
|
calculation algorithm.
|
||||||
|
|
||||||
|
http://androidbenchmark.com/howitworks.php
|
||||||
|
|
||||||
|
from the website:
|
||||||
|
|
||||||
|
The whole idea behind this application is to use the same Pi calculation
|
||||||
|
algorithm on every Android Device and check how fast that proccess is.
|
||||||
|
Better calculation times, conclude to faster Android devices. This way you
|
||||||
|
can also check how lightweight your custom made Android build is. Or not.
|
||||||
|
|
||||||
|
As Pi is an irrational number, Benchmark Pi does not calculate the actual Pi
|
||||||
|
number, but an approximation near the first digits of Pi over the same
|
||||||
|
calculation circles the algorithms needs.
|
||||||
|
|
||||||
|
So, the number you are getting in miliseconds is the time your mobile device
|
||||||
|
takes to run and complete the Pi calculation algorithm resulting in a
|
||||||
|
approximation of the first Pi digits.
|
||||||
|
"""
|
||||||
|
package = 'gr.androiddev.BenchmarkPi'
|
||||||
|
activity = '.BenchmarkPi'
|
||||||
|
|
||||||
|
regex = re.compile('You calculated Pi in ([0-9]+)')
|
||||||
|
|
||||||
|
def update_output(self, context):
|
||||||
|
super(BenchmarkPi, self).update_output(context)
|
||||||
|
logcat_file = context.get_artifact_path('logcat')
|
||||||
|
with open(logcat_file) as fh:
|
||||||
|
for line in fh:
|
||||||
|
match = self.regex.search(line)
|
||||||
|
if match:
|
||||||
|
result = int(match.group(1))
|
||||||
|
|
||||||
|
if result is not None:
|
||||||
|
context.add_metric('pi calculation', result,
|
||||||
|
'milliseconds', lower_is_better=True)
|
BIN
wa/workloads/benchmarkpi/com.arm.wa.uiauto.benchmarkpi.jar
Normal file
BIN
wa/workloads/benchmarkpi/com.arm.wa.uiauto.benchmarkpi.jar
Normal file
Binary file not shown.
27
wa/workloads/benchmarkpi/uiauto/build.sh
Executable file
27
wa/workloads/benchmarkpi/uiauto/build.sh
Executable file
@ -0,0 +1,27 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Copyright 2013-2015 ARM Limited
|
||||||
|
#
|
||||||
|
# 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 -e
|
||||||
|
|
||||||
|
class_dir=bin/classes/com/arm/wa/uiauto
|
||||||
|
base_class=`python -c "import os, wa; print os.path.join(os.path.dirname(wa.__file__), 'framework', 'uiauto', 'BaseUiAutomation.class')"`
|
||||||
|
mkdir -p $class_dir
|
||||||
|
cp $base_class $class_dir
|
||||||
|
|
||||||
|
ant build
|
||||||
|
|
||||||
|
if [[ -f bin/com.arm.wa.uiauto.benchmarkpi.jar ]]; then
|
||||||
|
cp bin/com.arm.wa.uiauto.benchmarkpi.jar ..
|
||||||
|
fi
|
92
wa/workloads/benchmarkpi/uiauto/build.xml
Normal file
92
wa/workloads/benchmarkpi/uiauto/build.xml
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project name="com.arm.wa.uiauto.benchmarkpi" default="help">
|
||||||
|
|
||||||
|
<!-- The local.properties file is created and updated by the 'android' tool.
|
||||||
|
It contains the path to the SDK. It should *NOT* be checked into
|
||||||
|
Version Control Systems. -->
|
||||||
|
<property file="local.properties" />
|
||||||
|
|
||||||
|
<!-- The ant.properties file can be created by you. It is only edited by the
|
||||||
|
'android' tool to add properties to it.
|
||||||
|
This is the place to change some Ant specific build properties.
|
||||||
|
Here are some properties you may want to change/update:
|
||||||
|
|
||||||
|
source.dir
|
||||||
|
The name of the source directory. Default is 'src'.
|
||||||
|
out.dir
|
||||||
|
The name of the output directory. Default is 'bin'.
|
||||||
|
|
||||||
|
For other overridable properties, look at the beginning of the rules
|
||||||
|
files in the SDK, at tools/ant/build.xml
|
||||||
|
|
||||||
|
Properties related to the SDK location or the project target should
|
||||||
|
be updated using the 'android' tool with the 'update' action.
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems.
|
||||||
|
|
||||||
|
-->
|
||||||
|
<property file="ant.properties" />
|
||||||
|
|
||||||
|
<!-- if sdk.dir was not set from one of the property file, then
|
||||||
|
get it from the ANDROID_HOME env var.
|
||||||
|
This must be done before we load project.properties since
|
||||||
|
the proguard config can use sdk.dir -->
|
||||||
|
<property environment="env" />
|
||||||
|
<condition property="sdk.dir" value="${env.ANDROID_HOME}">
|
||||||
|
<isset property="env.ANDROID_HOME" />
|
||||||
|
</condition>
|
||||||
|
|
||||||
|
<!-- The project.properties file is created and updated by the 'android'
|
||||||
|
tool, as well as ADT.
|
||||||
|
|
||||||
|
This contains project specific properties such as project target, and library
|
||||||
|
dependencies. Lower level build properties are stored in ant.properties
|
||||||
|
(or in .classpath for Eclipse projects).
|
||||||
|
|
||||||
|
This file is an integral part of the build system for your
|
||||||
|
application and should be checked into Version Control Systems. -->
|
||||||
|
<loadproperties srcFile="project.properties" />
|
||||||
|
|
||||||
|
<!-- quick check on sdk.dir -->
|
||||||
|
<fail
|
||||||
|
message="sdk.dir is missing. Make sure to generate local.properties using 'android update project' or to inject it through the ANDROID_HOME environment variable."
|
||||||
|
unless="sdk.dir"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
Import per project custom build rules if present at the root of the project.
|
||||||
|
This is the place to put custom intermediary targets such as:
|
||||||
|
-pre-build
|
||||||
|
-pre-compile
|
||||||
|
-post-compile (This is typically used for code obfuscation.
|
||||||
|
Compiled code location: ${out.classes.absolute.dir}
|
||||||
|
If this is not done in place, override ${out.dex.input.absolute.dir})
|
||||||
|
-post-package
|
||||||
|
-post-build
|
||||||
|
-pre-clean
|
||||||
|
-->
|
||||||
|
<import file="custom_rules.xml" optional="true" />
|
||||||
|
|
||||||
|
<!-- Import the actual build file.
|
||||||
|
|
||||||
|
To customize existing targets, there are two options:
|
||||||
|
- Customize only one target:
|
||||||
|
- copy/paste the target into this file, *before* the
|
||||||
|
<import> task.
|
||||||
|
- customize it to your needs.
|
||||||
|
- Customize the whole content of build.xml
|
||||||
|
- copy/paste the content of the rules files (minus the top node)
|
||||||
|
into this file, replacing the <import> task.
|
||||||
|
- customize to your needs.
|
||||||
|
|
||||||
|
***********************
|
||||||
|
****** IMPORTANT ******
|
||||||
|
***********************
|
||||||
|
In all cases you must update the value of version-tag below to read 'custom' instead of an integer,
|
||||||
|
in order to avoid having your file be overridden by tools such as "android update project"
|
||||||
|
-->
|
||||||
|
<!-- version-tag: VERSION_TAG -->
|
||||||
|
<import file="${sdk.dir}/tools/ant/uibuild.xml" />
|
||||||
|
|
||||||
|
</project>
|
14
wa/workloads/benchmarkpi/uiauto/project.properties
Normal file
14
wa/workloads/benchmarkpi/uiauto/project.properties
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# This file is automatically generated by Android Tools.
|
||||||
|
# Do not modify this file -- YOUR CHANGES WILL BE ERASED!
|
||||||
|
#
|
||||||
|
# This file must be checked in Version Control Systems.
|
||||||
|
#
|
||||||
|
# To customize properties used by the Ant build system edit
|
||||||
|
# "ant.properties", and override values to adapt the script to your
|
||||||
|
# project structure.
|
||||||
|
#
|
||||||
|
# To enable ProGuard to shrink and obfuscate your code, uncomment this (available properties: sdk.dir, user.home):
|
||||||
|
#proguard.config=${sdk.dir}/tools/proguard/proguard-android.txt:proguard-project.txt
|
||||||
|
|
||||||
|
# Project target.
|
||||||
|
target=android-17
|
@ -0,0 +1,62 @@
|
|||||||
|
/* Copyright 2013-2015 ARM Limited
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
package com.arm.wa.uiauto.benchmarkpi;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
// Import the uiautomator libraries
|
||||||
|
import com.android.uiautomator.core.UiObject;
|
||||||
|
import com.android.uiautomator.core.UiObjectNotFoundException;
|
||||||
|
import com.android.uiautomator.core.UiScrollable;
|
||||||
|
import com.android.uiautomator.core.UiSelector;
|
||||||
|
import com.android.uiautomator.testrunner.UiAutomatorTestCase;
|
||||||
|
|
||||||
|
import com.arm.wa.uiauto.BaseUiAutomation;
|
||||||
|
|
||||||
|
public class UiAutomation extends BaseUiAutomation {
|
||||||
|
|
||||||
|
public static String TAG = "benchmarkpi";
|
||||||
|
|
||||||
|
public void runWorkload() throws Exception {
|
||||||
|
startTest();
|
||||||
|
waitForResults();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void extractResults() throws Exception {
|
||||||
|
UiSelector selector = new UiSelector();
|
||||||
|
UiObject resultsText = new UiObject(selector.textContains("You calculated Pi in")
|
||||||
|
.className("android.widget.TextView"));
|
||||||
|
Log.v(TAG, resultsText.getText());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void startTest() throws Exception{
|
||||||
|
UiSelector selector = new UiSelector();
|
||||||
|
UiObject benchButton = new UiObject(selector.text("Benchmark my Android!")
|
||||||
|
.className("android.widget.Button"));
|
||||||
|
benchButton.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void waitForResults() throws Exception{
|
||||||
|
UiSelector selector = new UiSelector();
|
||||||
|
UiObject submitButton = new UiObject(selector.text("Submit")
|
||||||
|
.className("android.widget.Button"));
|
||||||
|
submitButton.waitForExists(10 * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -105,12 +105,13 @@ class Dhrystone(Workload):
|
|||||||
self.target.killall('dhrystone')
|
self.target.killall('dhrystone')
|
||||||
raise
|
raise
|
||||||
|
|
||||||
def update_result(self, context):
|
def extract_results(self, context):
|
||||||
outfile = os.path.join(context.output_directory, 'dhrystone.output')
|
outfile = os.path.join(context.output_directory, 'dhrystone.output')
|
||||||
with open(outfile, 'w') as wfh:
|
with open(outfile, 'w') as wfh:
|
||||||
wfh.write(self.output)
|
wfh.write(self.output)
|
||||||
context.add_artifact('dhrystone-output', outfile, 'raw', "dhrystone's stdout")
|
context.add_artifact('dhrystone-output', outfile, 'raw', "dhrystone's stdout")
|
||||||
|
|
||||||
|
def update_output(self, context):
|
||||||
score_count = 0
|
score_count = 0
|
||||||
dmips_count = 0
|
dmips_count = 0
|
||||||
total_score = 0
|
total_score = 0
|
||||||
|
Loading…
x
Reference in New Issue
Block a user