Skip to content

Commit

Permalink
Instead of ATOMIC_BEGIN,_END, we use atomic flags for variables
Browse files Browse the repository at this point in the history
  • Loading branch information
leventeBajczi committed Nov 11, 2024
1 parent 9af341c commit d85d701
Show file tree
Hide file tree
Showing 8 changed files with 314 additions and 211 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -743,10 +743,6 @@ public Expr<?> visitGccPrettyFunc(CParser.GccPrettyFuncContext ctx) {
@Override
public Expr<?> visitPrimaryExpressionId(CParser.PrimaryExpressionIdContext ctx) {
final var variable = getVar(ctx.Identifier().getText());
if (atomicVars.contains(variable)) {
preStatements.add(new CCall("__VERIFIER_atomic_begin", List.of(), parseContext));
postStatements.add(new CCall("__VERIFIER_atomic_end", List.of(), parseContext));
}
return variable.getRef();
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ public class FunctionVisitor extends CBaseVisitor<CStatement> {

public void clear() {
variables.clear();
atomicVariables.clear();
flatVariables.clear();
functions.clear();
}
Expand Down Expand Up @@ -138,6 +139,7 @@ public FunctionVisitor(final ParseContext parseContext, Logger uniqueWarningLogg
@Override
public CStatement visitCompilationUnit(CParser.CompilationUnitContext ctx) {
variables.clear();
atomicVariables.clear();
variables.push(Tuple2.of("", new LinkedHashMap<>()));
flatVariables.clear();
functions.clear();
Expand Down Expand Up @@ -269,14 +271,24 @@ public CStatement visitFunctionDefinition(CParser.FunctionDefinitionContext ctx)
CStatement accept = blockItemListContext.accept(this);
variables.pop();
CFunction cFunction =
new CFunction(funcDecl, accept, new ArrayList<>(flatVariables), parseContext);
new CFunction(
funcDecl,
accept,
new ArrayList<>(flatVariables),
parseContext,
atomicVariables);
recordMetadata(ctx, cFunction);
return cFunction;
}
variables.pop();
CCompound cCompound = new CCompound(parseContext);
CFunction cFunction =
new CFunction(funcDecl, cCompound, new ArrayList<>(flatVariables), parseContext);
new CFunction(
funcDecl,
cCompound,
new ArrayList<>(flatVariables),
parseContext,
atomicVariables);
recordMetadata(ctx, cCompound);
recordMetadata(ctx, cFunction);
return cFunction;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,27 +13,35 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package hu.bme.mit.theta.frontend.transformation.model.statements;

import hu.bme.mit.theta.core.decl.VarDecl;
import hu.bme.mit.theta.frontend.ParseContext;
import hu.bme.mit.theta.frontend.transformation.model.declaration.CDeclaration;

import java.util.List;
import java.util.Set;

public class CFunction extends CStatement {

private final CDeclaration funcDecl;
private final CStatement compound;
private final List<VarDecl<?>> flatVariables;
private final Set<VarDecl<?>> atomicVariables;

public CFunction(CDeclaration funcDecl, CStatement compound, List<VarDecl<?>> flatVariables, ParseContext parseContext) {
public CFunction(
CDeclaration funcDecl,
CStatement compound,
List<VarDecl<?>> flatVariables,
ParseContext parseContext,
Set<VarDecl<?>> atomicVariables) {
super(parseContext);
this.funcDecl = funcDecl;
this.compound = compound;
this.flatVariables = flatVariables;
parseContext.getMetadata().lookupMetadata(funcDecl)
this.atomicVariables = atomicVariables;
parseContext
.getMetadata()
.lookupMetadata(funcDecl)
.forEach((s, o) -> parseContext.getMetadata().create(this, s, o));
}

Expand All @@ -53,4 +61,8 @@ public List<VarDecl<?>> getFlatVariables() {
public <P, R> R accept(CStatementVisitor<P, R> visitor, P param) {
return visitor.visit(this, param);
}

public Set<VarDecl<?>> getAtomicVariables() {
return atomicVariables;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,13 @@ class FrontendXcfaBuilder(
"Not handling init expression of struct array ${globalDeclaration.get1()}",
)
}
builder.addVar(XcfaGlobalVar(globalDeclaration.get2(), type.nullValue))
builder.addVar(
XcfaGlobalVar(
globalDeclaration.get2(),
type.nullValue,
atomic = globalDeclaration.get1().type.isAtomic,
)
)
if (type is CArray) {
initStmtList.add(
StmtLabel(
Expand Down Expand Up @@ -203,6 +209,7 @@ class FrontendXcfaBuilder(
): XcfaProcedureBuilder {
locationLut.clear()
val flatVariables = function.flatVariables
val isAtomic = function.atomicVariables::contains
val funcDecl = function.funcDecl
val compound = function.compound
val builder =
Expand Down Expand Up @@ -238,6 +245,9 @@ class FrontendXcfaBuilder(

for (flatVariable in flatVariables) {
builder.addVar(flatVariable)
if (isAtomic(flatVariable)) {
builder.setAtomic(flatVariable)
}
val type = CComplexType.getType(flatVariable.ref, parseContext)
if (
(type is CArray || type is CStruct) && builder.getParams().none { it.first == flatVariable }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -228,9 +228,17 @@ fun getXcfaErrorPredicate(
val mutexes2 = s.mutexes.filterValues { it == process2.key }.keys
val globalVars1 = edge1.getGlobalVarsWithNeededMutexes(xcfa, mutexes1)
val globalVars2 = edge2.getGlobalVarsWithNeededMutexes(xcfa, mutexes2)
for (v1 in globalVars1) for (v2 in globalVars2) if (v1.varDecl == v2.varDecl)
if (v1.access.isWritten || v2.access.isWritten)
if ((v1.mutexes intersect v2.mutexes).isEmpty()) return@Predicate true
for (v1 in globalVars1) {
for (v2 in globalVars2) {
if (
v1.globalVar == v2.globalVar &&
!v1.globalVar.atomic &&
(v1.access.isWritten || v2.access.isWritten) &&
(v1.mutexes intersect v2.mutexes).isEmpty()
)
return@Predicate true
}
}
}
false
}
Expand Down
26 changes: 16 additions & 10 deletions subprojects/xcfa/xcfa/src/main/java/hu/bme/mit/theta/xcfa/Utils.kt
Original file line number Diff line number Diff line change
Expand Up @@ -121,6 +121,8 @@ typealias AccessType = Pair<Boolean, Boolean>

typealias VarAccessMap = Map<VarDecl<*>, AccessType>

typealias GlobalVarAccessMap = Map<XcfaGlobalVar, AccessType>

val AccessType?.isRead
get() = this?.first == true
val AccessType?.isWritten
Expand Down Expand Up @@ -245,17 +247,21 @@ fun XcfaLabel.collectVarsWithAccessType(): VarAccessMap =
/**
* Returns the global variables accessed by the label (the variables present in the given argument).
*/
private fun XcfaLabel.collectGlobalVars(globalVars: Set<VarDecl<*>>): VarAccessMap =
collectVarsWithAccessType().filter { labelVar -> globalVars.any { it == labelVar.key } }
private fun XcfaLabel.collectGlobalVars(globalVars: Set<XcfaGlobalVar>): GlobalVarAccessMap =
collectVarsWithAccessType()
.mapNotNull { labelVar ->
globalVars.firstOrNull { it.wrappedVar == labelVar.key }?.let { Pair(it, labelVar.value) }
}
.toMap()

/**
* Returns the global variables (potentially indirectly) accessed by the edge. If the edge starts an
* atomic block, all variable accesses in the atomic block are returned. Variables are associated
* with a pair of boolean values: the first is true if the variable is read and false otherwise. The
* second is similar for write access.
*/
fun XcfaEdge.collectIndirectGlobalVarAccesses(xcfa: XCFA): VarAccessMap {
val globalVars = xcfa.vars.map(XcfaGlobalVar::wrappedVar).toSet()
fun XcfaEdge.collectIndirectGlobalVarAccesses(xcfa: XCFA): GlobalVarAccessMap {
val globalVars = xcfa.vars
val flatLabels = getFlatLabels()
val mutexes =
flatLabels.filterIsInstance<FenceLabel>().flatMap { it.acquiredMutexes }.toMutableSet()
Expand All @@ -271,7 +277,7 @@ fun XcfaEdge.collectIndirectGlobalVarAccesses(xcfa: XCFA): VarAccessMap {
* (read/write) and the set of mutexes that are needed to perform the variable access.
*/
class GlobalVarAccessWithMutexes(
val varDecl: VarDecl<*>,
val globalVar: XcfaGlobalVar,
val access: AccessType,
val mutexes: Set<String>,
)
Expand All @@ -287,7 +293,7 @@ fun XcfaEdge.getGlobalVarsWithNeededMutexes(
xcfa: XCFA,
currentMutexes: Set<String>,
): List<GlobalVarAccessWithMutexes> {
val globalVars = xcfa.vars.map(XcfaGlobalVar::wrappedVar).toSet()
val globalVars = xcfa.vars
val neededMutexes = currentMutexes.toMutableSet()
val accesses = mutableListOf<GlobalVarAccessWithMutexes>()
getFlatLabels().forEach { label ->
Expand All @@ -299,7 +305,7 @@ fun XcfaEdge.getGlobalVarsWithNeededMutexes(
vars.mapNotNull { (varDecl, accessType) ->
if (
accesses.any {
it.varDecl == varDecl && (it.access == accessType && it.access == WRITE)
it.globalVar == varDecl && (it.access == accessType && it.access == WRITE)
}
) {
null
Expand All @@ -320,10 +326,10 @@ fun XcfaEdge.getGlobalVarsWithNeededMutexes(
* @return the set of encountered shared objects
*/
private fun XcfaEdge.collectGlobalVarsWithTraversal(
globalVars: Set<VarDecl<*>>,
globalVars: Set<XcfaGlobalVar>,
goFurther: Predicate<XcfaEdge>,
): VarAccessMap {
val vars = mutableMapOf<VarDecl<*>, AccessType>()
): GlobalVarAccessMap {
val vars = mutableMapOf<XcfaGlobalVar, AccessType>()
val exploredEdges = mutableListOf<XcfaEdge>()
val edgesToExplore = mutableListOf<XcfaEdge>()
edgesToExplore.add(this)
Expand Down
Loading

0 comments on commit d85d701

Please sign in to comment.