blob: 0ed2f27081578e0ab32d5f2c76c296968284ea93 [file] [log] [blame]
// Copyright 2016 The Bazel Authors. 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.testing.coverage;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
/** A branch coverage that must be evaluated as a combination of probes. */
public class BranchExp implements CovExp {
private final List<CovExp> branches;
private boolean hasValue;
private boolean[] probesUsed;
private boolean value;
public BranchExp(List<CovExp> branches) {
this.branches = branches;
hasValue = false;
}
/** Create a new BranchExp using this CovExp as the only branch. */
public BranchExp(CovExp exp) {
branches = new ArrayList<CovExp>();
branches.add(exp);
hasValue = false;
}
/** Gets the expressions for the branches. */
public List<CovExp> getBranches() {
return branches;
}
/**
* Add an expression to a branch expression.
*
* @return the index of the newly added branch.
*/
public int add(CovExp exp) {
branches.add(exp);
return branches.size() - 1;
}
/** Update an existing branch expression. */
public void update(int idx, CovExp exp) {
branches.set(idx, exp);
}
/** Make a new BranchExp representing the concatenation of branches in inputs. */
public static BranchExp concatenate(BranchExp first, BranchExp second) {
List<CovExp> branches = new ArrayList<>(first.branches);
branches.addAll(second.branches);
return new BranchExp(branches);
}
/** Make a new BranchExp representing the pairwise union of branches in inputs */
public static BranchExp zip(BranchExp left, BranchExp right) {
List<CovExp> zippedBranches = new ArrayList<>();
int leftSize = left.branches.size();
int rightSize = right.branches.size();
int i;
for (i = 0; i < leftSize && i < rightSize; i++) {
List<CovExp> branches = Arrays.asList(left.branches.get(i), right.branches.get(i));
zippedBranches.add(new BranchExp(branches));
}
List<CovExp> remainder = leftSize < rightSize ? right.branches : left.branches;
for (; i < remainder.size(); i++) {
zippedBranches.add(new BranchExp(remainder.get(i)));
}
return new BranchExp(zippedBranches);
}
/** Wraps a CovExp in a BranchExp if it isn't one already. */
public static BranchExp ensureIsBranchExp(CovExp exp) {
return exp instanceof BranchExp ? (BranchExp) exp : new BranchExp(exp);
}
@Override
public boolean eval(final boolean[] probes) {
if (hasValue && probes == probesUsed) {
return value;
}
value = false;
for (CovExp exp : branches) {
value = exp.eval(probes);
if (value) {
break;
}
}
hasValue = value; // The value is cached.
probesUsed = probes;
return value;
}
}