| | |
| |
|
| | import os |
| | import sys |
| | import string |
| | import argparse |
| | import subprocess |
| | import tempfile |
| |
|
| |
|
| | root_dir = os.path.abspath(os.path.dirname(__file__)) |
| |
|
| |
|
| | parser = argparse.ArgumentParser(description='Android system files extractor') |
| | parser.add_argument("-p", "--prefix", metavar="NAME", required=True, |
| | help="Prefix for stored files, e.g. galaxy-s7-us") |
| |
|
| |
|
| | |
| | |
| | SHELL_PREFIX = [ |
| | "/sys/class/kgsl/kgsl-3d0/", |
| | ] |
| |
|
| | SYSTEM_FILES = [ |
| | "/proc/cpuinfo", |
| | "/system/build.prop", |
| | "/sys/class/kgsl/kgsl-3d0/bus_split", |
| | "/sys/class/kgsl/kgsl-3d0/clock_mhz", |
| | "/sys/class/kgsl/kgsl-3d0/deep_nap_timer", |
| | "/sys/class/kgsl/kgsl-3d0/default_pwrlevel", |
| | "/sys/class/kgsl/kgsl-3d0/dev", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/available_frequencies", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/available_governors", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/cur_freq", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/governor", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/gpu_load", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/max_freq", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/min_freq", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/polling_interval", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/suspend_time", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/target_freq", |
| | "/sys/class/kgsl/kgsl-3d0/devfreq/trans_stat", |
| | "/sys/class/kgsl/kgsl-3d0/device/op_cpu_table", |
| | "/sys/class/kgsl/kgsl-3d0/freq_table_mhz", |
| | "/sys/class/kgsl/kgsl-3d0/ft_fast_hang_detect", |
| | "/sys/class/kgsl/kgsl-3d0/ft_hang_intr_status", |
| | "/sys/class/kgsl/kgsl-3d0/ft_long_ib_detect", |
| | "/sys/class/kgsl/kgsl-3d0/ft_pagefault_policy", |
| | "/sys/class/kgsl/kgsl-3d0/ft_policy", |
| | "/sys/class/kgsl/kgsl-3d0/gpu_available_frequencies", |
| | "/sys/class/kgsl/kgsl-3d0/gpu_busy_percentage", |
| | "/sys/class/kgsl/kgsl-3d0/gpu_clock_stats", |
| | "/sys/class/kgsl/kgsl-3d0/gpu_llc_slice_enable", |
| | "/sys/class/kgsl/kgsl-3d0/gpu_model", |
| | "/sys/class/kgsl/kgsl-3d0/gpubusy", |
| | "/sys/class/kgsl/kgsl-3d0/gpuclk", |
| | "/sys/class/kgsl/kgsl-3d0/gpuhtw_llc_slice_enable", |
| | "/sys/class/kgsl/kgsl-3d0/hwcg", |
| | "/sys/class/kgsl/kgsl-3d0/idle_timer", |
| | "/sys/class/kgsl/kgsl-3d0/lm", |
| | "/sys/class/kgsl/kgsl-3d0/max_gpuclk", |
| | "/sys/class/kgsl/kgsl-3d0/max_pwrlevel", |
| | "/sys/class/kgsl/kgsl-3d0/min_clock_mhz", |
| | "/sys/class/kgsl/kgsl-3d0/min_pwrlevel", |
| | "/sys/class/kgsl/kgsl-3d0/num_pwrlevels", |
| | "/sys/class/kgsl/kgsl-3d0/pmqos_active_latency", |
| | "/sys/class/kgsl/kgsl-3d0/popp", |
| | "/sys/class/kgsl/kgsl-3d0/preempt_count", |
| | "/sys/class/kgsl/kgsl-3d0/preempt_level", |
| | "/sys/class/kgsl/kgsl-3d0/preemption", |
| | "/sys/class/kgsl/kgsl-3d0/pwrscale", |
| | "/sys/class/kgsl/kgsl-3d0/reset_count", |
| | "/sys/class/kgsl/kgsl-3d0/skipsaverestore", |
| | "/sys/class/kgsl/kgsl-3d0/sptp_pc", |
| | "/sys/class/kgsl/kgsl-3d0/thermal_pwrlevel", |
| | "/sys/class/kgsl/kgsl-3d0/throttling", |
| | "/sys/class/kgsl/kgsl-3d0/usesgmem", |
| | "/sys/class/kgsl/kgsl-3d0/wake_nice", |
| | "/sys/class/kgsl/kgsl-3d0/wake_timeout", |
| | "/sys/devices/soc0/accessory_chip", |
| | "/sys/devices/soc0/build_id", |
| | "/sys/devices/soc0/chip_family", |
| | "/sys/devices/soc0/chip_name", |
| | "/sys/devices/soc0/family", |
| | "/sys/devices/soc0/foundry_id", |
| | "/sys/devices/soc0/hw_platform", |
| | "/sys/devices/soc0/image_crm_version", |
| | "/sys/devices/soc0/image_variant", |
| | "/sys/devices/soc0/image_version", |
| | "/sys/devices/soc0/images", |
| | "/sys/devices/soc0/machine", |
| | "/sys/devices/soc0/ncluster_array_offset", |
| | "/sys/devices/soc0/ndefective_parts_array_offset", |
| | "/sys/devices/soc0/nmodem_supported", |
| | "/sys/devices/soc0/nproduct_id", |
| | "/sys/devices/soc0/num_clusters", |
| | "/sys/devices/soc0/num_defective_parts", |
| | "/sys/devices/soc0/platform_subtype", |
| | "/sys/devices/soc0/platform_subtype_id", |
| | "/sys/devices/soc0/platform_version", |
| | "/sys/devices/soc0/pmic_die_revision", |
| | "/sys/devices/soc0/pmic_model", |
| | "/sys/devices/soc0/raw_device_family", |
| | "/sys/devices/soc0/raw_device_number", |
| | "/sys/devices/soc0/raw_id", |
| | "/sys/devices/soc0/raw_version", |
| | "/sys/devices/soc0/revision", |
| | "/sys/devices/soc0/select_image", |
| | "/sys/devices/soc0/serial_number", |
| | "/sys/devices/soc0/soc_id", |
| | "/sys/devices/soc0/vendor", |
| | "/sys/devices/system/b.L/big_threads", |
| | "/sys/devices/system/b.L/boot_cluster", |
| | "/sys/devices/system/b.L/core_status", |
| | "/sys/devices/system/b.L/little_threads", |
| | "/sys/devices/system/b.L/down_migrations", |
| | "/sys/devices/system/b.L/up_migrations", |
| | "/sys/devices/system/cpu/isolated", |
| | "/sys/devices/system/cpu/kernel_max", |
| | "/sys/devices/system/cpu/modalias", |
| | "/sys/devices/system/cpu/offline", |
| | "/sys/devices/system/cpu/online", |
| | "/sys/devices/system/cpu/possible", |
| | "/sys/devices/system/cpu/present", |
| | "/sys/devices/system/cpu/sched_isolated", |
| | "/sys/devices/system/cpu/clusterhotplug/cur_hstate", |
| | "/sys/devices/system/cpu/clusterhotplug/down_freq", |
| | "/sys/devices/system/cpu/clusterhotplug/down_tasks", |
| | "/sys/devices/system/cpu/clusterhotplug/down_threshold", |
| | "/sys/devices/system/cpu/clusterhotplug/sampling_rate", |
| | "/sys/devices/system/cpu/clusterhotplug/time_in_state", |
| | "/sys/devices/system/cpu/clusterhotplug/up_freq", |
| | "/sys/devices/system/cpu/clusterhotplug/up_tasks", |
| | "/sys/devices/system/cpu/clusterhotplug/up_threshold", |
| | "/sys/devices/system/cpu/cpufreq/all_time_in_state", |
| | "/sys/devices/system/cpu/cpufreq/current_in_state", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/big_cpu_num", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/big_max_freq", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/big_min_freq", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/hmp_boost_type", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/hmp_prev_boost_type", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/ltl_cpu_num", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/ltl_divider", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/ltl_max_freq", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/ltl_min_freq", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/ltl_min_lock", |
| | "/sys/devices/system/cpu/cpufreq/cpufreq_limit/requests", |
| | "/sys/devices/system/cpu/cpuidle/current_driver", |
| | "/sys/devices/system/cpu/cpuidle/current_governor_ro", |
| | "/sys/devices/system/cpu/cputopo/cpus_per_cluster", |
| | "/sys/devices/system/cpu/cputopo/big_cpumask", |
| | "/sys/devices/system/cpu/cputopo/glbinfo", |
| | "/sys/devices/system/cpu/cputopo/is_big_little", |
| | "/sys/devices/system/cpu/cputopo/is_multi_cluster", |
| | "/sys/devices/system/cpu/cputopo/little_cpumask", |
| | "/sys/devices/system/cpu/cputopo/nr_clusters", |
| | "/sys/devices/system/sched/idle_prefer", |
| | "/sys/devices/system/sched/sched_boost", |
| | ] |
| |
|
| | CPU_FILES = [ |
| | "core_ctl/active_cpus", |
| | "core_ctl/busy_up_thres", |
| | "core_ctl/busy_down_thres", |
| | "core_ctl/enable", |
| | "core_ctl/global_state", |
| | "core_ctl/is_big_cluster", |
| | "core_ctl/max_cpus", |
| | "core_ctl/min_cpus", |
| | "core_ctl/need_cpus", |
| | "core_ctl/not_preferred", |
| | "core_ctl/offline_delay_ms", |
| | "core_ctl/task_thres", |
| | "current_driver", |
| | "current_governor_ro", |
| | "cpuidle/driver/name", |
| | "cpufreq/affected_cpus", |
| | "cpufreq/cpuinfo_max_freq", |
| | "cpufreq/cpuinfo_min_freq", |
| | "cpufreq/cpuinfo_transition_latency", |
| | "cpufreq/related_cpus", |
| | "cpufreq/scaling_available_frequencies", |
| | "cpufreq/scaling_available_governors", |
| | "cpufreq/scaling_cur_freq", |
| | "cpufreq/scaling_driver", |
| | "cpufreq/scaling_governor", |
| | "cpufreq/scaling_max_freq", |
| | "cpufreq/scaling_min_freq", |
| | "cpufreq/sched/down_throttle_nsec", |
| | "cpufreq/sched/up_throttle_nsec", |
| | "cpufreq/stats/time_in_state", |
| | "cpufreq/stats/total_trans", |
| | "cpufreq/stats/trans_table", |
| | "isolate", |
| | "regs/identification/midr_el1", |
| | "regs/identification/revidr_el1", |
| | "sched_load_boost", |
| | "topology/core_id", |
| | "topology/core_siblings", |
| | "topology/core_siblings_list", |
| | "topology/cpu_capacity", |
| | "topology/max_cpu_capacity", |
| | "topology/physical_package_id", |
| | "topology/thread_siblings", |
| | "topology/thread_siblings_list", |
| | ] |
| |
|
| | CACHE_FILES = [ |
| | "allocation_policy", |
| | "coherency_line_size", |
| | "level", |
| | "number_of_sets", |
| | "shared_cpu_list", |
| | "shared_cpu_map", |
| | "size", |
| | "type", |
| | "ways_of_associativity", |
| | "write_policy", |
| | ] |
| |
|
| | def c_escape(string): |
| | c_string = "" |
| | for c in string: |
| | if c == "\\": |
| | c_string += "\\\\" |
| | elif c == "\"": |
| | c_string += "\\\"" |
| | elif c == "\t": |
| | c_string += "\\t" |
| | elif c == "\n": |
| | c_string += "\\n" |
| | elif c == "\r": |
| | c_string += "\\r" |
| | elif ord(c) == 0: |
| | c_string += "\\0" |
| | elif 32 <= ord(c) < 127: |
| | c_string += c |
| | else: |
| | c_string += "x%02X" % ord(c) |
| | return c_string |
| |
|
| | def adb_shell(commands): |
| | env = os.environ.copy() |
| | env["LC_ALL"] = "C" |
| |
|
| | adb = subprocess.Popen(["adb", "shell"] + commands, env=env, stdout=subprocess.PIPE) |
| | stdout, _ = adb.communicate() |
| | if adb.returncode == 0: |
| | return stdout |
| |
|
| | def adb_push(local_path, device_path): |
| | env = os.environ.copy() |
| | env["LC_ALL"] = "C" |
| |
|
| | adb = subprocess.Popen(["adb", "push", local_path, device_path], env=env) |
| | adb.communicate() |
| | return adb.returncode == 0 |
| |
|
| | def adb_pull(device_path, local_path): |
| | if any(device_path.startswith(prefix) for prefix in SHELL_PREFIX): |
| | content = adb_shell(["cat", device_path]) |
| | if content is not None: |
| | if not content.rstrip().endswith("No such file or directory"): |
| | with open(local_path, "wb") as local_file: |
| | local_file.write(content) |
| | return True |
| | else: |
| | env = os.environ.copy() |
| | env["LC_ALL"] = "C" |
| |
|
| | adb = subprocess.Popen(["adb", "pull", device_path, local_path], env=env) |
| | adb.communicate() |
| | return adb.returncode == 0 |
| |
|
| | def adb_getprop(): |
| | properties = adb_shell(["getprop"]) |
| | properties_list = list() |
| | while properties: |
| | assert properties.startswith("[") |
| | properties = properties[1:] |
| | key, properties = properties.split("]", 1) |
| | properties = properties.strip() |
| | assert properties.startswith(":") |
| | properties = properties[1:].strip() |
| | assert properties.startswith("[") |
| | properties = properties[1:] |
| | value, properties = properties.split("]", 1) |
| | properties = properties.strip() |
| | properties_list.append((key, value)) |
| | return properties_list |
| |
|
| | def add_mock_file(stream, path, content): |
| | assert content is not None |
| | stream.write("\t{\n") |
| | stream.write("\t\t.path = \"%s\",\n" % path) |
| | stream.write("\t\t.size = %d,\n" % len(content)) |
| | if len(content.splitlines()) > 1: |
| | stream.write("\t\t.content =") |
| | for line in content.splitlines(True): |
| | stream.write("\n\t\t\t\"%s\"" % c_escape(line)) |
| | stream.write(",\n") |
| | else: |
| | stream.write("\t\t.content = \"%s\",\n" % c_escape(content)) |
| | stream.write("\t},\n") |
| |
|
| |
|
| | def dump_device_file(stream, path, prefix_line=None): |
| | temp_fd, temp_path = tempfile.mkstemp() |
| | os.close(temp_fd) |
| | try: |
| | if adb_pull(path, temp_path): |
| | with open(temp_path, "rb") as temp_file: |
| | content = temp_file.read() |
| | if prefix_line is not None: |
| | stream.write(prefix_line) |
| | add_mock_file(stream, path, content) |
| | return content |
| | finally: |
| | if os.path.exists(temp_path): |
| | os.remove(temp_path) |
| |
|
| |
|
| | def main(args): |
| | options = parser.parse_args(args) |
| |
|
| | dmesg_content = adb_shell(["dmesg"]) |
| | if dmesg_content is not None and dmesg_content.strip() == "klogctl: Operation not permitted": |
| | dmesg_content = None |
| | if dmesg_content is not None: |
| | with open(os.path.join("test", "dmesg", options.prefix + ".log"), "w") as dmesg_dump: |
| | dmesg_dump.write(dmesg_content) |
| |
|
| | build_prop_content = None |
| | proc_cpuinfo_content = None |
| | proc_cpuinfo_content32 = None |
| | kernel_max = 0 |
| | with open(os.path.join("test", "mock", options.prefix + ".h"), "w") as file_header: |
| | file_header.write("struct cpuinfo_mock_file filesystem[] = {\n") |
| | android_props = adb_getprop() |
| | abi = None |
| | for key, value in android_props: |
| | if key == "ro.product.cpu.abi": |
| | abi = value |
| | for path in SYSTEM_FILES: |
| | arm64_prefix = None |
| | if path == "/proc/cpuinfo" and abi == "arm64-v8a": |
| | arm64_prefix = "#if CPUINFO_ARCH_ARM64\n" |
| | content = dump_device_file(file_header, path, prefix_line=arm64_prefix) |
| | if content is not None: |
| | if path == "/proc/cpuinfo": |
| | proc_cpuinfo_content = content |
| | elif path == "/system/build.prop": |
| | build_prop_content = content |
| | elif path == "/sys/devices/system/cpu/kernel_max": |
| | kernel_max = int(content.strip()) |
| | if arm64_prefix: |
| | cpuinfo_dump_binary = os.path.join(root_dir, "..", "build", "android", "armeabi-v7a", "cpuinfo-dump") |
| | assert os.path.isfile(cpuinfo_dump_binary) |
| | adb_push(cpuinfo_dump_binary, "/data/local/tmp/cpuinfo-dump") |
| | proc_cpuinfo_content32 = adb_shell(["/data/local/tmp/cpuinfo-dump"]) |
| | if proc_cpuinfo_content32: |
| | proc_cpuinfo_content32 = "\n".join(proc_cpuinfo_content32.splitlines()) |
| | file_header.write("#elif CPUINFO_ARCH_ARM\n") |
| | add_mock_file(file_header, "/proc/cpuinfo", proc_cpuinfo_content32) |
| | file_header.write("#endif\n") |
| |
|
| | for cpu in range(kernel_max + 1): |
| | for filename in CPU_FILES: |
| | path = "/sys/devices/system/cpu/cpu%d/%s" % (cpu, filename) |
| | dump_device_file(file_header, path) |
| | for index in range(5): |
| | for filename in CACHE_FILES: |
| | path = "/sys/devices/system/cpu/cpu%d/cache/index%d/%s" % (cpu, index, filename) |
| | dump_device_file(file_header, path) |
| | file_header.write("\t{ NULL },\n") |
| | file_header.write("};\n") |
| | file_header.write("#ifdef __ANDROID__\n") |
| | file_header.write("struct cpuinfo_mock_property properties[] = {\n") |
| | for key, value in android_props: |
| | file_header.write("\t{\n") |
| | file_header.write("\t\t.key = \"%s\",\n" % c_escape(key)) |
| | file_header.write("\t\t.value = \"%s\",\n" % c_escape(value)) |
| | file_header.write("\t},\n") |
| | file_header.write("\t{ NULL },\n") |
| | file_header.write("};\n") |
| | file_header.write("#endif /* __ANDROID__ */\n") |
| |
|
| | if proc_cpuinfo_content is not None: |
| | with open(os.path.join("test", "cpuinfo", options.prefix + ".log"), "w") as proc_cpuinfo_dump: |
| | proc_cpuinfo_dump.write(proc_cpuinfo_content) |
| | if proc_cpuinfo_content32 is not None: |
| | with open(os.path.join("test", "cpuinfo", options.prefix + ".armeabi.log"), "w") as proc_cpuinfo_dump32: |
| | proc_cpuinfo_dump32.write(proc_cpuinfo_content32) |
| | if build_prop_content is not None: |
| | with open(os.path.join("test", "build.prop", options.prefix + ".log"), "w") as build_prop_dump: |
| | build_prop_dump.write(build_prop_content) |
| |
|
| | if __name__ == "__main__": |
| | main(sys.argv[1:]) |
| |
|