SemanticAdaptationCanonicalGenerator.xtend 43 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202
  1. /*
  2. * generated by Xtext 2.10.0
  3. */
  4. package be.uantwerpen.ansymo.semanticadaptation.generator
  5. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Adaptation
  6. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.BoolLiteral
  7. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.BuiltinFunction
  8. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Close
  9. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.CompositeOutputFunction
  10. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Connection
  11. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DataRule
  12. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Declaration
  13. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DeclaredParameter
  14. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Expression
  15. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.FMU
  16. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMU
  17. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMUDeclaration
  18. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMUDeclarationFull
  19. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.IntLiteral
  20. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.IsSet
  21. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.OutputFunction
  22. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Port
  23. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.RealLiteral
  24. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SemanticAdaptation
  25. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SemanticAdaptationFactory
  26. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SingleParamDeclaration
  27. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SingleVarDeclaration
  28. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.StateTransitionFunction
  29. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.StringLiteral
  30. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Unity
  31. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Variable
  32. import java.io.ByteArrayOutputStream
  33. import java.util.HashMap
  34. import java.util.LinkedList
  35. import java.util.List
  36. import java.util.Map
  37. import org.eclipse.emf.ecore.EObject
  38. import org.eclipse.emf.ecore.resource.Resource
  39. import org.eclipse.xtext.EcoreUtil2
  40. import org.eclipse.xtext.generator.AbstractGenerator
  41. import org.eclipse.xtext.generator.IFileSystemAccess2
  42. import org.eclipse.xtext.generator.IGeneratorContext
  43. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Statement
  44. /**
  45. * Generates code from your model files on save.
  46. *
  47. * See https://www.eclipse.org/Xtext/documentation/303_runtime_concepts.html#code-generation
  48. */
  49. class SemanticAdaptationCanonicalGenerator extends AbstractGenerator {
  50. String CANONICAL_EXT = ".BASE.sa"
  51. String NAME_SUFFIX = "_BASE"
  52. override void doGenerate(Resource resource, IFileSystemAccess2 fsa, IGeneratorContext context) {
  53. Log.push("Generating canonical semantic adaptation for file " + resource.URI.toFileString() + "...")
  54. Log.println("Resource URI information:")
  55. Log.println("\t resource.URI.lastSegment = " + resource.URI.lastSegment())
  56. Log.println("\t resource.URI.trimFileExtension = " + resource.URI.trimFileExtension())
  57. // Create in memory representation of canonical SA file
  58. var adaptations = resource.allContents.toIterable.filter(SemanticAdaptation).last.elements.filter(Adaptation);
  59. if (adaptations.size > 1){
  60. throw new Exception("Only one semantic adaptation is supported per .sa file")
  61. }
  62. var adaptation = adaptations.head
  63. Log.println(prettyprint_model(adaptation, "File Read"))
  64. // Create file name for the canonical sa file
  65. var fileNameWithoutExt = resource.URI.trimFileExtension().lastSegment()
  66. var canonicalFileName = fileNameWithoutExt + CANONICAL_EXT
  67. Log.println("canonicalFileName = " + canonicalFileName)
  68. Log.println("Checking if file is already a canonical version...")
  69. if (adaptation.name.indexOf(NAME_SUFFIX) == -1){
  70. Log.println("It is not.")
  71. adaptation.name = adaptation.name + NAME_SUFFIX
  72. canonicalize(adaptation)
  73. Log.println(prettyprint_model(adaptation, "Generated File"))
  74. fsa.generateFile(canonicalFileName, adaptation.serialize_model)
  75. Log.println("File " + canonicalFileName + " written.")
  76. } else {
  77. Log.println("It is already a canonical version.")
  78. Log.println("Nothing to do.")
  79. }
  80. Log.pop("Generating canonical semantic adaptation for file " + resource.URI.toFileString() + "... DONE.")
  81. }
  82. def prettyprint_model(Adaptation sa, String title){
  83. var outputByteArray = new ByteArrayOutputStream()
  84. sa.eResource.save(outputByteArray,null)
  85. return "______________________________" + title + "______________________________\n" +
  86. sa.serialize_model +
  87. "\n__________________________________________________________________________"
  88. }
  89. def serialize_model(Adaptation sa){
  90. var outputByteArray = new ByteArrayOutputStream()
  91. sa.eResource.save(outputByteArray,null)
  92. return outputByteArray.toString()
  93. }
  94. def inferUnits(Adaptation sa){
  95. // Unit inference
  96. var unitlessElements = genericDeclarationInferenceAlgorithm(sa ,
  97. [// getField
  98. element | {
  99. var DUMMY_UNIT = "Dips"
  100. if (element instanceof SingleParamDeclaration) {
  101. return DUMMY_UNIT
  102. } else if (element instanceof Port){
  103. return element.unity
  104. } else if (element instanceof SingleVarDeclaration){
  105. return DUMMY_UNIT
  106. } else {
  107. throw new Exception("Unexpected element type: " + element)
  108. }
  109. }
  110. ],
  111. [// setField
  112. element, value | {
  113. if (element instanceof SingleParamDeclaration) {
  114. } else if (element instanceof Port){
  115. element.unity = EcoreUtil2.copy(value as Unity)
  116. } else if (element instanceof SingleVarDeclaration){
  117. } else {
  118. throw new Exception("Unexpected element type: " + element)
  119. }
  120. }
  121. ],
  122. [// inferField
  123. element | {
  124. var DUMMY_UNIT = "Dips"
  125. if (element instanceof SingleParamDeclaration) {
  126. return DUMMY_UNIT
  127. } else if (element instanceof Port){
  128. return getPortUnit(element)
  129. } else if (element instanceof SingleVarDeclaration){
  130. return DUMMY_UNIT
  131. } else {
  132. throw new Exception("Unexpected element type: " + element)
  133. }
  134. }
  135. ]
  136. )
  137. if (unitlessElements > 0){
  138. Log.println("Could not infer all element units. There are " + unitlessElements + " unitless elements.")
  139. }
  140. }
  141. def inferTypes(Adaptation sa){
  142. // Type inference
  143. var untypedElements = genericDeclarationInferenceAlgorithm(sa ,
  144. [// getField
  145. element | {
  146. if (element instanceof SingleParamDeclaration) {
  147. return element.type
  148. } else if (element instanceof Port){
  149. return element.type
  150. } else if (element instanceof SingleVarDeclaration){
  151. return element.type
  152. } else {
  153. throw new Exception("Unexpected element type: " + element)
  154. }
  155. }
  156. ],
  157. [// setField
  158. element, value | {
  159. if (element instanceof SingleParamDeclaration) {
  160. element.type = value as String
  161. } else if (element instanceof Port){
  162. element.type = value as String
  163. } else if (element instanceof SingleVarDeclaration){
  164. element.type = value as String
  165. } else {
  166. throw new Exception("Unexpected element type: " + element)
  167. }
  168. }
  169. ],
  170. [// inferField
  171. element | {
  172. if (element instanceof SingleParamDeclaration) {
  173. return extractTypeFromExpression(element.expr, element.name)
  174. } else if (element instanceof Port){
  175. return getPortType(element)
  176. } else if (element instanceof SingleVarDeclaration){
  177. return extractTypeFromExpression(element.expr, element.name)
  178. } else {
  179. throw new Exception("Unexpected element type: " + element)
  180. }
  181. }
  182. ]
  183. )
  184. if (untypedElements > 0){
  185. Log.println("Error: Could not infer all types. There are " + untypedElements + " untyped elements.")
  186. Log.println(prettyprint_model(sa, "Current File"))
  187. throw new Exception("Could not infer all types. There are " + untypedElements + " untyped elements.")
  188. }
  189. }
  190. def canonicalize(Adaptation sa){
  191. Log.push("Canonicalize")
  192. inferUnits(sa)
  193. inferTypes(sa)
  194. addInPorts(sa)
  195. val inputPort2parameterDeclaration = addInParams(sa)
  196. val externalInputPort2InVarDeclaration = addInVars(sa, inputPort2parameterDeclaration)
  197. addInRules_External2Stored_Assignments(sa, externalInputPort2InVarDeclaration)
  198. val internalPort2ExternalPortBindings = findAllExternalPort2InputPort_Bindings(sa)
  199. addInRules_External2Internal_Assignments(sa, internalPort2ExternalPortBindings)
  200. removeInBindings(internalPort2ExternalPortBindings, sa)
  201. addOutPorts(sa)
  202. val outputPort2parameterDeclaration = addOutParams(sa)
  203. val externalOutputPort2OutVarDeclaration = addOutVars(sa, outputPort2parameterDeclaration)
  204. val internalOutputPort2ExternalPortBindings = findAllInternalPort2ExternalOutputPort_Bindings(sa)
  205. val internalOutputPort2OutVarDeclaration = transitiveStep(internalOutputPort2ExternalPortBindings, externalOutputPort2OutVarDeclaration)
  206. addOutRules_Internal2Stored_Assignments(sa, internalOutputPort2OutVarDeclaration)
  207. addOutRules_Internal2External_Assignments(sa, internalOutputPort2ExternalPortBindings)
  208. Log.pop("Canonicalize")
  209. }
  210. def transitiveStep(HashMap<Port, Port> internalOutputPort2ExternalPortBindings, HashMap<Port, SingleVarDeclaration> externalOutputPort2OutVarDeclaration) {
  211. Log.push("transitiveStep")
  212. val internalOutputPort2OutVarDeclaration = new HashMap<Port, SingleVarDeclaration>()
  213. for(internalOutputPort : internalOutputPort2ExternalPortBindings.keySet){
  214. val externalOutputPort = internalOutputPort2ExternalPortBindings.get(internalOutputPort)
  215. if (externalOutputPort2OutVarDeclaration.containsKey(externalOutputPort)){
  216. val outVar = externalOutputPort2OutVarDeclaration.get(externalOutputPort)
  217. Log.println("Found binding: " + internalOutputPort.qualifiedName + "->" + externalOutputPort.qualifiedName + " to be stored in " + outVar.name)
  218. internalOutputPort2OutVarDeclaration.put(internalOutputPort, outVar)
  219. }
  220. }
  221. Log.pop("transitiveStep")
  222. return internalOutputPort2OutVarDeclaration
  223. }
  224. def String qualifiedName(Port port){
  225. if (port.eContainer instanceof FMU){
  226. return (port.eContainer as FMU).name + "." + port.name
  227. }
  228. return port.name
  229. }
  230. def removeInBindings(HashMap<Port, Port> internalPort2ExternalPortBindings, Adaptation sa) {
  231. Log.push("removeInBindings")
  232. for (internalPort : internalPort2ExternalPortBindings.keySet){
  233. val externalPort = internalPort2ExternalPortBindings.get(internalPort)
  234. Log.println("Removing binding " + externalPort.qualifiedName + "->" + internalPort.qualifiedName)
  235. externalPort.targetdependency = null
  236. }
  237. Log.pop("removeInBindings")
  238. }
  239. def findAllExternalPort2InputPort_Bindings(Adaptation sa) {
  240. Log.push("findAllExternalPort2InputPort_Bindings")
  241. val internalPort2ExternalPortBindings = new HashMap<Port, Port>()
  242. for (port : getAllInnerFMUInputPortDeclarations(sa)){
  243. var parentFMU = port.eContainer as InnerFMU
  244. Log.println("Checking if port " + port.qualifiedName + " is bound to an external port.")
  245. val externalPort = findExternalPortByTargetDependency(sa.inports, port)
  246. if (externalPort !== null){
  247. Log.println("Port " + port.qualifiedName + " is bound to an external port: " + externalPort.qualifiedName)
  248. internalPort2ExternalPortBindings.put(port, externalPort)
  249. } else {
  250. Log.println("Port " + port.qualifiedName + " is not bound to an external port.")
  251. }
  252. }
  253. Log.pop("findAllExternalPort2InputPort_Bindings")
  254. return internalPort2ExternalPortBindings
  255. }
  256. def findAllInternalPort2ExternalOutputPort_Bindings(Adaptation sa) {
  257. Log.push("findAllInternalPort2ExternalOutputPort_Bindings")
  258. val internalPort2ExternalPortBindings = new HashMap<Port, Port>()
  259. for (port : getAllInnerFMUOutputPortDeclarations(sa)){
  260. Log.println("Checking if port " + port.qualifiedName + " is bound to an external port.")
  261. val externalPort = findExternalPortBySourceDependency(sa.outports, port)
  262. if (externalPort !== null){
  263. Log.println("Port " + externalPort.qualifiedName + " is bound to an internal port: " + port.qualifiedName
  264. )
  265. internalPort2ExternalPortBindings.put(port, externalPort)
  266. } else {
  267. Log.println("Port " + port.qualifiedName + " is not bound to an external port.")
  268. }
  269. }
  270. Log.pop("findAllInternalPort2ExternalOutputPort_Bindings")
  271. return internalPort2ExternalPortBindings
  272. }
  273. def createExternalPortNameFromInternalPort(String parentFMUName, String internalPortName) {
  274. //return parentFMUName + "__" + internalPortName // Violates transparency
  275. return internalPortName
  276. }
  277. def addInRules_External2Internal_Assignments(Adaptation sa, HashMap<Port, Port> internalPort2ExternalPort) {
  278. Log.push("addInRules_External2Internal_Assignments")
  279. val dataRule = getOrPrependTrueRule(sa.in.rules)
  280. for(internalPort : internalPort2ExternalPort.keySet){
  281. val externalPort = internalPort2ExternalPort.get(internalPort)
  282. addPortAssignment(dataRule.outputfunction, internalPort, externalPort)
  283. }
  284. Log.pop("addInRules_External2Internal_Assignments")
  285. }
  286. def addOutRules_Internal2External_Assignments(Adaptation sa, HashMap<Port, Port> internalPort2ExternalPort){
  287. Log.push("addOutRules_Internal2External_Assignments")
  288. val dataRule = getOrPrependTrueRule(sa.out.rules)
  289. for(internalPort : internalPort2ExternalPort.keySet){
  290. val externalPort = internalPort2ExternalPort.get(internalPort)
  291. addPortAssignment(dataRule.outputfunction, externalPort, internalPort)
  292. }
  293. Log.pop("addOutRules_Internal2External_Assignments")
  294. }
  295. def addPortAssignment(OutputFunction function, Port toPort, Port fromPort) {
  296. Log.push("addPortAssignment")
  297. if(! (function instanceof CompositeOutputFunction) ){
  298. throw new Exception("Only CompositeOutputFunction is supported for now.")
  299. }
  300. val assignment = SemanticAdaptationFactory.eINSTANCE.createAssignment()
  301. assignment.lvalue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  302. (assignment.lvalue as Variable).owner = toPort.eContainer as FMU
  303. (assignment.lvalue as Variable).ref = toPort
  304. assignment.expr = SemanticAdaptationFactory.eINSTANCE.createVariable()
  305. (assignment.expr as Variable).owner = fromPort.eContainer as FMU
  306. (assignment.expr as Variable).ref = fromPort
  307. val outFunction = function as CompositeOutputFunction
  308. outFunction.statements.add(0, assignment)
  309. Log.println("Assignment " + toPort.qualifiedName + " := " + fromPort.qualifiedName + " created.")
  310. Log.pop("addPortAssignment")
  311. }
  312. def addInRules_External2Stored_Assignments(Adaptation sa, HashMap<Port, SingleVarDeclaration> inputPort2InVarDeclaration) {
  313. Log.push("addInRules_External2Stored_Assignments")
  314. if (sa.in === null){
  315. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  316. }
  317. addRules_Port2Stored_Assignments(sa.in.rules, inputPort2InVarDeclaration)
  318. Log.pop("addInRules_External2Stored_Assignments")
  319. }
  320. def addOutRules_Internal2Stored_Assignments(Adaptation sa, HashMap<Port, SingleVarDeclaration> internalOutputPort2OutVarDeclaration) {
  321. Log.push("addOutRules_Internal2Stored_Assignments")
  322. if (sa.out === null){
  323. sa.out = SemanticAdaptationFactory.eINSTANCE.createOutRulesBlock()
  324. }
  325. addRules_Port2Stored_Assignments(sa.out.rules, internalOutputPort2OutVarDeclaration)
  326. Log.pop("addOutRules_Internal2Stored_Assignments")
  327. }
  328. def addRules_Port2Stored_Assignments(List<DataRule> rules, HashMap<Port, SingleVarDeclaration> port2VarDeclaration) {
  329. Log.push("addRules_External2Stored_Assignments")
  330. val dataRule = getOrPrependTrueRule(rules)
  331. if (dataRule.statetransitionfunction.expression !== null){
  332. throw new Exception("Expressions in rules are not supported yet.")
  333. // This and the one below are asily solved with a syntactic sugar substitution.
  334. }
  335. if (dataRule.statetransitionfunction.assignment !== null){
  336. throw new Exception("Assignment in rules are not supported yet.")
  337. }
  338. for(port : port2VarDeclaration.keySet){
  339. val storedVarDecl = port2VarDeclaration.get(port)
  340. addAssignmentToStoredVar(dataRule.statetransitionfunction.statements, port, storedVarDecl)
  341. }
  342. Log.pop("addRules_External2Stored_Assignments")
  343. }
  344. def addAssignmentToStoredVar(List<Statement> statements, Port internalPort, SingleVarDeclaration storedVarDecl) {
  345. Log.push("addAssignmentToStoredVar")
  346. val assignment = SemanticAdaptationFactory.eINSTANCE.createAssignment()
  347. assignment.lvalue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  348. assignment.lvalue.ref = storedVarDecl
  349. assignment.expr = SemanticAdaptationFactory.eINSTANCE.createVariable()
  350. (assignment.expr as Variable).owner = internalPort.eContainer as FMU
  351. (assignment.expr as Variable).ref = internalPort
  352. statements.add(0, assignment)
  353. Log.println("Assignment " + storedVarDecl.name + " := " + internalPort.qualifiedName + " created.")
  354. Log.pop("addAssignmentToStoredVar")
  355. }
  356. def getOrPrependTrueRule(List<DataRule> rules) {
  357. var DataRule rule = null
  358. if (rules.size == 0 || !isTrueRule(rules.head)){
  359. Log.println("No existing rule found with true condition. Creating one.")
  360. val trueRule = SemanticAdaptationFactory.eINSTANCE.createDataRule()
  361. trueRule.condition = SemanticAdaptationFactory.eINSTANCE.createRuleCondition()
  362. val trueExpr = SemanticAdaptationFactory.eINSTANCE.createBoolLiteral()
  363. trueExpr.value = "true"
  364. trueRule.condition.condition = trueExpr
  365. trueRule.statetransitionfunction = SemanticAdaptationFactory.eINSTANCE.createStateTransitionFunction()
  366. trueRule.outputfunction = SemanticAdaptationFactory.eINSTANCE.createCompositeOutputFunction()
  367. rules.add(0, trueRule)
  368. rule = trueRule
  369. } else {
  370. Log.println("Existing rule with true condition found.")
  371. rule = rules.head
  372. }
  373. return rule
  374. }
  375. def isTrueRule(DataRule rule){
  376. if (rule.condition.condition instanceof BoolLiteral){
  377. return (rule.condition.condition as BoolLiteral).value == "true"
  378. }
  379. return false
  380. }
  381. def addInVars(Adaptation sa, Map<Port, SingleParamDeclaration> inputPort2parameterDeclaration){
  382. Log.push("addInVars")
  383. if (sa.in === null){
  384. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  385. }
  386. val inputPort2InVarDeclaration = addStorageVars(sa.in.globalInVars, inputPort2parameterDeclaration)
  387. Log.pop("addInVars")
  388. return inputPort2InVarDeclaration
  389. }
  390. def addOutVars(Adaptation sa, Map<Port, SingleParamDeclaration> outputPort2parameterDeclaration){
  391. Log.push("addOutVars")
  392. if (sa.out === null){
  393. sa.out = SemanticAdaptationFactory.eINSTANCE.createOutRulesBlock()
  394. }
  395. val outputPort2InVarDeclaration = addStorageVars(sa.out.globalOutVars, outputPort2parameterDeclaration)
  396. Log.pop("addOutVars")
  397. return outputPort2InVarDeclaration
  398. }
  399. def addStorageVars(List<Declaration> varDeclarations, Map<Port, SingleParamDeclaration> port2parameterDeclaration){
  400. Log.push("addStorageVars")
  401. var port2VarDeclaration = new HashMap<Port, SingleVarDeclaration>()
  402. for(port : port2parameterDeclaration.keySet){
  403. Log.println("Processing port " + port.qualifiedName)
  404. val paramDecl = port2parameterDeclaration.get(port)
  405. val varDeclarationName = getStorageVarDeclarationName(port)
  406. if (!varDeclarationExists(varDeclarationName, varDeclarations) ){
  407. Log.println("Creating new variable declaration " + varDeclarationName)
  408. val varDeclaration = addNewInputVarDeclaration(port, paramDecl, varDeclarations)
  409. port2VarDeclaration.put(port, varDeclaration)
  410. } else {
  411. Log.println("Input variable declaration " + varDeclarationName + " already exists.")
  412. }
  413. }
  414. Log.pop("addStorageVars")
  415. return port2VarDeclaration
  416. }
  417. def addNewInputVarDeclaration(Port externalInputPort, SingleParamDeclaration paramDecl, List<Declaration> varDeclarations) {
  418. /*
  419. if (sa.in === null){
  420. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  421. }
  422. */
  423. if (varDeclarations.size == 0){
  424. varDeclarations.add(SemanticAdaptationFactory.eINSTANCE.createDeclaration())
  425. }
  426. val newSingleVarDecl = SemanticAdaptationFactory.eINSTANCE.createSingleVarDeclaration()
  427. newSingleVarDecl.name = getStorageVarDeclarationName(externalInputPort)
  428. newSingleVarDecl.type = externalInputPort.type
  429. val initValue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  430. initValue.ref = paramDecl
  431. newSingleVarDecl.expr = initValue
  432. varDeclarations.head.declarations.add(newSingleVarDecl)
  433. Log.println("New variable declaration created: " + newSingleVarDecl.name + " := " + paramDecl.name)
  434. return newSingleVarDecl
  435. }
  436. def varDeclarationExists(String invarName, List<Declaration> varDeclarations) {
  437. for (declarations : varDeclarations){ // sa.in.globalInVars
  438. for (decl : declarations.declarations){
  439. if (decl.name == invarName){
  440. return true
  441. }
  442. }
  443. }
  444. return false
  445. }
  446. def getStorageVarDeclarationName(Port externalInputPort) {
  447. return "stored__" + externalInputPort.name;
  448. }
  449. def genericDeclarationInferenceAlgorithm(Adaptation sa,
  450. (EObject)=>Object getField,
  451. (EObject, Object)=>void setField,
  452. (EObject)=>Object inferField
  453. ){
  454. Log.push("Running generic inference algorithm...")
  455. /*
  456. * Dumbest (and simplest) algorithm for this is a fixed point computation:
  457. * 1. Look for every var/port declaration
  458. * 2. If that var has a XXX already, nothing else to be done.
  459. * 3. If that var has no XXX declared, then
  460. * 3.1 If var/port has an initial value or connection, then
  461. * 3.1.1 If the initial_value/connection has a XXX declared, then var gets that XXX.
  462. * 3.1.2 Otherwise, nothing else to be done.
  463. * 3.2 If var/port has no initial value or connection then this either is a missing feature, or an error.
  464. * 3.3 If something has changed, go to 1. Otherwise, end.
  465. *
  466. * An extra set of instructions is there to push the element field information using connections and bindings.
  467. */
  468. var fixedPoint = false
  469. var unfieldedElementsCounter = 0
  470. while (! fixedPoint){
  471. fixedPoint = true
  472. unfieldedElementsCounter = 0
  473. Log.println("Inferring parameter fields...")
  474. for (paramDeclarations : sa.params) {
  475. for (paramDeclaration : paramDeclarations.declarations) {
  476. Log.println("Computing field for param " + paramDeclaration.name)
  477. if(getField.apply(paramDeclaration) !== null){
  478. Log.println("Already has been inferred: " + getField.apply(paramDeclaration))
  479. } else {
  480. Log.println("Has not been inferred yet.")
  481. if (tryInferAndAssignField(paramDeclaration, getField, setField, inferField)){
  482. fixedPoint = false
  483. } else {
  484. unfieldedElementsCounter++
  485. }
  486. }
  487. }
  488. }
  489. if(sa.inner !== null){
  490. if(sa.inner instanceof InnerFMUDeclarationFull){
  491. var innerFMUFull = sa.inner as InnerFMUDeclarationFull
  492. for(fmu : innerFMUFull.fmus){
  493. Log.println("Inferring port fields of FMU " + fmu.name)
  494. for (port : EcoreUtil2.getAllContentsOfType(fmu, Port)) {
  495. if(getField.apply(port) !== null){
  496. Log.println("Already has a field: " + getField.apply(port))
  497. } else {
  498. if (tryInferAndAssignField(port, getField, setField, inferField)){
  499. fixedPoint = false
  500. } else {
  501. unfieldedElementsCounter++
  502. }
  503. }
  504. }
  505. }
  506. if (innerFMUFull.connection.size > 0){
  507. Log.println("Inferring port fields using internal scenario bindings.")
  508. for (binding : innerFMUFull.connection){
  509. if (getField.apply(binding.src.port) !== null && getField.apply(binding.tgt.port) !== null){
  510. Log.println("Both ports have fields already.")
  511. } else {
  512. var inferredFieldAttempt = inferPortFieldViaConnection(binding, getField, setField, inferField)
  513. if (inferredFieldAttempt !== null){
  514. if (getField.apply(binding.src.port) === null){
  515. setField.apply(binding.src.port, inferredFieldAttempt)
  516. } else if (getField.apply(binding.tgt.port) === null){
  517. setField.apply(binding.tgt.port, inferredFieldAttempt)
  518. }
  519. fixedPoint = false
  520. unfieldedElementsCounter--
  521. Log.println("Got new field: " + inferredFieldAttempt)
  522. } else {
  523. Log.println("Cannot infer field from binding now.")
  524. }
  525. }
  526. }
  527. }
  528. } else {
  529. throw new Exception("Field inference only supported for InnerFMUDeclarationFull.")
  530. }
  531. }
  532. Log.println("Inferring external port fields...")
  533. var externalPorts = new LinkedList(sa.inports)
  534. externalPorts.addAll(sa.outports)
  535. for (port : externalPorts) {
  536. if (getField.apply(port) !== null){
  537. Log.println("Already has a field: " + getField.apply(port))
  538. if (pushPortField(port, getField, setField, inferField)){
  539. fixedPoint = false
  540. unfieldedElementsCounter--
  541. }
  542. } else {
  543. if (tryInferAndAssignField(port, getField, setField, inferField)){
  544. fixedPoint = false
  545. } else {
  546. unfieldedElementsCounter++
  547. }
  548. }
  549. }
  550. Log.println("Inferring all other declaration fields...")
  551. for (varDeclaration : EcoreUtil2.getAllContentsOfType(sa, SingleVarDeclaration)) {
  552. Log.println("Computing field for declaration " + varDeclaration.name)
  553. if(getField.apply(varDeclaration) !== null){
  554. Log.println("Already has a field: " + getField.apply(varDeclaration))
  555. } else {
  556. if (tryInferAndAssignField(varDeclaration, getField, setField, inferField)){
  557. fixedPoint = false
  558. } else {
  559. unfieldedElementsCounter++
  560. }
  561. }
  562. }
  563. Log.println("Ended iteration with unfielded elements remaining: " + unfieldedElementsCounter)
  564. } // while (! fixedPoint)
  565. Log.pop("Running generic inference algorithm... DONE")
  566. return unfieldedElementsCounter
  567. }
  568. def tryInferAndAssignField(EObject element,
  569. (EObject)=>Object getField,
  570. (EObject, Object)=>void setField,
  571. (EObject)=>Object inferField) {
  572. var inferredFieldAttempt = inferField.apply(element)
  573. if (inferredFieldAttempt !== null){
  574. setField.apply(element, inferredFieldAttempt)
  575. Log.println("Got new field: " + inferredFieldAttempt)
  576. return true
  577. } else {
  578. Log.println("Cannot infer field now.")
  579. return false
  580. }
  581. }
  582. def extractTypeFromExpression(Expression expression, String declarationName){
  583. if (expression instanceof IntLiteral){
  584. return "Integer"
  585. } else if (expression instanceof RealLiteral){
  586. return "Real"
  587. } else if (expression instanceof BoolLiteral){
  588. return "Bool"
  589. } else if (expression instanceof StringLiteral){
  590. return "String"
  591. } else if (expression instanceof Variable){
  592. var varRef = expression as Variable
  593. if (varRef.ref instanceof Port){
  594. var decl = varRef.ref as Port
  595. if (decl.type !== null){
  596. return decl.type
  597. }
  598. } else if(varRef.ref instanceof SingleParamDeclaration){
  599. var decl = varRef.ref as SingleParamDeclaration
  600. if (decl.type !== null){
  601. return decl.type
  602. }
  603. } else if(varRef.ref instanceof SingleVarDeclaration){
  604. var decl = varRef.ref as SingleVarDeclaration
  605. if (decl.type !== null){
  606. return decl.type
  607. }
  608. } else if(varRef.ref instanceof DeclaredParameter){
  609. throw new Exception("Type cannot be inferred for references to DeclaredParameter (for now). Please specify the explicit type of declaration " + declarationName)
  610. } else {
  611. throw new Exception("Unexpected kind of Variable expression found.")
  612. }
  613. } else if(expression instanceof BuiltinFunction){
  614. if (expression instanceof IsSet || expression instanceof Close){
  615. return "Bool"
  616. } else {
  617. return "Real"
  618. }
  619. } else {
  620. throw new Exception("Initial value for declaration " + declarationName + " must be literal or var ref for now. Got instead " + expression + ". If you want complex expressions, give it an explicit type.")
  621. }
  622. return null
  623. }
  624. def inferPortFieldViaConnection(Connection binding,
  625. (EObject)=>Object getField,
  626. (EObject, Object)=>void setField,
  627. (EObject)=>Object inferField
  628. ){
  629. var Object resultField = null
  630. if (getField.apply(binding.src.port) !== null && getField.apply(binding.tgt.port) !== null){
  631. throw new Exception("Wrong way of using this function. It assumes type is not inferred yet.")
  632. } else if (getField.apply(binding.src.port) !== null){
  633. resultField = getField.apply(binding.src.port)
  634. Log.println("Target port "+ binding.tgt.port.qualifiedName +" got new type: " + resultField)
  635. } else if (getField.apply(binding.tgt.port) !== null){
  636. resultField = getField.apply(binding.tgt.port)
  637. Log.println("Target port "+ binding.src.port.qualifiedName +" got new type: " + resultField)
  638. }
  639. return resultField
  640. }
  641. def pushPortField(Port port,
  642. (EObject)=>Object getField,
  643. (EObject, Object)=>void setField,
  644. (EObject)=>Object inferField){
  645. var fieldInferred = false
  646. Log.println("Pushing field of port " + port.qualifiedName + " to its bindings.")
  647. if(getField.apply(port) === null){
  648. Log.println("Has no field to be pushed.")
  649. throw new Exception("Wrong way of using this function. It assumes field is already inferred.")
  650. } else {
  651. Log.println("Pushing field: " + getField.apply(port))
  652. if(port.sourcedependency !== null){
  653. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  654. if(getField.apply(port.sourcedependency.port) === null){
  655. setField.apply(port.sourcedependency.port, getField.apply(port))
  656. Log.println("Port " + port.sourcedependency.port.qualifiedName + " got new type: " + getField.apply(port.sourcedependency.port))
  657. fieldInferred = true
  658. } else {
  659. Log.println("Source port already has field.")
  660. }
  661. } else {
  662. Log.println("Has no source dependency.")
  663. }
  664. if (port.targetdependency !== null) {
  665. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  666. if(getField.apply(port.targetdependency.port) === null){
  667. Log.println("Dependency has no field yet.")
  668. setField.apply(port.targetdependency.port, getField.apply(port))
  669. Log.println("Port " + port.targetdependency.port.qualifiedName + " got new type: " + getField.apply(port.targetdependency.port))
  670. fieldInferred = true
  671. } else {
  672. Log.println("Target port already has field.")
  673. }
  674. } else {
  675. Log.println("Has no target dependency.")
  676. }
  677. }
  678. return fieldInferred
  679. }
  680. def getPortUnit(Port port){
  681. var unitInferred = false
  682. Log.println("Computing unit for port " + port.qualifiedName)
  683. var Unity returnUnit = null
  684. if(port.unity !== null){
  685. throw new Exception("Wrong way of using this function. It assumes unit is not inferred yet.")
  686. } else {
  687. Log.println("Has no unit.")
  688. Log.println("Attempting to infer unit from bindings.")
  689. if(port.sourcedependency !== null){
  690. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  691. if(port.sourcedependency.port.unity === null){
  692. Log.println("Dependency has no unit yet.")
  693. } else {
  694. returnUnit = port.sourcedependency.port.unity
  695. Log.println("Got new unit: " + returnUnit)
  696. unitInferred = true
  697. }
  698. } else {
  699. Log.println("Has no source dependency.")
  700. }
  701. if (port.targetdependency !== null && !unitInferred) {
  702. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  703. if(port.targetdependency.port.unity === null){
  704. Log.println("Dependency has no unit yet.")
  705. } else {
  706. returnUnit = port.targetdependency.port.unity
  707. Log.println("Got new unit: " + returnUnit)
  708. unitInferred = true
  709. }
  710. } else {
  711. Log.println("Has no target dependency, or unit has already been inferred from source dependency.")
  712. }
  713. }
  714. return returnUnit
  715. }
  716. def getPortType(Port port){
  717. var typeInferred = false
  718. Log.println("Computing type for port " + port.qualifiedName)
  719. var String returnType = null
  720. if(port.type !== null){
  721. throw new Exception("Wrong way of using this function. It assumes type is not inferred yet.")
  722. } else {
  723. Log.println("Has no type.")
  724. Log.println("Attempting to infer type from units.")
  725. if (port.unity !== null){
  726. returnType = "Real"
  727. Log.println("Got new type: " + returnType)
  728. typeInferred = true
  729. } else {
  730. Log.println("Attempting to infer type from bindings.")
  731. if(port.sourcedependency !== null){
  732. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  733. if(port.sourcedependency.port.type === null){
  734. Log.println("Dependency has no type yet.")
  735. } else {
  736. returnType = port.sourcedependency.port.type
  737. Log.println("Got new type: " + returnType)
  738. typeInferred = true
  739. }
  740. } else {
  741. Log.println("Has no source dependency.")
  742. }
  743. if (port.targetdependency !== null && !typeInferred) {
  744. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  745. if(port.targetdependency.port.type === null){
  746. //println("Port object: " + port.targetdependency.port)
  747. Log.println("Dependency has no type yet.")
  748. } else {
  749. returnType = port.targetdependency.port.type
  750. Log.println("Got new type: " + returnType)
  751. typeInferred = true
  752. }
  753. } else {
  754. Log.println("Has no target dependency, or type has already been inferred from source dependency.")
  755. }
  756. }
  757. }
  758. return returnType
  759. }
  760. def addInPorts(Adaptation sa) {
  761. Log.push("Adding input ports...")
  762. for (port : getAllInnerFMUInputPortDeclarations(sa)){
  763. var parentFMU = port.eContainer as InnerFMU
  764. Log.println("Checking if port " + port.qualifiedName + " has incoming connections"
  765. )
  766. if (! hasConnection(port, sa, true)){
  767. Log.println("Port " + port.qualifiedName + " has no incoming connections.")
  768. val externalPortName = createExternalPortNameFromInternalPort(parentFMU.name, port.name)
  769. if (findExternalPortByName(sa, externalPortName) === null){
  770. var newExternalPort = createExternalInputPortDeclarationFromInnerPort(port, parentFMU, sa)
  771. Log.println("External port " + newExternalPort.qualifiedName + " created.")
  772. newExternalPort.bindExternalInputPortTo(parentFMU, port)
  773. Log.println("External port " + newExternalPort.qualifiedName + " bound to port " + port.qualifiedName)
  774. } else {
  775. Log.println("Error: External port " + externalPortName + " already declared.")
  776. throw new Exception("Error: External port " + externalPortName + " already declared. Please rename it to avoid clashes.")
  777. }
  778. } else {
  779. Log.println("Port " + port.qualifiedName + " has an incoming connection.")
  780. }
  781. }
  782. Log.pop("Adding input ports... DONE")
  783. }
  784. def bindExternalInputPortTo(Port externalInputPort, InnerFMU internalPortParent, Port internalPort) {
  785. externalInputPort.targetdependency = SemanticAdaptationFactory.eINSTANCE.createSpecifiedPort()
  786. externalInputPort.targetdependency.owner = internalPortParent
  787. externalInputPort.targetdependency.port = internalPort
  788. }
  789. def bindExternalOutputPortTo(Port externalOutputPort, InnerFMU internalPortParent, Port internalPort) {
  790. externalOutputPort.sourcedependency = SemanticAdaptationFactory.eINSTANCE.createSpecifiedPort()
  791. externalOutputPort.sourcedependency.owner = internalPortParent
  792. externalOutputPort.sourcedependency.port = internalPort
  793. }
  794. def createExternalInputPortDeclarationFromInnerPort(Port port, FMU parent, Adaptation sa) {
  795. var externalInputPort = createExternalPortDeclarationFromInnerPort(port, parent)
  796. sa.inports.add(externalInputPort)
  797. return externalInputPort
  798. }
  799. def createExternalOutputPortDeclarationFromInnerPort(Port port, FMU parent, Adaptation sa) {
  800. var externalOutputPort = createExternalPortDeclarationFromInnerPort(port, parent)
  801. sa.outports.add(externalOutputPort)
  802. return externalOutputPort
  803. }
  804. def createExternalPortDeclarationFromInnerPort(Port port, FMU parent) {
  805. var externalInputPort = SemanticAdaptationFactory.eINSTANCE.createPort()
  806. externalInputPort.name = createExternalPortNameFromInternalPort(parent.name, port.name)
  807. externalInputPort.type = port.type
  808. externalInputPort.unity = EcoreUtil2.copy(port.unity)
  809. return externalInputPort
  810. }
  811. def findExternalPortByName(Adaptation adaptation, String name) {
  812. for (externalInputPort : adaptation.inports){
  813. if (externalInputPort.name == name){
  814. return externalInputPort
  815. }
  816. }
  817. return null
  818. }
  819. def findExternalPortByTargetDependency(List<Port> ports, Port targetDependency) {
  820. for (externalInputPort : ports){
  821. if (externalInputPort.targetdependency !== null && externalInputPort.targetdependency.port == targetDependency){
  822. return externalInputPort
  823. }
  824. }
  825. return null
  826. }
  827. def findExternalPortBySourceDependency(List<Port> ports, Port sourceDependency) {
  828. for (externalInputPort : ports){
  829. if (externalInputPort.sourcedependency !== null && externalInputPort.sourcedependency.port == sourceDependency){
  830. return externalInputPort
  831. }
  832. }
  833. return null
  834. }
  835. def hasConnection(Port port, Adaptation adaptation, Boolean checkForIncomming) {
  836. var result = false
  837. if ( (checkForIncomming && port.sourcedependency !== null) ||
  838. (! checkForIncomming && port.targetdependency !== null)
  839. ){
  840. result = true
  841. } else {
  842. if (port.eContainer instanceof InnerFMU){
  843. var innerScenarioDeclaration = EcoreUtil2.getContainerOfType(port, InnerFMUDeclaration)
  844. if (innerScenarioDeclaration instanceof InnerFMUDeclarationFull){
  845. var innerScenarioWithCoupling = innerScenarioDeclaration as InnerFMUDeclarationFull
  846. if (innerScenarioWithCoupling.connection.size > 0){
  847. for (connection : innerScenarioWithCoupling.connection ){
  848. if ( (checkForIncomming && connection.tgt.port == port)){
  849. Log.println("Port " + port.qualifiedName + " has an incoming connection from internal port " + connection.src.port.qualifiedName)
  850. result = true
  851. } else if (!checkForIncomming && connection.src.port == port) {
  852. Log.println("Port " + port.qualifiedName + " has an outgoing connection to internal port " + connection.tgt.port.qualifiedName)
  853. result = true
  854. }
  855. }
  856. }
  857. }
  858. for (externalInputPort : adaptation.inports.filter[p | (checkForIncomming && p.targetdependency !== null) || (!checkForIncomming && p.sourcedependency !== null) ]){
  859. if (checkForIncomming && externalInputPort.targetdependency.port == port){
  860. Log.println("Port " + port.qualifiedName + " has an incoming connection from external port " + externalInputPort.qualifiedName)
  861. result = true
  862. } else if ( !checkForIncomming && externalInputPort.sourcedependency.port == port){
  863. Log.println("Port " + port.qualifiedName + " has an outgoing connection to external port " + externalInputPort.qualifiedName)
  864. result = true
  865. }
  866. }
  867. }
  868. }
  869. return result
  870. }
  871. def getAllInnerFMUInputPortDeclarations(Adaptation sa){
  872. return mapAllInnerFMUs(sa, [fmu | fmu.inports]);
  873. }
  874. def getAllInnerFMUOutputPortDeclarations(Adaptation sa){
  875. return mapAllInnerFMUs(sa, [fmu | fmu.outports]);
  876. }
  877. def <T> List<T> mapAllInnerFMUs(Adaptation sa, (InnerFMU)=>List<T> map){
  878. var result = new LinkedList()
  879. if(sa.inner !== null){
  880. if(sa.inner instanceof InnerFMUDeclarationFull){
  881. var innerFMUFull = sa.inner as InnerFMUDeclarationFull
  882. for(fmu : innerFMUFull.fmus){
  883. result.addAll(map.apply(fmu))
  884. }
  885. } else {
  886. throw new Exception("Only support for InnerFMUDeclarationFull.")
  887. }
  888. }
  889. return result;
  890. }
  891. def addInParams(Adaptation sa) {
  892. Log.push("Adding input parameters...")
  893. val result = addParamForExternalPortDeclarations(sa, sa.inports)
  894. Log.pop("Adding input parameters... DONE")
  895. return result
  896. }
  897. def addOutParams(Adaptation sa) {
  898. Log.push("Adding output parameters...")
  899. val result = addParamForExternalPortDeclarations(sa, sa.outports)
  900. Log.pop("Adding output parameters... DONE")
  901. return result
  902. }
  903. def addParamForExternalPortDeclarations(Adaptation sa, List<Port> externalPortList){
  904. Log.push("addParamForExternalPortDeclarations")
  905. val PARAM_PREFIX = "INIT_"
  906. var externalPort2parameterDeclaration = new HashMap<Port, SingleParamDeclaration>(externalPortList.size)
  907. for (externalPortDecl : externalPortList) {
  908. Log.println("Generating parameter for port " + externalPortDecl.qualifiedName)
  909. var paramname = PARAM_PREFIX + externalPortDecl.name.toUpperCase()
  910. if (paramAlreadyDeclared(paramname, sa)){
  911. Log.println("Parameter " + paramname + " already declared for port " + externalPortDecl.qualifiedName)
  912. } else {
  913. Log.println("Declaring new parameter " + paramname + " for port " + externalPortDecl.qualifiedName)
  914. var paramDeclaration = addNewParamDeclaration(paramname, externalPortDecl, sa)
  915. externalPort2parameterDeclaration.put(externalPortDecl, paramDeclaration)
  916. }
  917. }
  918. Log.pop("addParamForExternalPortDeclarations")
  919. return externalPort2parameterDeclaration
  920. }
  921. def addNewParamDeclaration(String name, Port fromPort, Adaptation sa) {
  922. var factory = SemanticAdaptationFactory.eINSTANCE
  923. var paramDeclaration = factory.createSingleParamDeclaration()
  924. paramDeclaration.name = name
  925. paramDeclaration.type = fromPort.type
  926. paramDeclaration.expr = getDefaultTypeExpression(paramDeclaration.type)
  927. if (sa.params.size == 0){
  928. sa.params.add(factory.createParamDeclarations())
  929. }
  930. sa.params.head.declarations.add(paramDeclaration)
  931. return paramDeclaration
  932. }
  933. def getDefaultTypeExpression(String type) {
  934. switch (type) {
  935. case "Integer": {
  936. val result = SemanticAdaptationFactory.eINSTANCE.createIntLiteral
  937. result.value = 0
  938. return result
  939. }
  940. case "Real": {
  941. val result = SemanticAdaptationFactory.eINSTANCE.createRealLiteral
  942. result.value = 0.0f
  943. return result
  944. }
  945. case "Bool": {
  946. val result = SemanticAdaptationFactory.eINSTANCE.createBoolLiteral
  947. result.value = "false"
  948. return result
  949. }
  950. case "String": {
  951. val result = SemanticAdaptationFactory.eINSTANCE.createStringLiteral
  952. result.value = " "
  953. return result
  954. }
  955. default: {
  956. throw new Exception("Unexpected type.")
  957. }
  958. }
  959. }
  960. def paramAlreadyDeclared(String name, Adaptation sa) {
  961. for(paramDeclarations : sa.params){
  962. for(paramDeclaration : paramDeclarations.declarations){
  963. if(paramDeclaration.name == name){
  964. return true
  965. }
  966. }
  967. }
  968. return false
  969. }
  970. def addOutPorts(Adaptation sa) {
  971. Log.push("Adding output ports...")
  972. for (port : getAllInnerFMUOutputPortDeclarations(sa)){
  973. var parentFMU = port.eContainer as InnerFMU
  974. Log.println("Checking if port " + port.qualifiedName + " has outgoing connections")
  975. if (! hasConnection(port, sa, false)){
  976. Log.println("Port " + port.qualifiedName + " has no outgoing connections.")
  977. val externalPortName = createExternalPortNameFromInternalPort(parentFMU.name, port.name)
  978. if (findExternalPortByName(sa, externalPortName) === null){
  979. var newExternalPort = createExternalOutputPortDeclarationFromInnerPort(port, parentFMU, sa)
  980. Log.println("External port " + newExternalPort.qualifiedName + " created.")
  981. newExternalPort.bindExternalOutputPortTo(parentFMU, port)
  982. Log.println("External port " + newExternalPort.qualifiedName + " bound to port " + port.qualifiedName)
  983. } else {
  984. Log.println("Error: External port " + externalPortName + " already declared.")
  985. throw new Exception("Error: External port " + externalPortName + " already declared. Please rename it to avoid clashes.")
  986. }
  987. } else {
  988. Log.println("Port " + port.qualifiedName + " has an incoming connection.")
  989. }
  990. }
  991. Log.pop("Adding output ports... DONE")
  992. }
  993. }