JimpleBody body = Jimple.v().newBody(preinitializeMethod);
preinitializeMethod.setActiveBody(body);
body.insertIdentityStmts();
Stmt insertPoint = Jimple.v().newReturnVoidStmt();
body.getUnits().add(insertPoint);
ModelTransformer.initializeAttributesBefore(body, insertPoint,
entity, body.getThisLocal(), entity, body.getThisLocal(),
entityInstanceClass);
}
// Add a field to keep track of the current state.
SootField currentStateField = new SootField("_currentState", IntType
.v());
entityInstanceClass.addField(currentStateField);
SootField nextTransitionField = new SootField("_nextTransition",
IntType.v());
entityInstanceClass.addField(nextTransitionField);
// populate the initialize method.
{
System.out.println("create initialize()");
SootMethod initializeMethod = new SootMethod("initialize",
Collections.EMPTY_LIST, VoidType.v(), Modifier.PUBLIC);
entityInstanceClass.addMethod(initializeMethod);
JimpleBody body = Jimple.v().newBody(initializeMethod);
initializeMethod.setActiveBody(body);
body.insertIdentityStmts();
Chain units = body.getUnits();
Local thisLocal = body.getThisLocal();
// Set the initial state.
String initialStateName = ((StringAttribute) entity
.getAttribute("initialStateName")).getExpression();
int initialStateIndex = entity.entityList().indexOf(
entity.getEntity(initialStateName));
units.add(Jimple.v().newAssignStmt(
Jimple.v().newInstanceFieldRef(thisLocal,
currentStateField.makeRef()),
IntConstant.v(initialStateIndex)));
// return void
units.add(Jimple.v().newReturnVoidStmt());
LocalNameStandardizer.v().transform(body, "at.lns");
LocalSplitter.v().transform(body, "at.ls");
}
// populate the fire method.
{
System.out.println("create fire()");
SootMethod fireMethod = new SootMethod("fire",
Collections.EMPTY_LIST, VoidType.v(), Modifier.PUBLIC);
entityInstanceClass.addMethod(fireMethod);
JimpleBody body = Jimple.v().newBody(fireMethod);
fireMethod.setActiveBody(body);
body.insertIdentityStmts();
Chain units = body.getUnits();
Local thisLocal = body.getThisLocal();
Local hasTokenLocal = Jimple.v().newLocal("hasTokenLocal",
BooleanType.v());
body.getLocals().add(hasTokenLocal);
Local tokenLocal = Jimple.v().newLocal("tokenLocal",
PtolemyUtilities.tokenType);
body.getLocals().add(tokenLocal);
Iterator inputPorts = entity.inputPortList().iterator();
while (inputPorts.hasNext()) {
TypedIOPort port = (TypedIOPort) (inputPorts.next());
// FIXME: Handle multiports
if (port.getWidth() > 0) {
String name = port.getName(entity);
// Create an if statement.
//
Local portLocal = Jimple.v().newLocal("port",
PtolemyUtilities.componentPortType);
body.getLocals().add(portLocal);
SootField portField = entityInstanceClass
.getFieldByName(StringUtilities.sanitizeName(name));
units.add(Jimple.v().newAssignStmt(
portLocal,
Jimple.v().newInstanceFieldRef(thisLocal,
portField.makeRef())));
units.add(Jimple.v().newAssignStmt(
hasTokenLocal,
Jimple.v().newVirtualInvokeExpr(portLocal,
PtolemyUtilities.hasTokenMethod.makeRef(),
IntConstant.v(0))));
Local hasTokenToken = PtolemyUtilities.addTokenLocal(body,
"token", PtolemyUtilities.booleanTokenClass,
PtolemyUtilities.booleanTokenConstructor,
hasTokenLocal);
// store the isPresent
SootField tokenIsPresentField = (SootField) nameToField
.get(name + "_isPresent");
units.add(Jimple.v().newAssignStmt(
Jimple.v().newInstanceFieldRef(thisLocal,
tokenIsPresentField.makeRef()),
hasTokenToken));
Stmt target = Jimple.v().newNopStmt();
units.add(Jimple.v().newIfStmt(
Jimple.v().newEqExpr(hasTokenLocal,
IntConstant.v(0)), target));
units.add(Jimple.v().newAssignStmt(
tokenLocal,
Jimple.v().newVirtualInvokeExpr(portLocal,
PtolemyUtilities.getMethod.makeRef(),
IntConstant.v(0))));
SootField tokenField = (SootField) nameToField.get(name);
units.add(Jimple.v().newAssignStmt(
Jimple.v().newInstanceFieldRef(thisLocal,
tokenField.makeRef()), tokenLocal));
units.add(target);
}
}
Map stateToStartStmt = new HashMap();
List stateStmtList = new LinkedList();
int numberOfStates = entity.entityList().size();
// Figure out what state we are in.
for (Iterator states = entity.entityList().iterator(); states
.hasNext();) {
State state = (State) states.next();
Stmt startStmt = Jimple.v().newNopStmt();
stateToStartStmt.put(state, startStmt);
stateStmtList.add(startStmt);
}
Local currentStateLocal = Jimple.v().newLocal("currentStateLocal",
IntType.v());
body.getLocals().add(currentStateLocal);
Local nextStateLocal = Jimple.v().newLocal("nextStateLocal",
IntType.v());
body.getLocals().add(nextStateLocal);
Local flagLocal = Jimple.v().newLocal("flagLocal", BooleanType.v());
body.getLocals().add(flagLocal);
Local transitionTakenLocal = Jimple.v().newLocal(
"transitionTakenLocal", BooleanType.v());
body.getLocals().add(transitionTakenLocal);
Local nextTransitionLocal = Jimple.v().newLocal(
"nextTransitionLocal", IntType.v());
body.getLocals().add(nextTransitionLocal);
units.add(Jimple.v().newAssignStmt(
currentStateLocal,
Jimple.v().newInstanceFieldRef(thisLocal,
currentStateField.makeRef())));
// Start by doing nothing.
units.add(Jimple.v().newAssignStmt(transitionTakenLocal,
IntConstant.v(0)));
units.add(Jimple.v().newAssignStmt(nextTransitionLocal,
IntConstant.v(-1)));
// If no transition is taken, then stay in this state.
units.add(Jimple.v().newAssignStmt(nextStateLocal,
currentStateLocal));
Stmt finishedStmt = Jimple.v().newNopStmt();
Stmt errorStmt = Jimple.v().newNopStmt();
// Get the current state.
units.add(Jimple.v().newTableSwitchStmt(currentStateLocal, 0,
numberOfStates - 1, stateStmtList, errorStmt));
// Generate code for each state.
for (Iterator states = entity.entityList().iterator(); states
.hasNext();) {
State state = (State) states.next();
System.out.println("state " + state.getName());
Stmt startStmt = (Stmt) stateToStartStmt.get(state);
units.add(startStmt);
// Fire the refinement actor.
TypedActor[] refinements = null;
try {
refinements = state.getRefinement();
} catch (Exception ex) {
throw new RuntimeException(ex.getMessage());
}
if (refinements != null) {
for (int i = 0; i < refinements.length; i++) {
TypedActor refinement = refinements[i];
Local containerLocal = Jimple.v().newLocal("container",
RefType.v(PtolemyUtilities.namedObjClass));
body.getLocals().add(containerLocal);
Local entityLocal = Jimple.v().newLocal("entity",
RefType.v(PtolemyUtilities.entityClass));
body.getLocals().add(entityLocal);
NamedObj containerModel = entity.getContainer();
String deepName = ((NamedObj) refinement)
.getName(containerModel);
units.add(Jimple.v().newAssignStmt(
containerLocal,
Jimple.v().newInterfaceInvokeExpr(
thisLocal,
PtolemyUtilities.getContainerMethod
.makeRef())));
units
.add(Jimple
.v()
.newAssignStmt(
containerLocal,
Jimple
.v()
.newCastExpr(
containerLocal,
RefType
.v(PtolemyUtilities.compositeActorClass))));
units.add(Jimple.v().newAssignStmt(
entityLocal,
Jimple.v().newVirtualInvokeExpr(
containerLocal,
PtolemyUtilities.getEntityMethod
.makeRef(),
StringConstant.v(deepName))));
units
.add(Jimple
.v()
.newAssignStmt(
entityLocal,
Jimple
.v()
.newCastExpr(
entityLocal,
RefType
.v(PtolemyUtilities.compositeActorClass))));
SootMethod rprefireMethod;
SootMethod rfireMethod;
SootMethod rpostfireMethod;
if (refinement instanceof CompositeActor) {
rprefireMethod = SootUtilities
.searchForMethodByName(
PtolemyUtilities.compositeActorClass,
"prefire");
rfireMethod = SootUtilities.searchForMethodByName(
PtolemyUtilities.compositeActorClass,
"fire");
rpostfireMethod = SootUtilities
.searchForMethodByName(
PtolemyUtilities.compositeActorClass,
"postfire");
} else {
throw new RuntimeException();
}
units.add(Jimple.v().newInvokeStmt(
Jimple.v().newVirtualInvokeExpr(entityLocal,
rprefireMethod.makeRef())));
units.add(Jimple.v().newInvokeStmt(
Jimple.v().newVirtualInvokeExpr(entityLocal,
rfireMethod.makeRef())));
units.add(Jimple.v().newInvokeStmt(
Jimple.v().newVirtualInvokeExpr(entityLocal,
rpostfireMethod.makeRef())));
}
}
// Determine the next state in this state.
for (Iterator transitions = state.outgoingPort
.linkedRelationList().iterator(); transitions.hasNext();) {
Transition transition = (Transition) transitions.next();
System.out.println("transition = " + transition);
String guardExpression = transition.getGuardExpression();
Local guardLocal = DataUtilities.generateExpressionCode(
entity, entityInstanceClass, guardExpression,
nameToField, nameToType, body);
// Test the guard.
units
.add(Jimple
.v()
.newAssignStmt(
tokenLocal,
Jimple
.v()
.newCastExpr(
guardLocal,
RefType
.v(PtolemyUtilities.booleanTokenClass))));
units.add(Jimple.v().newAssignStmt(
flagLocal,
Jimple.v().newVirtualInvokeExpr(
tokenLocal,
PtolemyUtilities.booleanValueMethod
.makeRef())));
Stmt skipStmt = Jimple.v().newNopStmt();
units.add(Jimple.v().newIfStmt(
Jimple.v().newEqExpr(flagLocal, IntConstant.v(0)),
skipStmt));
units.add(Jimple.v().newIfStmt(
Jimple.v().newEqExpr(transitionTakenLocal,
IntConstant.v(1)), errorStmt));
// If transition taken, then store the next state
units.add(Jimple.v().newAssignStmt(transitionTakenLocal,
IntConstant.v(1)));
units.add(Jimple.v().newAssignStmt(
nextTransitionLocal,
IntConstant.v(entity.relationList().indexOf(
transition))));
int nextStateIndex = entity.entityList().indexOf(
transition.destinationState());
units.add(Jimple.v().newAssignStmt(nextStateLocal,
IntConstant.v(nextStateIndex)));
// Generate code for the outputExpression of the guard.
for (Iterator actions = transition.choiceActionList()
.iterator(); actions.hasNext();) {
AbstractActionsAttribute action = (AbstractActionsAttribute) actions
.next();
System.out.println("action = " + action);
_generateActionCode(entity, entityInstanceClass,
nameToField, nameToType, body, action);
}
units.add(skipStmt);
}
units.add(Jimple.v().newGotoStmt(finishedStmt));
}
units.add(errorStmt);
// throw an exception.
units.add(finishedStmt);
Local exceptionLocal = SootUtilities.createRuntimeException(body,
errorStmt, "state error");
units.insertBefore(Jimple.v().newThrowStmt(exceptionLocal),
errorStmt);
// Store the next state.
units.add(Jimple.v().newAssignStmt(
Jimple.v().newInstanceFieldRef(thisLocal,
currentStateField.makeRef()), nextStateLocal));
// And the next Transition.
units.add(Jimple.v()
.newAssignStmt(
Jimple.v().newInstanceFieldRef(thisLocal,
nextTransitionField.makeRef()),
nextTransitionLocal));
// return void
units.add(Jimple.v().newReturnVoidStmt());
LocalNameStandardizer.v().transform(body, "at.lns");
LocalSplitter.v().transform(body, "at.ls");
}
// populate the postfire method.
{
System.out.println("create postfire()");
SootMethod postfireMethod = new SootMethod("postfire",
Collections.EMPTY_LIST, BooleanType.v(), Modifier.PUBLIC);
entityInstanceClass.addMethod(postfireMethod);
JimpleBody body = Jimple.v().newBody(postfireMethod);
postfireMethod.setActiveBody(body);
body.insertIdentityStmts();
Chain units = body.getUnits();
Local thisLocal = body.getThisLocal();
Map transitionToStartStmt = new HashMap();
List transitionStmtList = new LinkedList();
int numberOfTransitions = entity.relationList().size();
// Figure out what transition we are in.
for (Iterator transitions = entity.relationList().iterator(); transitions
.hasNext();) {
Transition transition = (Transition) transitions.next();
Stmt startStmt = Jimple.v().newNopStmt();
transitionToStartStmt.put(transition, startStmt);
transitionStmtList.add(startStmt);
}
Local nextTransitionLocal = Jimple.v().newLocal(
"nextTransitionLocal", IntType.v());
body.getLocals().add(nextTransitionLocal);
units.add(Jimple.v().newAssignStmt(
nextTransitionLocal,
Jimple.v().newInstanceFieldRef(thisLocal,
nextTransitionField.makeRef())));
Stmt finishedStmt = Jimple.v().newNopStmt();
Stmt errorStmt = Jimple.v().newNopStmt();
// Get the current transition..
units.add(Jimple.v().newTableSwitchStmt(nextTransitionLocal, 0,
numberOfTransitions - 1, transitionStmtList, errorStmt));
// Generate code for each transition
for (Iterator transitions = entity.relationList().iterator(); transitions
.hasNext();) {
Transition transition = (Transition) transitions.next();
Stmt startStmt = (Stmt) transitionToStartStmt.get(transition);
units.add(startStmt);
// Generate code for the commitExpression of the guard.
for (Iterator actions = transition.commitActionList()
.iterator(); actions.hasNext();) {