# # Copyright (C) 2017 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # import os from vts.utils.python.os import path_utils from vts.testcases.fuzz.template.libfuzzer_test import libfuzzer_test_config as config class LibFuzzerTestCase(object): """Represents libfuzzer test case. Attributes: _bin_host_path: string, path to binary on host. _bin_name: string, name of the binary. _test_name: string, name of the test case. _libfuzzer_params: dict, libfuzzer-specific parameters. _additional_params: dict, additional parameters. """ def __init__(self, bin_host_path, libfuzzer_params, additional_params): self._bin_host_path = bin_host_path self._libfuzzer_params = libfuzzer_params self._additional_params = additional_params self._binary_name = os.path.basename(bin_host_path) self._test_name = self._binary_name def _GetCorpusDir(self): """Returns corpus directory name on target.""" corpus_dir = path_utils.JoinTargetPath(config.FUZZER_TEST_DIR, '%s_corpus' % self._test_name) return corpus_dir def GetCorpusOutDir(self): """Returns corpus output directory name on target.""" return self._GetCorpusDir() + '_out' def GetCorpusSeedDir(self): """Returns corpus seed directory name on target.""" return self._GetCorpusDir() + '_seed' def GetCorpusTriggerDir(self): """Returns basename of corpus trigger directory.""" return '%s_corpus_trigger' % self._test_name def CreateFuzzerFlags(self): """Creates flags for the fuzzer executable. Returns: string, of form '-<flag0>=<val0> -<flag1>=<val1> ... ' """ # Used to separate additional and libfuzzer flags. DELIMITER = '--' additional_flags = ' '.join( ['--%s=%s' % (k, v) for k, v in self._additional_params.items()]) libfuzzer_flags = ' '.join( ['-%s=%s' % (k, v) for k, v in self._libfuzzer_params.items()]) if not additional_flags: flags = libfuzzer_flags else: flags = '%s %s %s' % (additional_flags, DELIMITER, libfuzzer_flags) return flags def GetRunCommand(self, debug_mode=False): """Returns target shell command to run the fuzzer binary.""" test_flags = self.CreateFuzzerFlags() corpus_out = '' if debug_mode else self.GetCorpusOutDir() corpus_seed = '' if debug_mode else self.GetCorpusSeedDir() cd_cmd = 'cd %s' % config.FUZZER_TEST_DIR chmod_cmd = 'chmod 777 %s' % self._binary_name ld_path = 'LD_LIBRARY_PATH=/data/local/tmp/64:/data/local/tmp/32:$LD_LIBRARY_PATH' test_cmd = '%s ./%s %s %s %s' % (ld_path, self._binary_name, corpus_out, corpus_seed, test_flags) if not debug_mode: test_cmd += ' > /dev/null' return ' && '.join([cd_cmd, chmod_cmd, test_cmd]) @property def test_name(self): """Name of this test case.""" return str(self._test_name) @test_name.setter def test_name(self, name): """Set name of this test case.""" self._test_name = name @property def bin_host_path(self): """Host path to binary for this test case.""" return self._bin_host_path