/* * Copyright (c) 2009-2011, IETR/INSA of Rennes * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of the IETR/INSA of Rennes nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY * WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ package net.sf.orcc.df.impl; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Comparator; import java.util.HashSet; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; import java.util.Set; import net.sf.orcc.df.Actor; import net.sf.orcc.df.Connection; import net.sf.orcc.df.DfPackage; import net.sf.orcc.df.Entity; import net.sf.orcc.df.Instance; import net.sf.orcc.df.Network; import net.sf.orcc.df.Port; import net.sf.orcc.df.util.DfUtil; import net.sf.orcc.graph.Edge; import net.sf.orcc.graph.Vertex; import net.sf.orcc.graph.impl.GraphImpl; import net.sf.orcc.ir.Var; import net.sf.orcc.moc.MoC; import org.eclipse.core.resources.IFile; import org.eclipse.core.resources.IWorkspaceRoot; import org.eclipse.core.resources.ResourcesPlugin; import org.eclipse.core.runtime.Path; import org.eclipse.emf.common.notify.Notification; import org.eclipse.emf.common.notify.NotificationChain; import org.eclipse.emf.common.util.EList; import org.eclipse.emf.ecore.EClass; import org.eclipse.emf.ecore.InternalEObject; import org.eclipse.emf.ecore.impl.ENotificationImpl; import org.eclipse.emf.ecore.util.EObjectContainmentEList; import org.eclipse.emf.ecore.util.EObjectResolvingEList; import org.eclipse.emf.ecore.util.InternalEList; /** * This class defines a hierarchical XDF network. It contains several maps so * templates can walk through the graph of the network. * * @author Matthieu Wipliez * @author Herve Yviquel * @generated */ public class NetworkImpl extends GraphImpl implements Network { private Entity cachedAdaptedEntity; /** * The cached value of the '{@link #getChildren() <em>Children</em>}' * reference list. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getChildren() * @generated * @ordered */ protected EList<Vertex> children; /** * The default value of the '{@link #getFileName() <em>File Name</em>}' * attribute. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getFileName() * @generated * @ordered */ protected static final String FILE_NAME_EDEFAULT = null; /** * The cached value of the '{@link #getFileName() <em>File Name</em>}' * attribute. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getFileName() * @generated * @ordered */ protected String fileName = FILE_NAME_EDEFAULT; /** * The cached value of the '{@link #getInputs() <em>Inputs</em>}' reference * list. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getInputs() * @generated * @ordered */ protected EList<Port> inputs; /** * The cached value of the '{@link #getMoC() <em>Mo C</em>}' containment * reference. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getMoC() * @generated * @ordered */ protected MoC moC; /** * The default value of the '{@link #getName() <em>Name</em>}' attribute. * <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getName() * @generated * @ordered */ protected static final String NAME_EDEFAULT = null; /** * The cached value of the '{@link #getOutputs() <em>Outputs</em>}' * reference list. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getOutputs() * @generated * @ordered */ protected EList<Port> outputs; /** * The cached value of the '{@link #getParameters() <em>Parameters</em>}' * containment reference list. <!-- begin-user-doc --> <!-- end-user-doc --> * * @see #getParameters() * @generated * @ordered */ protected EList<Var> parameters; /** * @generated */ protected EList<Var> variables; /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ protected NetworkImpl() { super(); } @Override public void add(Vertex vertex) { super.add(vertex); getChildren().add(vertex); } @Override public void addInput(Port port) { super.add(port); getInputs().add(port); } @Override public void addOutput(Port port) { super.add(port); getOutputs().add(port); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public NotificationChain basicSetMoC(MoC newMoC, NotificationChain msgs) { MoC oldMoC = moC; moC = newMoC; if (eNotificationRequired()) { ENotificationImpl notification = new ENotificationImpl(this, Notification.SET, DfPackage.NETWORK__MO_C, oldMoC, newMoC); if (msgs == null) msgs = notification; else msgs.add(notification); } return msgs; } /** * Computes the source map and target maps that associate each connection to * its source vertex (respectively target vertex). */ public void computeTemplateMaps() { int i = 0; for (Connection connection : getConnections()) { connection.setAttribute("id", i++); } i = 0; for (Vertex vertex : getChildren()) { Entity entity = vertex.getAdapter(Entity.class); Map<Port, List<Connection>> map = entity.getOutgoingPortMap(); for (List<Connection> connections : map.values()) { int j = 0; for (Connection connection : connections) { connection.setAttribute("idNoBcast", i); connection.setAttribute("fifoId", j); j++; } i++; } } for (Port port : inputs) { List<Edge> edges = port.getOutgoing(); for (Edge edge : edges) { edge.setAttribute("idNoBcast", i); } i++; } } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @Override public Object eGet(int featureID, boolean resolve, boolean coreType) { switch (featureID) { case DfPackage.NETWORK__CHILDREN: return getChildren(); case DfPackage.NETWORK__FILE_NAME: return getFileName(); case DfPackage.NETWORK__INPUTS: return getInputs(); case DfPackage.NETWORK__MO_C: return getMoC(); case DfPackage.NETWORK__NAME: return getName(); case DfPackage.NETWORK__OUTPUTS: return getOutputs(); case DfPackage.NETWORK__PARAMETERS: return getParameters(); case DfPackage.NETWORK__VARIABLES: return getVariables(); } return super.eGet(featureID, resolve, coreType); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @Override public NotificationChain eInverseRemove(InternalEObject otherEnd, int featureID, NotificationChain msgs) { switch (featureID) { case DfPackage.NETWORK__MO_C: return basicSetMoC(null, msgs); case DfPackage.NETWORK__PARAMETERS: return ((InternalEList<?>) getParameters()).basicRemove(otherEnd, msgs); case DfPackage.NETWORK__VARIABLES: return ((InternalEList<?>) getVariables()).basicRemove(otherEnd, msgs); } return super.eInverseRemove(otherEnd, featureID, msgs); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @Override public boolean eIsSet(int featureID) { switch (featureID) { case DfPackage.NETWORK__CHILDREN: return children != null && !children.isEmpty(); case DfPackage.NETWORK__FILE_NAME: return FILE_NAME_EDEFAULT == null ? fileName != null : !FILE_NAME_EDEFAULT.equals(fileName); case DfPackage.NETWORK__INPUTS: return inputs != null && !inputs.isEmpty(); case DfPackage.NETWORK__MO_C: return moC != null; case DfPackage.NETWORK__NAME: return NAME_EDEFAULT == null ? getName() != null : !NAME_EDEFAULT.equals(getName()); case DfPackage.NETWORK__OUTPUTS: return outputs != null && !outputs.isEmpty(); case DfPackage.NETWORK__PARAMETERS: return parameters != null && !parameters.isEmpty(); case DfPackage.NETWORK__VARIABLES: return variables != null && !variables.isEmpty(); } return super.eIsSet(featureID); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @SuppressWarnings("unchecked") @Override public void eSet(int featureID, Object newValue) { switch (featureID) { case DfPackage.NETWORK__CHILDREN: getChildren().clear(); getChildren().addAll((Collection<? extends Vertex>) newValue); return; case DfPackage.NETWORK__FILE_NAME: setFileName((String) newValue); return; case DfPackage.NETWORK__INPUTS: getInputs().clear(); getInputs().addAll((Collection<? extends Port>) newValue); return; case DfPackage.NETWORK__MO_C: setMoC((MoC) newValue); return; case DfPackage.NETWORK__NAME: setName((String) newValue); return; case DfPackage.NETWORK__OUTPUTS: getOutputs().clear(); getOutputs().addAll((Collection<? extends Port>) newValue); return; case DfPackage.NETWORK__PARAMETERS: getParameters().clear(); getParameters().addAll((Collection<? extends Var>) newValue); return; case DfPackage.NETWORK__VARIABLES: getVariables().clear(); getVariables().addAll((Collection<? extends Var>) newValue); return; } super.eSet(featureID, newValue); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @Override protected EClass eStaticClass() { return DfPackage.Literals.NETWORK; } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ @Override public void eUnset(int featureID) { switch (featureID) { case DfPackage.NETWORK__CHILDREN: getChildren().clear(); return; case DfPackage.NETWORK__FILE_NAME: setFileName(FILE_NAME_EDEFAULT); return; case DfPackage.NETWORK__INPUTS: getInputs().clear(); return; case DfPackage.NETWORK__MO_C: setMoC((MoC) null); return; case DfPackage.NETWORK__NAME: setName(NAME_EDEFAULT); return; case DfPackage.NETWORK__OUTPUTS: getOutputs().clear(); return; case DfPackage.NETWORK__PARAMETERS: getParameters().clear(); return; case DfPackage.NETWORK__VARIABLES: getVariables().clear(); return; } super.eUnset(featureID); } @Override @SuppressWarnings("unchecked") public <T> T getAdapter(Class<T> type) { if (type == Entity.class) { if (cachedAdaptedEntity == null) { cachedAdaptedEntity = new EntityImpl(this, getAttributes(), getInputs(), getOutputs(), getParameters()); } return (T) cachedAdaptedEntity; } return super.getAdapter(type); } /** * Returns the list of actors referenced by the graph of this network. This * is different from the list of instances of this network: There are * typically more instances than there are actors, because an actor may be * instantiated several times. * * <p> * The list is computed on the fly by adding all the actors referenced in a * set. * </p> * * @return a list of actors */ public List<Actor> getAllActors() { Set<Actor> actors = new HashSet<Actor>(); for (Vertex vertex : getChildren()) { Actor actor = vertex.getAdapter(Actor.class); if (actor == null) { Network network = vertex.getAdapter(Network.class); if (network != null) { actors.addAll(network.getAllActors()); } } else { actors.add(actor); } } List<Actor> list = new ArrayList<Actor>(actors); Collections.sort(list, new Comparator<Actor>() { @Override public int compare(Actor o1, Actor o2) { return o1.getName().compareTo(o2.getName()); } }); return list; } @Override public List<Network> getAllNetworks() { Set<Network> networks = new LinkedHashSet<Network>(); for (Vertex vertex : getChildren()) { Network network = vertex.getAdapter(Network.class); if (network != null) { networks.add(network); networks.addAll(network.getAllNetworks()); } } return new ArrayList<Network>(networks); } @Override public Vertex getChild(String name) { for (Vertex vertex : getChildren()) { if (vertex.getLabel().equals(name)) { return vertex; } } return null; } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public EList<Vertex> getChildren() { if (children == null) { children = new EObjectResolvingEList<Vertex>(Vertex.class, this, DfPackage.NETWORK__CHILDREN); } return children; } @Override @SuppressWarnings("unchecked") public EList<Connection> getConnections() { return (EList<Connection>) (EList<?>) getEdges(); } @Override public IFile getFile() { String fileName = getFileName(); if (fileName == null) { return null; } IWorkspaceRoot root = ResourcesPlugin.getWorkspace().getRoot(); return root.getFile(new Path(fileName)); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public String getFileName() { return fileName; } @Override public Port getInput(String name) { for (Port port : getInputs()) { if (port.getName().equals(name)) { return port; } } return null; } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public EList<Port> getInputs() { if (inputs == null) { inputs = new EObjectResolvingEList<Port>(Port.class, this, DfPackage.NETWORK__INPUTS); } return inputs; } /** * Returns the list of instances of the given actor in the graph. * * @param actor * the actor to get the instance of * * @return a list of instances */ public List<Instance> getInstancesOf(Actor actor) { List<Instance> instances = new ArrayList<Instance>(); for (Vertex vertex : getChildren()) { Actor candidate = vertex.getAdapter(Actor.class); if (candidate == null) { Network network = vertex.getAdapter(Network.class); if (network != null) { // if vertex is a network call getInstancesOf recursively instances.addAll(network.getInstancesOf(actor)); } } else if (candidate == actor) { Instance instance = vertex.getAdapter(Instance.class); if (instance != null) { // make sure the vertex is an instance instances.add(instance); } } } return instances; } /** * Returns the MoC of the network. * * @return the network MoC. * @generated */ public MoC getMoC() { return moC; } @Override public String getName() { return getLabel(); } @Override public Port getOutput(String name) { for (Port port : getOutputs()) { if (port.getName().equals(name)) { return port; } } return null; } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public EList<Port> getOutputs() { if (outputs == null) { outputs = new EObjectResolvingEList<Port>(Port.class, this, DfPackage.NETWORK__OUTPUTS); } return outputs; } @Override public String getPackage() { return DfUtil.getPackage(getName()); } @Override public Var getParameter(String name) { for (Var var : getParameters()) { if (var.getName().equals(name)) { return var; } } return null; } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public EList<Var> getParameters() { if (parameters == null) { parameters = new EObjectContainmentEList<Var>(Var.class, this, DfPackage.NETWORK__PARAMETERS); } return parameters; } @Override public String getSimpleName() { return DfUtil.getSimpleName(getName()); } @Override public Var getVariable(String name) { for (Var var : getVariables()) { if (var.getName().equals(name)) { return var; } } return null; } /** * Returns the list of this network's variables * * @return the list of this network's variables * @generated */ public EList<Var> getVariables() { if (variables == null) { variables = new EObjectContainmentEList<Var>(Var.class, this, DfPackage.NETWORK__VARIABLES); } return variables; } public boolean isNetwork() { return true; } @Override public void remove(Vertex vertex) { getChildren().remove(vertex); super.remove(vertex); } /** * <!-- begin-user-doc --> <!-- end-user-doc --> * * @generated */ public void setFileName(String newFileName) { String oldFileName = fileName; fileName = newFileName; if (eNotificationRequired()) eNotify(new ENotificationImpl(this, Notification.SET, DfPackage.NETWORK__FILE_NAME, oldFileName, fileName)); } /** * Sets the MoC of this network. * * @param moc * the new MoC of this network * @generated */ public void setMoC(MoC newMoC) { if (newMoC != moC) { NotificationChain msgs = null; if (moC != null) msgs = ((InternalEObject) moC).eInverseRemove(this, EOPPOSITE_FEATURE_BASE - DfPackage.NETWORK__MO_C, null, msgs); if (newMoC != null) msgs = ((InternalEObject) newMoC).eInverseAdd(this, EOPPOSITE_FEATURE_BASE - DfPackage.NETWORK__MO_C, null, msgs); msgs = basicSetMoC(newMoC, msgs); if (msgs != null) msgs.dispatch(); } else if (eNotificationRequired()) eNotify(new ENotificationImpl(this, Notification.SET, DfPackage.NETWORK__MO_C, newMoC, newMoC)); } @Override public void setName(String newName) { setLabel(newName); } @Override public String toString() { if (eIsProxy()) return super.toString(); return getName(); } }