/*
Copyright 2014 Google Inc. All Rights Reserved.
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 com.google.security.zynamics.binnavi.disassembly.types;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import com.google.security.zynamics.binnavi.Database.Exceptions.CouldntLoadDataException;
import com.google.security.zynamics.binnavi.yfileswrap.disassembly.types.TypeDependenceGraph;
import com.google.security.zynamics.binnavi.yfileswrap.disassembly.types.TypeDependenceGraph.DependenceResult;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import java.util.Set;
/**
* Test suite for the type dependence graph class.
*/
@RunWith(JUnit4.class)
public class TypeDependenceGraphTests {
private TestTypeSystem typeSystem;
private TypeDependenceGraph dependenceGraph;
@Before
public void initializeTypeSystem() throws CouldntLoadDataException {
typeSystem = new TestTypeSystem(new TypeManager(new TypeManagerMockBackend()));
dependenceGraph = new TypeDependenceGraph(typeSystem.getTypes(), typeSystem.getTypeMembers());
}
@Test
public void testAddMember() {
final ImmutableSet<BaseType> affectedTypes =
dependenceGraph.addMember(typeSystem.simpleStruct, typeSystem.intType).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(typeSystem.simpleStruct, typeSystem.nestedStruct,
typeSystem.doubleNestedStruct, typeSystem.complexUnion), affectedTypes);
}
@Test
public void testAddMemberNoDependency() {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.addMember(
typeSystem.doubleNestedStruct, typeSystem.intType).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(typeSystem.doubleNestedStruct, typeSystem.complexUnion),
affectedTypes);
}
@Test(expected = NullPointerException.class)
public void testAddNullMember() {
final TypeDependenceGraph dependenceGraph = new TypeDependenceGraph(
ImmutableList.<BaseType>builder().build(), ImmutableList.<TypeMember>builder().build());
dependenceGraph.addMember(null, null);
}
@Test(expected = NullPointerException.class)
public void testAddNullType() {
final TypeDependenceGraph dependenceGraph = new TypeDependenceGraph(
ImmutableList.<BaseType>builder().build(), ImmutableList.<TypeMember>builder().build());
dependenceGraph.addType(null);
}
@Test
public void testAddType() {
final BaseType newCompoundType =
new BaseType(100, "new_compound_type", 0, false, BaseTypeCategory.STRUCT);
dependenceGraph.addType(newCompoundType);
final ImmutableSet<BaseType> affectedTypes =
dependenceGraph.addMember(newCompoundType, typeSystem.simpleStruct).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(newCompoundType), affectedTypes);
final ImmutableSet<BaseType> newAffectedTypes =
dependenceGraph.addMember(typeSystem.simpleStruct, typeSystem.intType).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(newCompoundType, typeSystem.simpleStruct,
typeSystem.nestedStruct, typeSystem.doubleNestedStruct, typeSystem.complexUnion),
newAffectedTypes);
}
@Test
public void testDeleteFooType() {
final ImmutableSet<BaseType> affectedTypes =
dependenceGraph.deleteType(typeSystem.doubleNestedStruct);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.doubleNestedStruct, typeSystem.complexUnion),
affectedTypes);
}
@Test
public void testDeleteIntType() {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.deleteType(typeSystem.intType);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.intType,
typeSystem.simpleStruct,
typeSystem.nestedStruct,
typeSystem.doubleNestedStruct,
typeSystem.simpleUnion,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testDeleteMember() {
final TypeMember member = typeSystem.simpleStruct.iterator().next();
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.deleteMember(member);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.simpleStruct, typeSystem.nestedStruct,
typeSystem.doubleNestedStruct, typeSystem.complexUnion), affectedTypes);
}
@Test
public void testDeleteMemberTwoDependencies() {
final TypeMember member = typeSystem.nestedStruct.iterator().next();
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.deleteMember(member);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.nestedStruct, typeSystem.doubleNestedStruct,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testDeleteSimpleStruct() {
final ImmutableSet<BaseType> affectedTypes =
dependenceGraph.deleteType(typeSystem.simpleStruct);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.simpleStruct, typeSystem.nestedStruct,
typeSystem.doubleNestedStruct, typeSystem.complexUnion), affectedTypes);
}
@Test(expected = NullPointerException.class)
public void testDeleteNullType() {
dependenceGraph.deleteType(null);
}
@Test
public void testDeleteUintType() {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.deleteType(typeSystem.uintType);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.uintType,
typeSystem.uintArrayType,
typeSystem.simpleStruct,
typeSystem.nestedStruct,
typeSystem.doubleNestedStruct,
typeSystem.simpleUnion,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testIsContainedIn() {
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.simpleStruct, typeSystem.intType));
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.simpleStruct, typeSystem.uintType));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.simpleStruct, typeSystem.uintArrayType));
}
@Test
public void testIsContainedInDoubleNested() {
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct, typeSystem.intType));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct, typeSystem.uintType));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct, typeSystem.uintArrayType));
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct,
typeSystem.uintPointerType));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct, typeSystem.simpleStruct));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.doubleNestedStruct, typeSystem.nestedStruct));
}
@Test
public void testIsContainedInNested() {
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.nestedStruct, typeSystem.intType));
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.nestedStruct, typeSystem.uintType));
assertTrue(
dependenceGraph.isTypeContainedIn(typeSystem.nestedStruct, typeSystem.uintArrayType));
assertTrue(dependenceGraph.isTypeContainedIn(typeSystem.nestedStruct, typeSystem.simpleStruct));
}
@Test(expected = NullPointerException.class)
public void testNullInstantiation() {
new TypeDependenceGraph(null, null);
}
@Test
public void testRecursiveType() {
final DependenceResult result =
dependenceGraph.addMember(typeSystem.simpleStruct, typeSystem.doubleNestedStruct);
assertFalse(result.isValid());
}
@Test
public void testRecursiveTypeWithSelf() {
final DependenceResult result =
dependenceGraph.addMember(typeSystem.simpleStruct, typeSystem.simpleStruct);
assertFalse(result.isValid());
}
@Test
public void testUpdateDoubleNestedStructMembers() {
final BaseType newMemberType = typeSystem.intType;
final BaseType parentType = typeSystem.doubleNestedStruct;
for (final TypeMember member : parentType) {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.updateMember(parentType,
member.getBaseType(), newMemberType).getAffectedTypes();
assertEquals(
ImmutableSet.<BaseType>of(typeSystem.doubleNestedStruct, typeSystem.complexUnion),
affectedTypes);
}
}
@Test
public void testUpdateIntType() {
final Set<BaseType> affectedTypes = dependenceGraph.updateType(typeSystem.intType);
assertEquals(Sets.newHashSet(typeSystem.intType,
typeSystem.simpleStruct,
typeSystem.nestedStruct,
typeSystem.doubleNestedStruct,
typeSystem.simpleUnion,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testUpdateSimpleStructMembers() {
for (final TypeMember member : typeSystem.simpleStruct) {
final Set<BaseType> affectedTypes = dependenceGraph.updateMember(member.getParentType(),
member.getBaseType(), typeSystem.intType).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(typeSystem.simpleStruct, typeSystem.nestedStruct,
typeSystem.doubleNestedStruct, typeSystem.complexUnion), affectedTypes);
}
}
@Test(expected = NullPointerException.class)
public void testUpdateNullMember() {
dependenceGraph.updateMember(null, null, null);
}
@Test(expected = NullPointerException.class)
public void testUpdateNullType() {
dependenceGraph.updateType(null);
}
@Test
public void testUpdateTypeTransitive() {
// "NestedStruct" is only transitively reachable from "unsigned int" so this test is
// conceptually different from testUpdateType().
final Set<BaseType> affectedTypes = dependenceGraph.updateType(typeSystem.uintType);
assertEquals(Sets.newHashSet(typeSystem.uintType,
typeSystem.uintArrayType,
typeSystem.simpleStruct,
typeSystem.nestedStruct,
typeSystem.doubleNestedStruct,
typeSystem.simpleUnion,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testUpdateUintType() {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.updateType(typeSystem.uintType);
assertEquals(ImmutableSet.<BaseType>of(typeSystem.uintType,
typeSystem.uintArrayType,
typeSystem.simpleStruct,
typeSystem.nestedStruct,
typeSystem.doubleNestedStruct,
typeSystem.simpleUnion,
typeSystem.complexUnion), affectedTypes);
}
@Test
public void testUpdateNestedStructMembers() {
for (final TypeMember member : typeSystem.nestedStruct) {
final ImmutableSet<BaseType> affectedTypes = dependenceGraph.updateMember(
member.getParentType(), member.getBaseType(), typeSystem.intType).getAffectedTypes();
assertEquals(ImmutableSet.<BaseType>of(typeSystem.nestedStruct, typeSystem.doubleNestedStruct,
typeSystem.complexUnion), affectedTypes);
}
}
}