|
|
|
|
|
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:]) |
|
|