blob: 9cb5fb10c080e1a591c207bbb2dc51be92224e15 [file] [log] [blame]
/*
* Copyright (C) 2018 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 android.signature.cts.api;
import java.util.function.Predicate;
import android.signature.cts.DexField;
import android.signature.cts.DexMember;
import android.signature.cts.DexMemberChecker;
import android.signature.cts.DexMethod;
import android.signature.cts.FailureType;
import org.junit.Test;
public abstract class BaseKillswitchTest extends AbstractApiTest {
protected String mErrorMessageAppendix;
@Override
public void setUp() throws Exception {
super.setUp();
DexMemberChecker.init();
}
// We have four methods to split up the load, keeping individual test runs small.
// Tests shared by all the subclasses.
private final static Predicate<DexMember> METHOD_FILTER =
dexMember -> (dexMember instanceof DexMethod);
private final static Predicate<DexMember> FIELD_FILTER =
dexMember -> (dexMember instanceof DexField);
@Test
public void testKillswitchMechanismMethodsThroughReflection() {
doTestKillswitchMechanism(METHOD_FILTER, /* reflection= */ true, /* jni= */ false);
}
@Test
public void testKillswitchMechanismMethodsThroughJni() {
doTestKillswitchMechanism(METHOD_FILTER, /* reflection= */ false, /* jni= */ true);
}
@Test
public void testKillswitchMechanismFieldsThroughReflection() {
doTestKillswitchMechanism(FIELD_FILTER, /* reflection= */ true, /* jni= */ false);
}
@Test(timeout = 900000)
public void testKillswitchMechanismFieldsThroughJni() {
doTestKillswitchMechanism(FIELD_FILTER, /* reflection= */ false, /* jni= */ true);
}
private void doTestKillswitchMechanism(Predicate<DexMember> memberFilter, boolean reflection,
boolean jni) {
runWithTestResultObserver(resultObserver -> {
DexMemberChecker.Observer observer = new DexMemberChecker.Observer() {
@Override
public void classAccessible(boolean accessible, DexMember member) {
if (!accessible) {
resultObserver.notifyFailure(
FailureType.MISSING_CLASS,
member.toString(),
"Class from boot classpath is not accessible"
+ mErrorMessageAppendix);
}
}
@Override
public void fieldAccessibleViaReflection(boolean accessible, DexField field) {
if (!accessible) {
resultObserver.notifyFailure(
FailureType.MISSING_FIELD,
field.toString(),
"Field from boot classpath is not accessible via reflection"
+ mErrorMessageAppendix);
}
}
@Override
public void fieldAccessibleViaJni(boolean accessible, DexField field) {
if (!accessible) {
resultObserver.notifyFailure(
FailureType.MISSING_FIELD,
field.toString(),
"Field from boot classpath is not accessible via JNI"
+ mErrorMessageAppendix);
}
}
@Override
public void methodAccessibleViaReflection(boolean accessible, DexMethod method) {
if (method.isStaticConstructor()) {
// Skip static constructors. They cannot be discovered with reflection.
return;
}
if (!accessible) {
resultObserver.notifyFailure(
FailureType.MISSING_METHOD,
method.toString(),
"Method from boot classpath is not accessible via reflection"
+ mErrorMessageAppendix);
}
}
@Override
public void methodAccessibleViaJni(boolean accessible, DexMethod method) {
if (!accessible) {
resultObserver.notifyFailure(
FailureType.MISSING_METHOD,
method.toString(),
"Method from boot classpath is not accessible via JNI"
+ mErrorMessageAppendix);
}
}
};
mClassProvider.getAllClasses().forEach(klass -> {
mClassProvider.getAllMembers(klass)
.filter(memberFilter)
.forEach(member -> {
DexMemberChecker.checkSingleMember(member, reflection, jni, observer);
});
});
});
}
}