blob: 8ab610f0662fb2ab9c8c547f698b44ce578c355d [file] [log] [blame]
/*
* Copyright 2016 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.
*/
package libcore.java.security;
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import dalvik.system.VMRuntime;
class CpuFeatures {
private CpuFeatures() {
}
static boolean isAESHardwareAccelerated() {
// Expectations based on CPU type: If these aren't met then Conscrypt
// integration tests will fail and the cause should be investigated.
String instructionSet = VMRuntime.getCurrentInstructionSet();
if (instructionSet.startsWith("arm")) {
// All ARM CPUs with the "aes" feature should have hardware AES.
List<String> features = getListFromCpuinfo("Features");
if (features != null && features.contains("aes")) {
return true;
}
} else if (instructionSet.startsWith("x86")) {
// x86 CPUs with the "aes" flag and running in 64bit mode should have hardware AES.
if ("x86_64".equals(instructionSet)) {
List<String> flags = getListFromCpuinfo("flags");
if (flags != null && flags.contains("aes")) {
return true;
}
} else {
// Hardware AES not supported in 32bit mode.
return false;
}
}
// Otherwise trust Conscrypt NativeCrypto's own checks, for example if we're in an
// emulated ABI, it might bridge to a library that has accelerated AES instructions.
try {
Class<?> nativeCrypto = Class.forName("com.android.org.conscrypt.NativeCrypto");
Method EVP_has_aes_hardware = nativeCrypto.getDeclaredMethod("EVP_has_aes_hardware");
EVP_has_aes_hardware.setAccessible(true);
return ((Integer) EVP_has_aes_hardware.invoke(null)) == 1;
} catch (ClassNotFoundException | NoSuchMethodException | SecurityException
| IllegalAccessException | IllegalArgumentException ignored) {
} catch (InvocationTargetException e) {
throw new IllegalArgumentException(e);
}
return false;
}
private static String getFieldFromCpuinfo(String field) {
try {
BufferedReader br = new BufferedReader(new FileReader("/proc/cpuinfo"));
Pattern p = Pattern.compile(field + "\\s*:\\s*(.*)");
try {
String line;
while ((line = br.readLine()) != null) {
Matcher m = p.matcher(line);
if (m.matches()) {
return m.group(1);
}
}
} finally {
br.close();
}
} catch (IOException ignored) {
}
return null;
}
private static List<String> getListFromCpuinfo(String fieldName) {
String features = getFieldFromCpuinfo(fieldName);
if (features == null)
return null;
return Arrays.asList(features.split("\\s"));
}
}