SemanticAdaptationCanonicalGenerator.xtend 55 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527
  1. /*
  2. * generated by Xtext 2.10.0
  3. */
  4. package be.uantwerpen.ansymo.semanticadaptation.cg.canonical
  5. import be.uantwerpen.ansymo.semanticadaptation.cg.canonical.graph.DirectedGraph
  6. import be.uantwerpen.ansymo.semanticadaptation.cg.canonical.graph.FMUGraph
  7. import be.uantwerpen.ansymo.semanticadaptation.cg.canonical.graph.TopologicalSort
  8. import be.uantwerpen.ansymo.semanticadaptation.log.Log
  9. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Adaptation
  10. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Assignment
  11. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.AtomicUnity
  12. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.BoolLiteral
  13. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.BuiltinFunction
  14. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Close
  15. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.CompositeOutputFunction
  16. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Connection
  17. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.CustomControlRule
  18. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DataRule
  19. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Declaration
  20. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DeclaredParameter
  21. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DivideUnity
  22. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DoStep
  23. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.DoStepFun
  24. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Expression
  25. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.FMU
  26. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMU
  27. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMUDeclaration
  28. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.InnerFMUDeclarationFull
  29. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.IntLiteral
  30. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.IsSet
  31. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.MooreOrMealy
  32. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.MultiplyUnity
  33. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Port
  34. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.ReactiveOrDelayed
  35. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.RealLiteral
  36. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SemanticAdaptationFactory
  37. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SingleParamDeclaration
  38. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.SingleVarDeclaration
  39. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Statement
  40. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.StringLiteral
  41. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Unity
  42. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.Variable
  43. import java.io.ByteArrayOutputStream
  44. import java.util.HashMap
  45. import java.util.LinkedList
  46. import java.util.List
  47. import java.util.Map
  48. import org.eclipse.emf.common.util.EList
  49. import org.eclipse.emf.common.util.URI
  50. import org.eclipse.emf.ecore.EObject
  51. import org.eclipse.xtext.EcoreUtil2
  52. import org.eclipse.xtext.generator.IFileSystemAccess2
  53. import be.uantwerpen.ansymo.semanticadaptation.semanticAdaptation.CurrentTime
  54. /**
  55. * Generates code from your model files on save.
  56. *
  57. * See https://www.eclipse.org/Xtext/documentation/303_runtime_concepts.html#code-generation
  58. */
  59. class SemanticAdaptationCanonicalGenerator {
  60. String CANONICAL_SUFIX = "_canonical"
  61. def Adaptation doGenerate(Adaptation adaptation, IFileSystemAccess2 fsa, URI mainFile) {
  62. Log.push("Generating canonical semantic adaptation for file...")
  63. Log.println(prettyprint_model(adaptation, "File Read"))
  64. // Create file name for the canonical sa file
  65. var fileNameWithoutExt = mainFile.trimFileExtension().lastSegment()
  66. Log.println("Checking if file is already a canonical version...")
  67. if (fileNameWithoutExt.indexOf(CANONICAL_SUFIX) == -1){
  68. Log.println("It is not.")
  69. var canonicalFileName = fileNameWithoutExt + CANONICAL_SUFIX + ".sa"
  70. Log.println("canonicalFileName = " + canonicalFileName)
  71. canonicalize(adaptation)
  72. Log.println(prettyprint_model(adaptation, "Generated File"))
  73. fsa.generateFile(canonicalFileName, adaptation.serialize_model)
  74. Log.println("File " + canonicalFileName + " written.")
  75. } else {
  76. Log.println("It is already a canonical version.")
  77. Log.println("Nothing to do.")
  78. }
  79. Log.pop("Generating canonical semantic adaptation for file... DONE")
  80. return adaptation
  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. removeBindings(internalPort2ExternalPortBindings, sa)
  201. if (sa.outports.size==0){
  202. addOutPorts(sa)
  203. }
  204. val outputPort2parameterDeclaration = addOutParams(sa)
  205. val internalOutputPort2OutVarDeclaration = addOutVars(sa, outputPort2parameterDeclaration)
  206. val internalOutputPort2ExternalPortBindings = findAllInternalPort2ExternalOutputPort_Bindings(sa)
  207. //val internalOutputPort2OutVarDeclaration = transitiveStep(internalOutputPort2ExternalPortBindings, internalOutputPort2OutVarDeclaration)
  208. addOutRules_Internal2Stored_Assignments(sa, internalOutputPort2OutVarDeclaration)
  209. addOutRules_Internal2External_Assignments(sa, internalOutputPort2ExternalPortBindings)
  210. removeBindings(internalOutputPort2ExternalPortBindings, sa)
  211. if (sa.control === null){
  212. sa.control = SemanticAdaptationFactory.eINSTANCE.createControlRuleBlock()
  213. sa.control.rule = SemanticAdaptationFactory.eINSTANCE.createCustomControlRule()
  214. createCoSimStepInstructions(sa)
  215. }
  216. createInternalBindingAssignments(sa)
  217. Log.push("Replace port refs in input rules")
  218. for (rule : sa.in.rules){
  219. check(rule.outputfunction instanceof CompositeOutputFunction, "Only CompositeOutputFunction are supported in DataRules.")
  220. replacePortRefsByVarDecl((rule.outputfunction as CompositeOutputFunction).statements, externalInputPort2InVarDeclaration)
  221. }
  222. Log.pop("Replace port refs in input rules")
  223. Log.push("Replace port refs in control rule")
  224. if (sa.control.rule instanceof CustomControlRule){
  225. replacePortRefsByVarDecl((sa.control.rule as CustomControlRule).controlRulestatements,internalOutputPort2OutVarDeclaration)
  226. }
  227. Log.pop("Replace port refs in control rule")
  228. Log.push("Replace port refs in output rule")
  229. for (rule : sa.out.rules){
  230. check(rule.outputfunction instanceof CompositeOutputFunction, "Only CompositeOutputFunction are supported in DataRules.")
  231. replacePortRefsByVarDecl((rule.outputfunction as CompositeOutputFunction).statements, internalOutputPort2OutVarDeclaration)
  232. }
  233. Log.pop("Replace port refs in output rule")
  234. Log.pop("Canonicalize")
  235. }
  236. def replacePortRefsByVarDecl(EList<Statement> statements, HashMap<Port, SingleVarDeclaration> port2VarDecl) {
  237. Log.push("replacePortRefsByVarDecl")
  238. for(statement : statements){
  239. val vars = statement.eAllContents.filter[v | v instanceof Variable]
  240. var Variable v
  241. while (vars.hasNext) {
  242. v = vars.next as Variable
  243. if (port2VarDecl.containsKey(v.ref)){
  244. val port = v.ref as Port
  245. v.owner = null
  246. val varDecl = port2VarDecl.get(v.ref)
  247. v.ref = varDecl
  248. Log.println("Replaced ref to " + port.qualifiedName + " by " + varDecl.name)
  249. } else {
  250. Log.println("Var ref not substituted: " + v.ref)
  251. }
  252. }
  253. }
  254. Log.pop("replacePortRefsByVarDecl")
  255. }
  256. def createInternalBindingAssignments(Adaptation sa) {
  257. Log.push("createInternalBindingAssignments")
  258. val scenario = (sa.inner as InnerFMUDeclarationFull)
  259. for (connection : scenario.connection){
  260. val trgFMU = connection.tgt.port.eContainer as InnerFMU
  261. val ctrlRule = sa.control.rule as CustomControlRule
  262. val doStepIndex = findDoStepStatementIndex(ctrlRule, trgFMU)
  263. if (doStepIndex < 0){
  264. throw new IllegalArgumentException("DoStep instruction for FMU " + trgFMU.name + " not found in control rule block.")
  265. }
  266. if (! existsAssignmentToPort_BeforeIndex(connection.tgt.port, doStepIndex, ctrlRule)){
  267. Log.println("Creating assignment to port " + connection.tgt.port.qualifiedName + " at position " + doStepIndex)
  268. addPortAssignment(ctrlRule.controlRulestatements, connection.tgt.port, connection.src.port, doStepIndex, true)
  269. } else {
  270. Log.println("There is already an assignment to port " + connection.tgt.port.qualifiedName + " before position " + doStepIndex)
  271. }
  272. }
  273. Log.pop("createInternalBindingAssignments")
  274. }
  275. def existsAssignmentToPort_BeforeIndex(Port port, int index, CustomControlRule rule) {
  276. val assignmentsToPort = rule.controlRulestatements.indexed
  277. .filter[s | s.value instanceof Assignment &&
  278. (s.value as Assignment).lvalue.ref == port
  279. ]
  280. check(assignmentsToPort.size <= 1, "Multiple assignments to the same port are not supported yet. Use a loop with a single call.")
  281. return assignmentsToPort.size == 1
  282. }
  283. def findDoStepStatementIndex(CustomControlRule rule, InnerFMU fmu) {
  284. Log.push("findDoStepStatementIndex")
  285. var result = -1
  286. val doStepProcedures = rule.controlRulestatements.indexed.filter[s | s.value instanceof DoStep && (s.value as DoStep).fmu == fmu]
  287. check(doStepProcedures.size <= 1, "Multiple calls to the doStep function for the same FMU are not supported yet. Use a loop with a single call.")
  288. if (doStepProcedures.size == 1){
  289. Log.println("Found a doStep procedure for fmu " + fmu.name + " at position " + doStepProcedures.head.key)
  290. result = doStepProcedures.head.key
  291. }
  292. // Warning: this does not support instructions such as:
  293. // var someVar = 10, h = doStep(f), etc...
  294. // Only single declarations are supported, such as:
  295. // var h = doStep(f)
  296. // FIXME: This needs to look into hirarchical statements (like for loops).
  297. val doStepAssignments = rule.controlRulestatements.indexed
  298. .filter[s | s.value instanceof Declaration &&
  299. (s.value as Declaration).declarations.size == 1 &&
  300. (s.value as Declaration).declarations.head instanceof SingleVarDeclaration &&
  301. (s.value as Declaration).declarations.head.expr instanceof DoStepFun &&
  302. ((s.value as Declaration).declarations.head.expr as DoStepFun).fmu == fmu
  303. ]
  304. check(doStepAssignments.size <= 1, "Multiple calls to the doStep function for the same FMU are not supported yet. Use a loop with a single call.")
  305. if (doStepAssignments.size == 1){
  306. check(result==-1, "Multiple calls to the doStep function for the same FMU are not supported yet. Use a loop with a single call.")
  307. Log.println("Found a doStep function for fmu " + fmu.name + " at position " + doStepAssignments.head.key)
  308. result = doStepAssignments.head.key
  309. }
  310. Log.pop("findDoStepStatementIndex")
  311. return result
  312. }
  313. def createCoSimStepInstructions(Adaptation sa) {
  314. check((sa.inner as InnerFMUDeclarationFull).fmus.size > 0, "At least one internal FMU is expected...")
  315. if (onlyOneInternalFMU(sa)){
  316. createCosimStepForOneFMU(sa)
  317. } else {
  318. createCosimStepForMultipleFMUs(sa)
  319. }
  320. }
  321. def createCosimStepForOneFMU(Adaptation sa) {
  322. Log.push("createCosimStepForOneFMU")
  323. val fmu = (sa.inner as InnerFMUDeclarationFull).fmus.head
  324. val controlRule = sa.control.rule as CustomControlRule
  325. val returnedStepVar = controlRule.appendDoStep(fmu)
  326. val stepVarSingleton = new LinkedList()
  327. stepVarSingleton.add(returnedStepVar)
  328. controlRule.appendReturnCosimStep(stepVarSingleton)
  329. Log.pop("createCosimStepForOneFMU")
  330. }
  331. def appendReturnCosimStep(CustomControlRule rule, LinkedList<SingleVarDeclaration> stepVariables) {
  332. Log.push("appendReturnCosimStep")
  333. if (stepVariables.size == 0){
  334. throw new Exception("Does not make sense. At least one inner FMU is expected.")
  335. }
  336. val minExpression = SemanticAdaptationFactory.eINSTANCE.createMin()
  337. Log.println("Creating min(...) expression with the following arguments:")
  338. for(varDecl : stepVariables){
  339. val varRef = SemanticAdaptationFactory.eINSTANCE.createVariable()
  340. varRef.ref = varDecl
  341. minExpression.args.add(varRef)
  342. Log.println(varDecl.name)
  343. }
  344. rule.returnstatement = SemanticAdaptationFactory.eINSTANCE.createReturnStatement()
  345. rule.returnstatement.expr = minExpression
  346. Log.pop("appendReturnCosimStep")
  347. }
  348. def appendDoStep(CustomControlRule rule, InnerFMU fmu) {
  349. Log.push("appendDoStep")
  350. val t = SemanticAdaptationFactory.eINSTANCE.createCurrentTime()
  351. val H = SemanticAdaptationFactory.eINSTANCE.createStepSize()
  352. val doStep = SemanticAdaptationFactory.eINSTANCE.createDoStepFun()
  353. doStep.t = t
  354. doStep.h = H
  355. doStep.fmu = fmu
  356. val step_var = SemanticAdaptationFactory.eINSTANCE.createSingleVarDeclaration()
  357. step_var.name = "H_" + fmu.name
  358. step_var.expr = doStep
  359. val step_decl = SemanticAdaptationFactory.eINSTANCE.createDeclaration()
  360. step_decl.declarations.add(step_var)
  361. rule.controlRulestatements.add(step_decl)
  362. Log.println("Added var " + step_var.name + " := doStep(t, H, " + fmu.name + ")")
  363. Log.pop("appendDoStep")
  364. return step_var
  365. }
  366. def createCosimStepForMultipleFMUs(Adaptation sa) {
  367. Log.push("createCosimStepForMultipleFMUs")
  368. val innerDeclaration = (sa.inner as InnerFMUDeclarationFull)
  369. val controlRule = sa.control.rule as CustomControlRule
  370. val stepVarSingleton = new LinkedList()
  371. for (fmu : innerDeclaration.topologicalSort()){
  372. val returnedStepVar = controlRule.appendDoStep(fmu)
  373. stepVarSingleton.add(returnedStepVar)
  374. }
  375. controlRule.appendReturnCosimStep(stepVarSingleton)
  376. Log.pop("createCosimStepForMultipleFMUs")
  377. }
  378. def topologicalSort(InnerFMUDeclarationFull scenario){
  379. Log.push("topologicalSort")
  380. val DirectedGraph<InnerFMU> inner_fmu_graph = createFMUGraph(scenario)
  381. val result = TopologicalSort.sort(inner_fmu_graph)
  382. Log.println("Sorting: " + result)
  383. Log.pop("topologicalSort")
  384. return result
  385. }
  386. def createFMUGraph(InnerFMUDeclarationFull scenario) {
  387. Log.push("createFMUGraph")
  388. val graph = new FMUGraph()
  389. for (fmu : scenario.fmus){
  390. graph.addNode(fmu)
  391. }
  392. for (connection : scenario.connection){
  393. check(connection.src.port.eContainer instanceof InnerFMU &&
  394. connection.tgt.port.eContainer instanceof InnerFMU, "Weird connection found: " + connection)
  395. if (reactiveMealyFMU(connection.tgt.port.eContainer as InnerFMU)){
  396. graph.addEdge(connection.src.port.eContainer as InnerFMU, connection.tgt.port.eContainer as InnerFMU)
  397. } else {
  398. Log.println("FMU " + (connection.tgt.port.eContainer as InnerFMU).name + " is not reactive mealy, so it has no algebraic dependencies.")
  399. }
  400. }
  401. Log.println(graph.toString())
  402. Log.pop("createFMUGraph")
  403. return graph
  404. }
  405. def reactiveMealyFMU(InnerFMU fmu) {
  406. return fmu.reactiveness == ReactiveOrDelayed.REACTIVE && fmu.machine == MooreOrMealy.MEALY
  407. }
  408. def check(Boolean condition, String msg){
  409. if (! condition){
  410. throw new Exception("Assertion error: " + msg)
  411. }
  412. }
  413. def onlyOneInternalFMU(Adaptation sa) {
  414. if(sa.inner instanceof InnerFMUDeclarationFull){
  415. return (sa.inner as InnerFMUDeclarationFull).fmus.size == 1
  416. } else {
  417. throw new Exception('This kind of internal scenario is not supported yet.')
  418. }
  419. }
  420. def transitiveStep(HashMap<Port, Port> internalOutputPort2ExternalPortBindings, HashMap<Port, SingleVarDeclaration> externalOutputPort2OutVarDeclaration) {
  421. Log.push("transitiveStep")
  422. val internalOutputPort2OutVarDeclaration = new HashMap<Port, SingleVarDeclaration>()
  423. for(internalOutputPort : internalOutputPort2ExternalPortBindings.keySet){
  424. val externalOutputPort = internalOutputPort2ExternalPortBindings.get(internalOutputPort)
  425. if (externalOutputPort2OutVarDeclaration.containsKey(externalOutputPort)){
  426. val outVar = externalOutputPort2OutVarDeclaration.get(externalOutputPort)
  427. Log.println("Found binding: " + internalOutputPort.qualifiedName + "->" + externalOutputPort.qualifiedName + " to be stored in " + outVar.name)
  428. internalOutputPort2OutVarDeclaration.put(internalOutputPort, outVar)
  429. }
  430. }
  431. Log.pop("transitiveStep")
  432. return internalOutputPort2OutVarDeclaration
  433. }
  434. def String qualifiedName(Port port){
  435. if (port.eContainer instanceof FMU){
  436. return (port.eContainer as FMU).name + "." + port.name
  437. }
  438. return port.name
  439. }
  440. def removeBindings(HashMap<Port, Port> internalPort2ExternalPortBindings, Adaptation sa) {
  441. Log.push("removeBindings")
  442. for (internalPort : internalPort2ExternalPortBindings.keySet){
  443. val externalPort = internalPort2ExternalPortBindings.get(internalPort)
  444. Log.println("Removing binding " + externalPort.qualifiedName + "->" + internalPort.qualifiedName)
  445. externalPort.targetdependency = null
  446. Log.println("Removing binding " + externalPort.qualifiedName + "<-" + internalPort.qualifiedName)
  447. externalPort.sourcedependency = null
  448. }
  449. Log.pop("removeBindings")
  450. }
  451. def findAllExternalPort2InputPort_Bindings(Adaptation sa) {
  452. Log.push("findAllExternalPort2InputPort_Bindings")
  453. val internalPort2ExternalPortBindings = new HashMap<Port, Port>()
  454. for (port : getAllInnerFMUInputPortDeclarations(sa)){
  455. Log.println("Checking if port " + port.qualifiedName + " is bound to an external port.")
  456. val externalPort = findExternalPortByTargetDependency(sa.inports, port)
  457. if (externalPort !== null){
  458. Log.println("Port " + port.qualifiedName + " is bound to an external port: " + externalPort.qualifiedName)
  459. internalPort2ExternalPortBindings.put(port, externalPort)
  460. } else {
  461. Log.println("Port " + port.qualifiedName + " is not bound to an external port.")
  462. }
  463. }
  464. Log.pop("findAllExternalPort2InputPort_Bindings")
  465. return internalPort2ExternalPortBindings
  466. }
  467. def findAllInternalPort2ExternalOutputPort_Bindings(Adaptation sa) {
  468. Log.push("findAllInternalPort2ExternalOutputPort_Bindings")
  469. val internalPort2ExternalPortBindings = new HashMap<Port, Port>()
  470. for (port : getAllInnerFMUOutputPortDeclarations(sa)){
  471. Log.println("Checking if port " + port.qualifiedName + " is bound to an external port.")
  472. val externalPort = findExternalPortBySourceDependency(sa.outports, port)
  473. if (externalPort !== null){
  474. Log.println("Port " + externalPort.qualifiedName + " is bound to an internal port: " + port.qualifiedName
  475. )
  476. internalPort2ExternalPortBindings.put(port, externalPort)
  477. } else {
  478. Log.println("Port " + port.qualifiedName + " is not bound to an external port.")
  479. }
  480. }
  481. Log.pop("findAllInternalPort2ExternalOutputPort_Bindings")
  482. return internalPort2ExternalPortBindings
  483. }
  484. def createExternalPortNameFromInternalPort(String parentFMUName, String internalPortName) {
  485. //return parentFMUName + "__" + internalPortName // Violates transparency
  486. return internalPortName
  487. }
  488. def addInRules_External2Internal_Assignments(Adaptation sa, HashMap<Port, Port> internalPort2ExternalPort) {
  489. Log.push("addInRules_External2Internal_Assignments")
  490. val dataRule = getOrPrependTrueRule(sa.in.rules)
  491. for(internalPort : internalPort2ExternalPort.keySet){
  492. val externalPort = internalPort2ExternalPort.get(internalPort)
  493. check((dataRule.outputfunction instanceof CompositeOutputFunction), "Only CompositeOutputFunction is supported for now.")
  494. val outFunction = dataRule.outputfunction as CompositeOutputFunction
  495. addPortAssignment(outFunction.statements, internalPort, externalPort, 0, true)
  496. }
  497. Log.pop("addInRules_External2Internal_Assignments")
  498. }
  499. def addOutRules_Internal2External_Assignments(Adaptation sa, HashMap<Port, Port> internalPort2ExternalPort){
  500. Log.push("addOutRules_Internal2External_Assignments")
  501. val dataRule = getOrPrependTrueRule(sa.out.rules)
  502. for(internalPort : internalPort2ExternalPort.keySet){
  503. val externalPort = internalPort2ExternalPort.get(internalPort)
  504. check((dataRule.outputfunction instanceof CompositeOutputFunction), "Only CompositeOutputFunction is supported for now.")
  505. val outFunction = dataRule.outputfunction as CompositeOutputFunction
  506. addPortAssignment(outFunction.statements, externalPort, internalPort, 0, true)
  507. }
  508. Log.pop("addOutRules_Internal2External_Assignments")
  509. }
  510. def addPortAssignment(List<Statement> statements, Port toPort, Port fromPort, int position, boolean convertUnits) {
  511. Log.push("addPortAssignment")
  512. val assignment = SemanticAdaptationFactory.eINSTANCE.createAssignment()
  513. assignment.lvalue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  514. assignment.lvalue.owner = toPort.eContainer as FMU
  515. assignment.lvalue.ref = toPort
  516. val varRef = SemanticAdaptationFactory.eINSTANCE.createVariable()
  517. varRef.owner = fromPort.eContainer as FMU
  518. varRef.ref = fromPort
  519. if (convertUnits && compatibleUnits(toPort.unity, fromPort.unity) && !unitsEqual(toPort.unity, fromPort.unity)){
  520. Log.println("Converting units " + fromPort.unity + " to " + toPort.unity)
  521. assignment.expr = getConversionExpression(toPort.unity, fromPort.unity, varRef)
  522. } else {
  523. assignment.expr = varRef
  524. Log.println("Assignment " + toPort.qualifiedName + " := " + fromPort.qualifiedName + " created.")
  525. }
  526. statements.add(position, assignment)
  527. Log.pop("addPortAssignment")
  528. }
  529. def getConversionExpression(Unity toUnits, Unity fromUnit, Variable varFrom) {
  530. check(toUnits instanceof AtomicUnity && fromUnit instanceof AtomicUnity, "Conversion between units is only supported for AtomicUnits.")
  531. var toAtomic = toUnits as AtomicUnity
  532. var fromAtomic = fromUnit as AtomicUnity
  533. check(fromAtomic.name == "cm" && toAtomic.name == "m", "Conversion only possible between cm and m... Not very usefull :)")
  534. var division = SemanticAdaptationFactory.eINSTANCE.createDiv()
  535. division.left = varFrom
  536. var hundred = SemanticAdaptationFactory.eINSTANCE.createRealLiteral()
  537. hundred.value=100f
  538. division.right = hundred
  539. return division
  540. }
  541. def dispatch boolean unitsEqual(DivideUnity u1, DivideUnity u2) {
  542. return unitsEqual(u1.left, u2.left) && unitsEqual(u1.right, u2.right)
  543. }
  544. def dispatch boolean unitsEqual(Void u1, Void u2) {
  545. return true
  546. }
  547. def dispatch boolean unitsEqual(MultiplyUnity u1, MultiplyUnity u2) {
  548. return unitsEqual(u1.left, u2.left) && unitsEqual(u1.right, u2.right)
  549. }
  550. def dispatch boolean unitsEqual(AtomicUnity u1, AtomicUnity u2) {
  551. return u1.name == u2.name && u1.power == u2.power
  552. }
  553. def compatibleUnits(Unity u1, Unity u2) {
  554. if ((u1 === null && u2===null) ||
  555. (u1 !== null && u2!==null)
  556. ){
  557. // TODO What makes two units compatible?
  558. return true
  559. } else {
  560. return false
  561. }
  562. }
  563. def addInRules_External2Stored_Assignments(Adaptation sa, HashMap<Port, SingleVarDeclaration> inputPort2InVarDeclaration) {
  564. Log.push("addInRules_External2Stored_Assignments")
  565. if (sa.in === null){
  566. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  567. }
  568. addRules_Port2Stored_Assignments(sa.in.rules, inputPort2InVarDeclaration)
  569. Log.pop("addInRules_External2Stored_Assignments")
  570. }
  571. def addOutRules_Internal2Stored_Assignments(Adaptation sa, HashMap<Port, SingleVarDeclaration> internalOutputPort2OutVarDeclaration) {
  572. Log.push("addOutRules_Internal2Stored_Assignments")
  573. if (sa.out === null){
  574. sa.out = SemanticAdaptationFactory.eINSTANCE.createOutRulesBlock()
  575. }
  576. addRules_Port2Stored_Assignments(sa.out.rules, internalOutputPort2OutVarDeclaration)
  577. Log.pop("addOutRules_Internal2Stored_Assignments")
  578. }
  579. def addRules_Port2Stored_Assignments(List<DataRule> rules, HashMap<Port, SingleVarDeclaration> port2VarDeclaration) {
  580. Log.push("addRules_External2Stored_Assignments")
  581. val dataRule = getOrPrependTrueRule(rules)
  582. if (dataRule.statetransitionfunction.expression !== null){
  583. throw new Exception("Expressions in rules are not supported yet.")
  584. // This and the one below are asily solved with a syntactic sugar substitution.
  585. }
  586. if (dataRule.statetransitionfunction.assignment !== null){
  587. throw new Exception("Assignment in rules are not supported yet.")
  588. }
  589. for(port : port2VarDeclaration.keySet){
  590. val storedVarDecl = port2VarDeclaration.get(port)
  591. addAssignmentToStoredVar(dataRule.statetransitionfunction.statements, port, storedVarDecl)
  592. }
  593. Log.pop("addRules_External2Stored_Assignments")
  594. }
  595. def addAssignmentToStoredVar(List<Statement> statements, Port internalPort, SingleVarDeclaration storedVarDecl) {
  596. Log.push("addAssignmentToStoredVar")
  597. val assignment = SemanticAdaptationFactory.eINSTANCE.createAssignment()
  598. assignment.lvalue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  599. assignment.lvalue.ref = storedVarDecl
  600. assignment.expr = SemanticAdaptationFactory.eINSTANCE.createVariable()
  601. (assignment.expr as Variable).owner = internalPort.eContainer as FMU
  602. (assignment.expr as Variable).ref = internalPort
  603. statements.add(0, assignment)
  604. Log.println("Assignment " + storedVarDecl.name + " := " + internalPort.qualifiedName + " created.")
  605. Log.pop("addAssignmentToStoredVar")
  606. }
  607. def getOrPrependTrueRule(List<DataRule> rules) {
  608. var DataRule rule = null
  609. if (rules.size == 0 || !isTrueRule(rules.head)){
  610. Log.println("No existing rule found with true condition. Creating one.")
  611. val trueRule = SemanticAdaptationFactory.eINSTANCE.createDataRule()
  612. trueRule.condition = SemanticAdaptationFactory.eINSTANCE.createRuleCondition()
  613. val trueExpr = SemanticAdaptationFactory.eINSTANCE.createBoolLiteral()
  614. trueExpr.value = "true"
  615. trueRule.condition.condition = trueExpr
  616. trueRule.statetransitionfunction = SemanticAdaptationFactory.eINSTANCE.createStateTransitionFunction()
  617. trueRule.outputfunction = SemanticAdaptationFactory.eINSTANCE.createCompositeOutputFunction()
  618. rules.add(0, trueRule)
  619. rule = trueRule
  620. } else {
  621. Log.println("Existing rule with true condition found.")
  622. rule = rules.head
  623. }
  624. return rule
  625. }
  626. def isTrueRule(DataRule rule){
  627. if (rule.condition.condition instanceof BoolLiteral){
  628. return (rule.condition.condition as BoolLiteral).value == "true"
  629. }
  630. return false
  631. }
  632. def addInVars(Adaptation sa, Map<Port, SingleParamDeclaration> inputPort2parameterDeclaration){
  633. Log.push("addInVars")
  634. if (sa.in === null){
  635. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  636. }
  637. val inputPort2InVarDeclaration = addStorageVars(sa.in.globalInVars, inputPort2parameterDeclaration)
  638. Log.pop("addInVars")
  639. return inputPort2InVarDeclaration
  640. }
  641. def addOutVars(Adaptation sa, Map<Port, SingleParamDeclaration> outputPort2parameterDeclaration){
  642. Log.push("addOutVars")
  643. if (sa.out === null){
  644. sa.out = SemanticAdaptationFactory.eINSTANCE.createOutRulesBlock()
  645. }
  646. val outputPort2InVarDeclaration = addStorageVars(sa.out.globalOutVars, outputPort2parameterDeclaration)
  647. Log.pop("addOutVars")
  648. return outputPort2InVarDeclaration
  649. }
  650. def addStorageVars(List<Declaration> varDeclarations, Map<Port, SingleParamDeclaration> port2parameterDeclaration){
  651. Log.push("addStorageVars")
  652. var port2VarDeclaration = new HashMap<Port, SingleVarDeclaration>()
  653. for(port : port2parameterDeclaration.keySet){
  654. Log.println("Processing port " + port.qualifiedName)
  655. val paramDecl = port2parameterDeclaration.get(port)
  656. val varDeclarationName = getStorageVarDeclarationName(port)
  657. if (!varDeclarationExists(varDeclarationName, varDeclarations) ){
  658. Log.println("Creating new variable declaration " + varDeclarationName)
  659. val varDeclaration = addNewVarDeclaration(port, paramDecl, varDeclarations)
  660. port2VarDeclaration.put(port, varDeclaration)
  661. } else {
  662. Log.println("Input variable declaration " + varDeclarationName + " already exists.")
  663. }
  664. }
  665. Log.pop("addStorageVars")
  666. return port2VarDeclaration
  667. }
  668. def addNewVarDeclaration(Port externalInputPort, SingleParamDeclaration paramDecl, List<Declaration> varDeclarations) {
  669. /*
  670. if (sa.in === null){
  671. sa.in = SemanticAdaptationFactory.eINSTANCE.createInRulesBlock()
  672. }
  673. */
  674. if (varDeclarations.size == 0){
  675. varDeclarations.add(SemanticAdaptationFactory.eINSTANCE.createDeclaration())
  676. }
  677. val newSingleVarDecl = SemanticAdaptationFactory.eINSTANCE.createSingleVarDeclaration()
  678. newSingleVarDecl.name = getStorageVarDeclarationName(externalInputPort)
  679. newSingleVarDecl.type = externalInputPort.type
  680. val initValue = SemanticAdaptationFactory.eINSTANCE.createVariable()
  681. initValue.ref = paramDecl
  682. newSingleVarDecl.expr = initValue
  683. varDeclarations.head.declarations.add(newSingleVarDecl)
  684. Log.println("New variable declaration created: " + newSingleVarDecl.name + " := " + paramDecl.name)
  685. return newSingleVarDecl
  686. }
  687. def varDeclarationExists(String invarName, List<Declaration> varDeclarations) {
  688. for (declarations : varDeclarations){ // sa.in.globalInVars
  689. for (decl : declarations.declarations){
  690. if (decl.name == invarName){
  691. return true
  692. }
  693. }
  694. }
  695. return false
  696. }
  697. def getStorageVarDeclarationName(Port externalInputPort) {
  698. return "stored__" + externalInputPort.name;
  699. }
  700. def genericDeclarationInferenceAlgorithm(Adaptation sa,
  701. (EObject)=>Object getField,
  702. (EObject, Object)=>void setField,
  703. (EObject)=>Object inferField
  704. ){
  705. Log.push("Running generic inference algorithm...")
  706. /*
  707. * Dumbest (and simplest) algorithm for this is a fixed point computation:
  708. * 1. Look for every var/port declaration
  709. * 2. If that var has a XXX already, nothing else to be done.
  710. * 3. If that var has no XXX declared, then
  711. * 3.1 If var/port has an initial value or connection, then
  712. * 3.1.1 If the initial_value/connection has a XXX declared, then var gets that XXX.
  713. * 3.1.2 Otherwise, nothing else to be done.
  714. * 3.2 If var/port has no initial value or connection then this either is a missing feature, or an error.
  715. * 3.3 If something has changed, go to 1. Otherwise, end.
  716. *
  717. * An extra set of instructions is there to push the element field information using connections and bindings.
  718. */
  719. var fixedPoint = false
  720. var unfieldedElementsCounter = 0
  721. while (! fixedPoint){
  722. fixedPoint = true
  723. unfieldedElementsCounter = 0
  724. Log.println("Inferring parameter fields...")
  725. for (paramDeclarations : sa.params) {
  726. for (paramDeclaration : paramDeclarations.declarations) {
  727. Log.println("Computing field for param " + paramDeclaration.name)
  728. if(getField.apply(paramDeclaration) !== null){
  729. Log.println("Already has been inferred: " + getField.apply(paramDeclaration))
  730. } else {
  731. Log.println("Has not been inferred yet.")
  732. if (tryInferAndAssignField(paramDeclaration, getField, setField, inferField)){
  733. fixedPoint = false
  734. } else {
  735. unfieldedElementsCounter++
  736. }
  737. }
  738. }
  739. }
  740. if(sa.inner !== null){
  741. if(sa.inner instanceof InnerFMUDeclarationFull){
  742. var innerFMUFull = sa.inner as InnerFMUDeclarationFull
  743. for(fmu : innerFMUFull.fmus){
  744. Log.println("Inferring port fields of FMU " + fmu.name)
  745. for (port : EcoreUtil2.getAllContentsOfType(fmu, Port)) {
  746. if(getField.apply(port) !== null){
  747. Log.println("Already has a field: " + getField.apply(port))
  748. } else {
  749. if (tryInferAndAssignField(port, getField, setField, inferField)){
  750. fixedPoint = false
  751. } else {
  752. unfieldedElementsCounter++
  753. }
  754. }
  755. }
  756. }
  757. if (innerFMUFull.connection.size > 0){
  758. Log.println("Inferring port fields using internal scenario bindings.")
  759. for (binding : innerFMUFull.connection){
  760. if (getField.apply(binding.src.port) !== null && getField.apply(binding.tgt.port) !== null){
  761. Log.println("Both ports have fields already.")
  762. } else {
  763. var inferredFieldAttempt = inferPortFieldViaConnection(binding, getField, setField, inferField)
  764. if (inferredFieldAttempt !== null){
  765. if (getField.apply(binding.src.port) === null){
  766. setField.apply(binding.src.port, inferredFieldAttempt)
  767. } else if (getField.apply(binding.tgt.port) === null){
  768. setField.apply(binding.tgt.port, inferredFieldAttempt)
  769. }
  770. fixedPoint = false
  771. unfieldedElementsCounter--
  772. Log.println("Got new field: " + inferredFieldAttempt)
  773. } else {
  774. Log.println("Cannot infer field from binding now.")
  775. }
  776. }
  777. }
  778. }
  779. } else {
  780. throw new Exception("Field inference only supported for InnerFMUDeclarationFull.")
  781. }
  782. }
  783. Log.println("Inferring external port fields...")
  784. var externalPorts = new LinkedList(sa.inports)
  785. externalPorts.addAll(sa.outports)
  786. for (port : externalPorts) {
  787. if (getField.apply(port) !== null){
  788. Log.println("Already has a field: " + getField.apply(port))
  789. if (pushPortField(port, getField, setField, inferField)){
  790. fixedPoint = false
  791. unfieldedElementsCounter--
  792. }
  793. } else {
  794. if (tryInferAndAssignField(port, getField, setField, inferField)){
  795. fixedPoint = false
  796. } else {
  797. unfieldedElementsCounter++
  798. }
  799. }
  800. }
  801. Log.println("Inferring all other declaration fields...")
  802. for (varDeclaration : EcoreUtil2.getAllContentsOfType(sa, SingleVarDeclaration)) {
  803. Log.println("Computing field for declaration " + varDeclaration.name)
  804. if(getField.apply(varDeclaration) !== null){
  805. Log.println("Already has a field: " + getField.apply(varDeclaration))
  806. } else {
  807. if (tryInferAndAssignField(varDeclaration, getField, setField, inferField)){
  808. fixedPoint = false
  809. } else {
  810. unfieldedElementsCounter++
  811. }
  812. }
  813. }
  814. Log.println("Ended iteration with unfielded elements remaining: " + unfieldedElementsCounter)
  815. } // while (! fixedPoint)
  816. Log.pop("Running generic inference algorithm... DONE")
  817. return unfieldedElementsCounter
  818. }
  819. def tryInferAndAssignField(EObject element,
  820. (EObject)=>Object getField,
  821. (EObject, Object)=>void setField,
  822. (EObject)=>Object inferField) {
  823. var inferredFieldAttempt = inferField.apply(element)
  824. if (inferredFieldAttempt !== null){
  825. setField.apply(element, inferredFieldAttempt)
  826. Log.println("Got new field: " + inferredFieldAttempt)
  827. return true
  828. } else {
  829. Log.println("Cannot infer field now.")
  830. return false
  831. }
  832. }
  833. def extractTypeFromExpression(Expression expression, String declarationName){
  834. if (expression instanceof IntLiteral){
  835. return "Integer"
  836. } else if (expression instanceof RealLiteral){
  837. return "Real"
  838. } else if (expression instanceof BoolLiteral){
  839. return "Bool"
  840. } else if (expression instanceof StringLiteral){
  841. return "String"
  842. } else if (expression instanceof Variable){
  843. var varRef = expression
  844. if (varRef.ref instanceof Port){
  845. var decl = varRef.ref as Port
  846. if (decl.type !== null){
  847. return decl.type
  848. }
  849. } else if(varRef.ref instanceof SingleParamDeclaration){
  850. var decl = varRef.ref as SingleParamDeclaration
  851. if (decl.type !== null){
  852. return decl.type
  853. }
  854. } else if(varRef.ref instanceof SingleVarDeclaration){
  855. var decl = varRef.ref as SingleVarDeclaration
  856. if (decl.type !== null){
  857. return decl.type
  858. }
  859. } else if(varRef.ref instanceof DeclaredParameter){
  860. throw new Exception("Type cannot be inferred for references to DeclaredParameter (for now). Please specify the explicit type of declaration " + declarationName)
  861. } else {
  862. throw new Exception("Unexpected kind of Variable expression found.")
  863. }
  864. } else if(expression instanceof BuiltinFunction){
  865. if (expression instanceof IsSet || expression instanceof Close){
  866. return "Bool"
  867. } else {
  868. return "Real"
  869. }
  870. } else if (expression instanceof CurrentTime) {
  871. return "Real"
  872. }
  873. else {
  874. 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.")
  875. }
  876. return null
  877. }
  878. def inferPortFieldViaConnection(Connection binding,
  879. (EObject)=>Object getField,
  880. (EObject, Object)=>void setField,
  881. (EObject)=>Object inferField
  882. ){
  883. var Object resultField = null
  884. if (getField.apply(binding.src.port) !== null && getField.apply(binding.tgt.port) !== null){
  885. throw new Exception("Wrong way of using this function. It assumes type is not inferred yet.")
  886. } else if (getField.apply(binding.src.port) !== null){
  887. resultField = getField.apply(binding.src.port)
  888. Log.println("Target port "+ binding.tgt.port.qualifiedName +" got new type: " + resultField)
  889. } else if (getField.apply(binding.tgt.port) !== null){
  890. resultField = getField.apply(binding.tgt.port)
  891. Log.println("Target port "+ binding.src.port.qualifiedName +" got new type: " + resultField)
  892. }
  893. return resultField
  894. }
  895. def pushPortField(Port port,
  896. (EObject)=>Object getField,
  897. (EObject, Object)=>void setField,
  898. (EObject)=>Object inferField){
  899. var fieldInferred = false
  900. Log.println("Pushing field of port " + port.qualifiedName + " to its bindings.")
  901. if(getField.apply(port) === null){
  902. Log.println("Has no field to be pushed.")
  903. throw new Exception("Wrong way of using this function. It assumes field is already inferred.")
  904. } else {
  905. Log.println("Pushing field: " + getField.apply(port))
  906. if(port.sourcedependency !== null){
  907. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  908. if(getField.apply(port.sourcedependency.port) === null){
  909. setField.apply(port.sourcedependency.port, getField.apply(port))
  910. Log.println("Port " + port.sourcedependency.port.qualifiedName + " got new type: " + getField.apply(port.sourcedependency.port))
  911. fieldInferred = true
  912. } else {
  913. Log.println("Source port already has field.")
  914. }
  915. } else {
  916. Log.println("Has no source dependency.")
  917. }
  918. if (port.targetdependency !== null) {
  919. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  920. if(getField.apply(port.targetdependency.port) === null){
  921. Log.println("Dependency has no field yet.")
  922. setField.apply(port.targetdependency.port, getField.apply(port))
  923. Log.println("Port " + port.targetdependency.port.qualifiedName + " got new type: " + getField.apply(port.targetdependency.port))
  924. fieldInferred = true
  925. } else {
  926. Log.println("Target port already has field.")
  927. }
  928. } else {
  929. Log.println("Has no target dependency.")
  930. }
  931. }
  932. return fieldInferred
  933. }
  934. def getPortUnit(Port port){
  935. var unitInferred = false
  936. Log.println("Computing unit for port " + port.qualifiedName)
  937. var Unity returnUnit = null
  938. if(port.unity !== null){
  939. throw new Exception("Wrong way of using this function. It assumes unit is not inferred yet.")
  940. } else {
  941. Log.println("Has no unit.")
  942. Log.println("Attempting to infer unit from bindings.")
  943. if(port.sourcedependency !== null){
  944. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  945. if(port.sourcedependency.port.unity === null){
  946. Log.println("Dependency has no unit yet.")
  947. } else {
  948. returnUnit = port.sourcedependency.port.unity
  949. Log.println("Got new unit: " + returnUnit)
  950. unitInferred = true
  951. }
  952. } else {
  953. Log.println("Has no source dependency.")
  954. }
  955. if (port.targetdependency !== null && !unitInferred) {
  956. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  957. if(port.targetdependency.port.unity === null){
  958. Log.println("Dependency has no unit yet.")
  959. } else {
  960. returnUnit = port.targetdependency.port.unity
  961. Log.println("Got new unit: " + returnUnit)
  962. unitInferred = true
  963. }
  964. } else {
  965. Log.println("Has no target dependency, or unit has already been inferred from source dependency.")
  966. }
  967. }
  968. return returnUnit
  969. }
  970. def getPortType(Port port){
  971. var typeInferred = false
  972. Log.println("Computing type for port " + port.qualifiedName)
  973. var String returnType = null
  974. if(port.type !== null){
  975. throw new Exception("Wrong way of using this function. It assumes type is not inferred yet.")
  976. } else {
  977. Log.println("Has no type.")
  978. Log.println("Attempting to infer type from units.")
  979. if (port.unity !== null){
  980. returnType = "Real"
  981. Log.println("Got new type: " + returnType)
  982. typeInferred = true
  983. } else {
  984. Log.println("Attempting to infer type from bindings.")
  985. if(port.sourcedependency !== null){
  986. Log.println("Has a source dependency: " + port.sourcedependency.port.qualifiedName)
  987. if(port.sourcedependency.port.type === null){
  988. Log.println("Dependency has no type yet.")
  989. } else {
  990. returnType = port.sourcedependency.port.type
  991. Log.println("Got new type: " + returnType)
  992. typeInferred = true
  993. }
  994. } else {
  995. Log.println("Has no source dependency.")
  996. }
  997. if (port.targetdependency !== null && !typeInferred) {
  998. Log.println("Has a target dependency: " + port.targetdependency.port.qualifiedName)
  999. if(port.targetdependency.port.type === null){
  1000. //println("Port object: " + port.targetdependency.port)
  1001. Log.println("Dependency has no type yet.")
  1002. } else {
  1003. returnType = port.targetdependency.port.type
  1004. Log.println("Got new type: " + returnType)
  1005. typeInferred = true
  1006. }
  1007. } else {
  1008. Log.println("Has no target dependency, or type has already been inferred from source dependency.")
  1009. }
  1010. }
  1011. }
  1012. return returnType
  1013. }
  1014. def addInPorts(Adaptation sa) {
  1015. Log.push("Adding input ports...")
  1016. for (port : getAllInnerFMUInputPortDeclarations(sa)){
  1017. var parentFMU = port.eContainer as InnerFMU
  1018. Log.println("Checking if port " + port.qualifiedName + " has incoming connections"
  1019. )
  1020. if (! hasConnection(port, sa, true)){
  1021. Log.println("Port " + port.qualifiedName + " has no incoming connections.")
  1022. val externalPortName = createExternalPortNameFromInternalPort(parentFMU.name, port.name)
  1023. if (findExternalPortByName(sa, externalPortName) === null){
  1024. var newExternalPort = createExternalInputPortDeclarationFromInnerPort(port, parentFMU, sa)
  1025. Log.println("External port " + newExternalPort.qualifiedName + " created.")
  1026. newExternalPort.bindExternalInputPortTo(parentFMU, port)
  1027. Log.println("External port " + newExternalPort.qualifiedName + " bound to port " + port.qualifiedName)
  1028. } else {
  1029. Log.println("Error: External port " + externalPortName + " already declared.")
  1030. throw new Exception("Error: External port " + externalPortName + " already declared. Please rename it, or connect it to an internal fmu port, to avoid clashes.")
  1031. }
  1032. } else {
  1033. Log.println("Port " + port.qualifiedName + " has an incoming connection.")
  1034. }
  1035. }
  1036. Log.pop("Adding input ports... DONE")
  1037. }
  1038. def bindExternalInputPortTo(Port externalInputPort, InnerFMU internalPortParent, Port internalPort) {
  1039. externalInputPort.targetdependency = SemanticAdaptationFactory.eINSTANCE.createSpecifiedPort()
  1040. externalInputPort.targetdependency.owner = internalPortParent
  1041. externalInputPort.targetdependency.port = internalPort
  1042. }
  1043. def bindExternalOutputPortTo(Port externalOutputPort, InnerFMU internalPortParent, Port internalPort) {
  1044. externalOutputPort.sourcedependency = SemanticAdaptationFactory.eINSTANCE.createSpecifiedPort()
  1045. externalOutputPort.sourcedependency.owner = internalPortParent
  1046. externalOutputPort.sourcedependency.port = internalPort
  1047. }
  1048. def createExternalInputPortDeclarationFromInnerPort(Port port, FMU parent, Adaptation sa) {
  1049. var externalInputPort = createExternalPortDeclarationFromInnerPort(port, parent)
  1050. sa.inports.add(externalInputPort)
  1051. return externalInputPort
  1052. }
  1053. def createExternalOutputPortDeclarationFromInnerPort(Port port, FMU parent, Adaptation sa) {
  1054. var externalOutputPort = createExternalPortDeclarationFromInnerPort(port, parent)
  1055. sa.outports.add(externalOutputPort)
  1056. return externalOutputPort
  1057. }
  1058. def createExternalPortDeclarationFromInnerPort(Port port, FMU parent) {
  1059. var externalInputPort = SemanticAdaptationFactory.eINSTANCE.createPort()
  1060. externalInputPort.name = createExternalPortNameFromInternalPort(parent.name, port.name)
  1061. externalInputPort.type = port.type
  1062. externalInputPort.unity = EcoreUtil2.copy(port.unity)
  1063. return externalInputPort
  1064. }
  1065. def findExternalPortByName(Adaptation adaptation, String name) {
  1066. for (externalInputPort : adaptation.inports){
  1067. if (externalInputPort.name == name){
  1068. return externalInputPort
  1069. }
  1070. }
  1071. return null
  1072. }
  1073. def findExternalPortByTargetDependency(List<Port> ports, Port targetDependency) {
  1074. for (externalInputPort : ports){
  1075. if (externalInputPort.targetdependency !== null && externalInputPort.targetdependency.port == targetDependency){
  1076. return externalInputPort
  1077. }
  1078. }
  1079. return null
  1080. }
  1081. def findExternalPortBySourceDependency(List<Port> ports, Port sourceDependency) {
  1082. for (externalInputPort : ports){
  1083. if (externalInputPort.sourcedependency !== null && externalInputPort.sourcedependency.port == sourceDependency){
  1084. return externalInputPort
  1085. }
  1086. }
  1087. return null
  1088. }
  1089. def hasConnection(Port port, Adaptation adaptation, Boolean checkForIncomming) {
  1090. var result = false
  1091. if ( (checkForIncomming && port.sourcedependency !== null) ||
  1092. (! checkForIncomming && port.targetdependency !== null)
  1093. ){
  1094. result = true
  1095. } else {
  1096. if (port.eContainer instanceof InnerFMU){
  1097. var innerScenarioDeclaration = EcoreUtil2.getContainerOfType(port, InnerFMUDeclaration)
  1098. if (innerScenarioDeclaration instanceof InnerFMUDeclarationFull){
  1099. var innerScenarioWithCoupling = innerScenarioDeclaration as InnerFMUDeclarationFull
  1100. if (innerScenarioWithCoupling.connection.size > 0){
  1101. for (connection : innerScenarioWithCoupling.connection ){
  1102. if ( (checkForIncomming && connection.tgt.port == port)){
  1103. Log.println("Port " + port.qualifiedName + " has an incoming connection from internal port " + connection.src.port.qualifiedName)
  1104. result = true
  1105. } else if (!checkForIncomming && connection.src.port == port) {
  1106. Log.println("Port " + port.qualifiedName + " has an outgoing connection to internal port " + connection.tgt.port.qualifiedName)
  1107. result = true
  1108. }
  1109. }
  1110. }
  1111. }
  1112. for (externalInputPort : adaptation.inports.filter[p | (checkForIncomming && p.targetdependency !== null) || (!checkForIncomming && p.sourcedependency !== null) ]){
  1113. if (checkForIncomming && externalInputPort.targetdependency.port == port){
  1114. Log.println("Port " + port.qualifiedName + " has an incoming connection from external port " + externalInputPort.qualifiedName)
  1115. result = true
  1116. } else if ( !checkForIncomming && externalInputPort.sourcedependency.port == port){
  1117. Log.println("Port " + port.qualifiedName + " has an outgoing connection to external port " + externalInputPort.qualifiedName)
  1118. result = true
  1119. }
  1120. }
  1121. }
  1122. }
  1123. return result
  1124. }
  1125. def getAllInnerFMUInputPortDeclarations(Adaptation sa){
  1126. return mapAllInnerFMUs(sa, [fmu | fmu.inports]);
  1127. }
  1128. def getAllInnerFMUOutputPortDeclarations(Adaptation sa){
  1129. return mapAllInnerFMUs(sa, [fmu | fmu.outports]);
  1130. }
  1131. def <T> List<T> mapAllInnerFMUs(Adaptation sa, (InnerFMU)=>List<T> map){
  1132. var result = new LinkedList()
  1133. if(sa.inner !== null){
  1134. if(sa.inner instanceof InnerFMUDeclarationFull){
  1135. var innerFMUFull = sa.inner as InnerFMUDeclarationFull
  1136. for(fmu : innerFMUFull.fmus){
  1137. result.addAll(map.apply(fmu))
  1138. }
  1139. } else {
  1140. throw new Exception("Only support for InnerFMUDeclarationFull.")
  1141. }
  1142. }
  1143. return result;
  1144. }
  1145. def addInParams(Adaptation sa) {
  1146. Log.push("Adding input parameters...")
  1147. val result = addParamForPortDeclarations(sa, sa.inports)
  1148. Log.pop("Adding input parameters... DONE")
  1149. return result
  1150. }
  1151. def addOutParams(Adaptation sa) {
  1152. Log.push("Adding output parameters...")
  1153. val result = addParamForPortDeclarations(sa, getAllInnerFMUOutputPortDeclarations(sa))
  1154. Log.pop("Adding output parameters... DONE")
  1155. return result
  1156. }
  1157. def addParamForPortDeclarations(Adaptation sa, List<Port> ports){
  1158. Log.push("addParamForPortDeclarations")
  1159. val PARAM_PREFIX = "INIT_"
  1160. var port2parameterDeclaration = new HashMap<Port, SingleParamDeclaration>(ports.size)
  1161. for (externalPortDecl : ports) {
  1162. Log.println("Generating parameter for port " + externalPortDecl.qualifiedName)
  1163. var paramname = PARAM_PREFIX + externalPortDecl.name.toUpperCase()
  1164. if (paramAlreadyDeclared(paramname, sa)){
  1165. Log.println("Parameter " + paramname + " already declared for port " + externalPortDecl.qualifiedName)
  1166. } else {
  1167. Log.println("Declaring new parameter " + paramname + " for port " + externalPortDecl.qualifiedName)
  1168. var paramDeclaration = addNewParamDeclaration(paramname, externalPortDecl, sa)
  1169. port2parameterDeclaration.put(externalPortDecl, paramDeclaration)
  1170. }
  1171. }
  1172. Log.pop("addParamForPortDeclarations")
  1173. return port2parameterDeclaration
  1174. }
  1175. def addNewParamDeclaration(String name, Port fromPort, Adaptation sa) {
  1176. var factory = SemanticAdaptationFactory.eINSTANCE
  1177. var paramDeclaration = factory.createSingleParamDeclaration()
  1178. paramDeclaration.name = name
  1179. paramDeclaration.type = fromPort.type
  1180. paramDeclaration.expr = getDefaultTypeExpression(paramDeclaration.type)
  1181. if (sa.params.size == 0){
  1182. sa.params.add(factory.createParamDeclarations())
  1183. }
  1184. sa.params.head.declarations.add(paramDeclaration)
  1185. return paramDeclaration
  1186. }
  1187. def getDefaultTypeExpression(String type) {
  1188. switch (type) {
  1189. case "Integer": {
  1190. val result = SemanticAdaptationFactory.eINSTANCE.createIntLiteral
  1191. result.value = 0
  1192. return result
  1193. }
  1194. case "Real": {
  1195. val result = SemanticAdaptationFactory.eINSTANCE.createRealLiteral
  1196. result.value = 0.0f
  1197. return result
  1198. }
  1199. case "Bool": {
  1200. val result = SemanticAdaptationFactory.eINSTANCE.createBoolLiteral
  1201. result.value = "false"
  1202. return result
  1203. }
  1204. case "String": {
  1205. val result = SemanticAdaptationFactory.eINSTANCE.createStringLiteral
  1206. result.value = " "
  1207. return result
  1208. }
  1209. default: {
  1210. throw new Exception("Unexpected type.")
  1211. }
  1212. }
  1213. }
  1214. def paramAlreadyDeclared(String name, Adaptation sa) {
  1215. for(paramDeclarations : sa.params){
  1216. for(paramDeclaration : paramDeclarations.declarations){
  1217. if(paramDeclaration.name == name){
  1218. return true
  1219. }
  1220. }
  1221. }
  1222. return false
  1223. }
  1224. def addOutPorts(Adaptation sa) {
  1225. Log.push("Adding output ports...")
  1226. for (port : getAllInnerFMUOutputPortDeclarations(sa)){
  1227. var parentFMU = port.eContainer as InnerFMU
  1228. Log.println("Checking if port " + port.qualifiedName + " has outgoing connections")
  1229. if (! hasConnection(port, sa, false)){
  1230. Log.println("Port " + port.qualifiedName + " has no outgoing connections.")
  1231. val externalPortName = createExternalPortNameFromInternalPort(parentFMU.name, port.name)
  1232. if (findExternalPortByName(sa, externalPortName) === null){
  1233. var newExternalPort = createExternalOutputPortDeclarationFromInnerPort(port, parentFMU, sa)
  1234. Log.println("External port " + newExternalPort.qualifiedName + " created.")
  1235. newExternalPort.bindExternalOutputPortTo(parentFMU, port)
  1236. Log.println("External port " + newExternalPort.qualifiedName + " bound to port " + port.qualifiedName)
  1237. } else {
  1238. Log.println("Error: External port " + externalPortName + " already declared.")
  1239. throw new Exception("Error: External port " + externalPortName + " already declared. Please rename it, or connect it to an internal fmu port, to avoid clashes.")
  1240. }
  1241. } else {
  1242. Log.println("Port " + port.qualifiedName + " has an incoming connection.")
  1243. }
  1244. }
  1245. Log.pop("Adding output ports... DONE")
  1246. }
  1247. }