Android 如何判斷CPU是32位還是64位
阿新 • • 發佈:2019-01-07
可以利用三種方式來判斷CPU是32位還是64位:
1. 讀取Android 的system property ("ro.product.cpu.abilist64")來判斷
2. 讀取"/proc/cpuinfo"檔案的第一行來判斷
3. 讀取libc.so檔案的ELF頭部e_indent[]陣列,根據陣列第e_indent[4]的取值來判斷
public static final String CPU_ARCHITECTURE_TYPE_32 = "32"; public static final String CPU_ARCHITECTURE_TYPE_64 = "64"; /** ELF檔案頭 e_indent[]陣列檔案類標識索引 */ private static final int EI_CLASS = 4; /** ELF檔案頭 e_indent[EI_CLASS]的取值:ELFCLASS32表示32位目標 */ private static final int ELFCLASS32 = 1; /** ELF檔案頭 e_indent[EI_CLASS]的取值:ELFCLASS64表示64位目標 */ private static final int ELFCLASS64 = 2; /** The system property key of CPU arch type */ private static final String CPU_ARCHITECTURE_KEY_64 = "ro.product.cpu.abilist64"; /** The system libc.so file path */ private static final String SYSTEM_LIB_C_PATH = "/system/lib/libc.so"; private static final String SYSTEM_LIB_C_PATH_64 = "/system/lib64/libc.so"; private static final String PROC_CPU_INFO_PATH = "/proc/cpuinfo"; private static boolean LOGENABLE = false; /** * Check if the CPU architecture is x86 */ public static boolean checkIfCPUx86() { //1. Check CPU architecture: arm or x86 if (getSystemProperty("ro.product.cpu.abi", "arm").contains("x86")) { //The CPU is x86 return true; } else { return false; } } /** * Get the CPU arch type: x32 or x64 */ public static String getArchType(Context context) { if (getSystemProperty(CPU_ARCHITECTURE_KEY_64, "").length() > 0) { if (LOGENABLE) { Log.d("###############getSystemProperty","CPU arch is 64bit"); } return CPU_ARCHITECTURE_TYPE_64; } else if (isCPUInfo64()) { return CPU_ARCHITECTURE_TYPE_64; } else if (isLibc64()) { return CPU_ARCHITECTURE_TYPE_64; } else { if (LOGENABLE) { Log.d("###############getArchType()","return cpu DEFAULT 32bit!"); } return CPU_ARCHITECTURE_TYPE_32; } } private static String getSystemProperty(String key, String defaultValue) { String value = defaultValue; try { Class<?> clazz= Class.forName("android.os.SystemProperties"); Method get = clazz.getMethod("get", String.class, String.class); value = (String)(get.invoke(clazz, key, "")); } catch (Exception e) { if (LOGENABLE) { Log.d("getSystemProperty", "key = " + key + ", error = " + e.getMessage()); } } if (LOGENABLE) { Log.d("getSystemProperty", key + " = " + value); } return value; } /** * Read the first line of "/proc/cpuinfo" file, and check if it is 64 bit. */ private static boolean isCPUInfo64() { File cpuInfo = new File(PROC_CPU_INFO_PATH); if (cpuInfo != null && cpuInfo.exists()) { InputStream inputStream = null; BufferedReader bufferedReader = null; try { inputStream = new FileInputStream(cpuInfo); bufferedReader = new BufferedReader(new InputStreamReader(inputStream), 512); String line = bufferedReader.readLine(); if (line != null && line.length() > 0 && line.toLowerCase(Locale.US).contains("arch64")) { if (LOGENABLE) { Log.d("###############isCPUInfo64()", PROC_CPU_INFO_PATH + " contains is arch64"); } return true; } else { if (LOGENABLE) { Log.d("###############isCPUInfo64()", PROC_CPU_INFO_PATH + " is not arch64"); } } } catch (Throwable t) { if (LOGENABLE) { Log.d("###############isCPUInfo64()","read " + PROC_CPU_INFO_PATH + " error = " + t.toString()); } } finally { try { if (bufferedReader != null) { bufferedReader.close(); } } catch (Exception e) { e.printStackTrace(); } try { if (inputStream != null) { inputStream.close(); } } catch (Exception e) { e.printStackTrace(); } } } return false; } /** * Check if system libc.so is 32 bit or 64 bit */ private static boolean isLibc64() { File libcFile = new File(SYSTEM_LIB_C_PATH); if (libcFile != null && libcFile.exists()) { byte[] header = readELFHeadrIndentArray(libcFile); if (header != null && header[EI_CLASS] == ELFCLASS64) { if (LOGENABLE) { Log.d("###############isLibc64()", SYSTEM_LIB_C_PATH + " is 64bit"); } return true; } } File libcFile64 = new File(SYSTEM_LIB_C_PATH_64); if (libcFile64 != null && libcFile64.exists()) { byte[] header = readELFHeadrIndentArray(libcFile64); if (header != null && header[EI_CLASS] == ELFCLASS64) { if (LOGENABLE) { Log.d("###############isLibc64()", SYSTEM_LIB_C_PATH_64 + " is 64bit"); } return true; } } return false; } /** * ELF檔案頭格式是固定的:檔案開始是一個16位元組的byte陣列e_indent[16] * e_indent[4]的值可以判斷ELF是32位還是64位 */ private static byte[] readELFHeadrIndentArray(File libFile) { if (libFile != null && libFile.exists()) { FileInputStream inputStream = null; try { inputStream = new FileInputStream(libFile); if (inputStream != null) { byte[] tempBuffer = new byte[16]; int count = inputStream.read(tempBuffer, 0, 16); if (count == 16) { return tempBuffer; } else { if (LOGENABLE) { Log.e("readELFHeadrIndentArray", "Error: e_indent lenght should be 16, but actual is " + count); } } } } catch (Throwable t) { if (LOGENABLE) { Log.e("readELFHeadrIndentArray", "Error:" + t.toString()); } } finally { if (inputStream != null) { try { inputStream.close(); } catch (Exception e) { e.printStackTrace(); } } } } return null; }