From e378746783df218e95341db22e4515bc92144ef8 Mon Sep 17 00:00:00 2001 From: lhstrh Date: Wed, 8 Nov 2023 00:47:47 +0000 Subject: [PATCH] deploy: 1590f3bda395a37215b9cf6d8327b77fed9ed15b --- assets/lingua-franca-handbook.epub | Bin 255360 -> 255367 bytes assets/lingua-franca-handbook_lf-c.pdf | Bin 718027 -> 718394 bytes assets/lingua-franca-handbook_lf-cpp.pdf | Bin 652621 -> 652929 bytes assets/lingua-franca-handbook_lf-py.pdf | Bin 747887 -> 747910 bytes assets/lingua-franca-handbook_lf-rs.pdf | Bin 649723 -> 650063 bytes assets/lingua-franca-handbook_lf-ts.pdf | Bin 808247 -> 808266 bytes chunk-map.json | 2 +- ...c839.js => commons-192078f8b1bf52769f18.js | 6 +++--- ...ommons-192078f8b1bf52769f18.js.LICENSE.txt | 0 ...map => commons-192078f8b1bf52769f18.js.map | 2 +- community/index.html | 2 +- ...-documentation-tsx-04d26b5abab5434898b4.js | 2 -- ...-documentation-tsx-b5d67be0887eb73cf7b6.js | 2 ++ ...umentation-tsx-b5d67be0887eb73cf7b6.js.map | 2 +- docs/handbook/a-first-reactor/index.html | 2 +- docs/handbook/actions/index.html | 2 +- docs/handbook/arduino/index.html | 2 +- docs/handbook/causality-loops/index.html | 2 +- docs/handbook/code-extension/index.html | 2 +- docs/handbook/command-line-tools/index.html | 2 +- docs/handbook/composing-reactors/index.html | 2 +- .../containerized-execution/index.html | 6 +++--- docs/handbook/contributing/index.html | 2 +- docs/handbook/deadlines/index.html | 2 +- docs/handbook/developer-setup/index.html | 2 +- .../handbook/distributed-execution/index.html | 2 +- docs/handbook/eclipse-oomph/index.html | 2 +- docs/handbook/epoch-ide/index.html | 2 +- docs/handbook/expressions/index.html | 2 +- docs/handbook/extending-reactors/index.html | 2 +- docs/handbook/features/index.html | 2 +- .../index.html | 2 +- docs/handbook/generics/index.html | 2 +- docs/handbook/import-system/index.html | 2 +- docs/handbook/index.html | 2 +- docs/handbook/inputs-and-outputs/index.html | 2 +- docs/handbook/intellij/index.html | 2 +- .../language-specification/index.html | 2 +- .../logical-execution-time/index.html | 2 +- docs/handbook/methods/index.html | 2 +- docs/handbook/modal-models/index.html | 2 +- docs/handbook/multiports-and-banks/index.html | 2 +- docs/handbook/overview/index.html | 2 +- .../parameters-and-state-variables/index.html | 2 +- docs/handbook/preambles/index.html | 2 +- docs/handbook/proof-import/index.html | 2 +- .../handbook/reaction-declarations/index.html | 2 +- docs/handbook/reactions/index.html | 2 +- docs/handbook/reactors-on-patmos/index.html | 2 +- docs/handbook/regression-tests/index.html | 2 +- docs/handbook/related-work/index.html | 2 +- docs/handbook/running-benchmarks/index.html | 2 +- docs/handbook/security/index.html | 2 +- docs/handbook/superdense-time/index.html | 2 +- docs/handbook/target-declaration/index.html | 2 +- .../target-language-details/index.html | 2 +- docs/handbook/termination/index.html | 2 +- docs/handbook/time-and-timers/index.html | 2 +- docs/handbook/timing-analysis/index.html | 2 +- docs/handbook/tools/index.html | 2 +- docs/handbook/tracing/index.html | 2 +- docs/handbook/troubleshooting/index.html | 2 +- docs/handbook/tutorial-video/index.html | 2 +- docs/handbook/website-development/index.html | 2 +- docs/handbook/zephyr/index.html | 2 +- docs/index.html | 2 +- download/index.html | 2 +- empty/index.html | 2 +- index.html | 2 +- page-data/app-data.json | 2 +- .../handbook/a-first-reactor/page-data.json | 2 +- .../docs/handbook/actions/page-data.json | 2 +- .../docs/handbook/arduino/page-data.json | 2 +- .../handbook/causality-loops/page-data.json | 2 +- .../handbook/code-extension/page-data.json | 2 +- .../command-line-tools/page-data.json | 2 +- .../composing-reactors/page-data.json | 2 +- .../containerized-execution/page-data.json | 2 +- .../docs/handbook/contributing/page-data.json | 2 +- .../docs/handbook/deadlines/page-data.json | 2 +- .../handbook/developer-setup/page-data.json | 2 +- .../distributed-execution/page-data.json | 2 +- .../handbook/eclipse-oomph/page-data.json | 2 +- .../docs/handbook/epoch-ide/page-data.json | 2 +- .../docs/handbook/expressions/page-data.json | 2 +- .../extending-reactors/page-data.json | 2 +- .../docs/handbook/features/page-data.json | 2 +- .../page-data.json | 2 +- .../docs/handbook/generics/page-data.json | 2 +- .../handbook/import-system/page-data.json | 2 +- .../inputs-and-outputs/page-data.json | 2 +- .../docs/handbook/intellij/page-data.json | 2 +- .../language-specification/page-data.json | 2 +- .../logical-execution-time/page-data.json | 2 +- .../docs/handbook/methods/page-data.json | 2 +- .../docs/handbook/modal-models/page-data.json | 2 +- .../multiports-and-banks/page-data.json | 2 +- .../docs/handbook/overview/page-data.json | 2 +- .../page-data.json | 2 +- .../docs/handbook/preambles/page-data.json | 2 +- .../docs/handbook/proof-import/page-data.json | 2 +- .../reaction-declarations/page-data.json | 2 +- .../docs/handbook/reactions/page-data.json | 2 +- .../reactors-on-patmos/page-data.json | 2 +- .../handbook/regression-tests/page-data.json | 2 +- .../docs/handbook/related-work/page-data.json | 2 +- .../running-benchmarks/page-data.json | 2 +- .../docs/handbook/security/page-data.json | 2 +- .../handbook/superdense-time/page-data.json | 2 +- .../target-declaration/page-data.json | 2 +- .../target-language-details/page-data.json | 2 +- .../docs/handbook/termination/page-data.json | 2 +- .../handbook/time-and-timers/page-data.json | 2 +- .../handbook/timing-analysis/page-data.json | 2 +- page-data/docs/handbook/tools/page-data.json | 2 +- .../docs/handbook/tracing/page-data.json | 2 +- .../handbook/troubleshooting/page-data.json | 2 +- .../handbook/tutorial-video/page-data.json | 2 +- .../website-development/page-data.json | 2 +- page-data/docs/handbook/zephyr/page-data.json | 2 +- publications-and-presentations/index.html | 2 +- sitemap/sitemap-0.xml | 2 +- ...=> webpack-runtime-959e45056baa6f96280c.js | 4 ++-- ...ebpack-runtime-959e45056baa6f96280c.js.map | 2 +- webpack.stats.json | 2 +- 125 files changed, 123 insertions(+), 123 deletions(-) rename commons-c37c3719903bf568c839.js => commons-192078f8b1bf52769f18.js (99%) rename commons-c37c3719903bf568c839.js.LICENSE.txt => commons-192078f8b1bf52769f18.js.LICENSE.txt (100%) rename commons-c37c3719903bf568c839.js.map => commons-192078f8b1bf52769f18.js.map (99%) delete mode 100644 component---src-templates-documentation-tsx-04d26b5abab5434898b4.js create mode 100644 component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js rename component---src-templates-documentation-tsx-04d26b5abab5434898b4.js.map => component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js.map (99%) rename webpack-runtime-e4cdf7e8b18db5d0f606.js => webpack-runtime-959e45056baa6f96280c.js (97%) rename webpack-runtime-e4cdf7e8b18db5d0f606.js.map => webpack-runtime-959e45056baa6f96280c.js.map (99%) diff --git a/assets/lingua-franca-handbook.epub b/assets/lingua-franca-handbook.epub index 04727c3a61f6c6cfe7a0fd57094d8d4209dc417e..3b2534fc1fbdb993cf364ce45695b3682684af76 100644 GIT binary patch delta 2499 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 AWdHyG diff --git a/assets/lingua-franca-handbook_lf-c.pdf b/assets/lingua-franca-handbook_lf-c.pdf index a9ee1fcc0a70fc96ab27c6001ed544e5b3093759..61f4f62110e6524ca5cf759c31ceabd3077be373 100644 GIT binary patch delta 68147 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 delta 67811 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 diff --git a/assets/lingua-franca-handbook_lf-cpp.pdf b/assets/lingua-franca-handbook_lf-cpp.pdf index 51f41141dc0e1ad53cf1c9f2d9312b48ebfe3e53..0a245b45c63f4f626c1280847a5eba1d98a1e34d 100644 GIT binary patch delta 64099 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 C7WTye delta 51409 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 diff --git a/assets/lingua-franca-handbook_lf-rs.pdf b/assets/lingua-franca-handbook_lf-rs.pdf index a4d85cfce3b23743d4682771a84ab86ae04e82b7..ad6df1c6ff62e1179d0afdb01d698cb4494ea940 100644 GIT binary patch delta 64307 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 AqyPW_ delta 63813 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 I(4*`B1Dt>kumAu6 diff --git a/assets/lingua-franca-handbook_lf-ts.pdf b/assets/lingua-franca-handbook_lf-ts.pdf index 538daeb67abdc5d6c2dcfa3a72ac6421e14bb92b..8a3006de2702366bc51ad6ed5585507e97e06311 100644 GIT binary patch delta 58019 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.pageYOffset>n;n=window.pageYOffset,r?(e.classList.add("down"),e.classList.remove("up"),null==t||t.classList.add("hidden")):(e.classList.add("up"),e.classList.remove("down"),null==t||t.classList.remove("hidden"))}};document.removeEventListener("scroll",r,{capture:!0,passive:!0}),document.addEventListener("scroll",r,{capture:!0,passive:!0})}(),window.docsearch&&u(),!document.getElementById("algolia-search")){var e=document.createElement("script");e.id="algolia-search";var t=document.createElement("link");e.src=(0,o.withPrefix)("/js/docsearch.js"),e.async=!0,e.onload=function(){var e;window.docsearch&&(u(),t.rel="stylesheet",t.href=(0,o.withPrefix)("/css/docsearch.css"),t.type="text/css",document.body.appendChild(t),null===(e=document.getElementById("search-form"))||void 0===e||e.classList.add("search-enabled"))},document.body.appendChild(e)}}),[]),r.createElement("header",{dir:"ltr"},r.createElement("a",{className:"skip-to-main",href:"#site-content",tabIndex:0},t("skip_to_content")),r.createElement("div",{id:"top-menu",className:"up"},r.createElement("div",{className:"left below-small"},r.createElement(c,{id:"home-page-logo",to:"/","aria-label":"Lingua Franca Home Page"},r.createElement("picture",null,r.createElement("source",{media:"(min-width: 600px)",srcSet:n(2285).Z}),r.createElement("img",{src:n(6732).Z}))),r.createElement("nav",{role:"navigation"},r.createElement("ul",null,r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/download"},t("nav_download"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/docs/"},t("nav_documentation_short"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/docs/handbook/overview"},t("nav_handbook"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/community"},t("nav_community")))))),r.createElement("div",{className:"right above-small"},r.createElement("div",{className:"search-section"},r.createElement("div",{className:"nav-item"},r.createElement("form",{id:"search-form",className:"search top-nav",role:"search"},r.createElement("svg",{fill:"none",height:"16",viewBox:"0 0 16 6",width:"20",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"m10.5 0c.5052 0 .9922.0651042 1.4609.195312.4688.130209.9063.315105 1.3125.554688.4063.239583.7761.52865 1.1094.86719.3386.33333.6276.70312.8672 1.10937s.4245.84375.5547 1.3125.1953.95573.1953 1.46094-.0651.99219-.1953 1.46094-.3151.90625-.5547 1.3125-.5286.77864-.8672 1.11718c-.3333.33334-.7031.61978-1.1094.85938-.4062.2396-.8437.4245-1.3125.5547-.4687.1302-.9557.1953-1.4609.1953-.65104 0-1.27604-.1094-1.875-.3281-.59375-.2188-1.14062-.5339-1.64062-.94534l-6.132818 6.12504c-.098958.0989-.216145.1484-.351562.1484s-.252604-.0495-.351562-.1484c-.0989588-.099-.148438-.2162-.148438-.3516s.0494792-.2526.148438-.3516l6.125002-6.13278c-.41146-.5-.72656-1.04687-.94532-1.64062-.21874-.59896-.32812-1.22396-.32812-1.875 0-.50521.0651-.99219.19531-1.46094s.31511-.90625.55469-1.3125.52604-.77604.85938-1.10937c.33854-.33854.71093-.627607 1.11718-.86719s.84375-.424479 1.3125-.554688c.46875-.1302078.95573-.195312 1.46094-.195312zm0 10c.6198 0 1.2031-.11719 1.75-.35156.5469-.23959 1.0234-.5625 1.4297-.96875.4062-.40625.7265-.88281.9609-1.42969.2396-.54688.3594-1.13021.3594-1.75s-.1198-1.20312-.3594-1.75c-.2344-.54688-.5547-1.02344-.9609-1.42969-.4063-.40625-.8828-.72656-1.4297-.96093-.5469-.23959-1.1302-.35938-1.75-.35938-.61979 0-1.20312.11979-1.75.35938-.54688.23437-1.02344.55468-1.42969.96093s-.72916.88281-.96875 1.42969c-.23437.54688-.35156 1.13021-.35156 1.75s.11719 1.20312.35156 1.75c.23959.54688.5625 1.02344.96875 1.42969s.88281.72916 1.42969.96875c.54688.23437 1.13021.35156 1.75.35156z",fill:"#fff"})),r.createElement("span",null,r.createElement("input",{id:"search-box-top",type:"search",placeholder:t("nav_search_placeholder"),"aria-label":t("nav_search_aria")})),r.createElement("input",{type:"submit",style:{display:"none"}})))))),r.createElement("div",{id:"site-content"}))},u=n(8760),l=function(){document.documentElement.classList.remove("light-theme"),document.documentElement.classList.add("dark-theme")},h=function(){document.documentElement.classList.remove("dark-theme"),document.documentElement.classList.add("light-theme")},f=function(){var e=(0,s.D)((0,i.Z)()),t="undefined"!=typeof window&&window.matchMedia&&window.matchMedia("(prefers-color-scheme: dark)").matches,n=u.e&&localStorage.getItem("force-color-theme")||"system",o=(0,r.useState)(n),a=o[0],c=o[1],f=u.e&&localStorage.getItem("force-font")||"cascadia",m=(0,r.useState)(f),p=m[0],g=m[1];return r.createElement("section",{id:"customize"},r.createElement("article",null,r.createElement("h3",null,e("footer_customize")),r.createElement("label",null,r.createElement("p",null,e("footer_site_colours"),":"),r.createElement("div",{className:"switch-wrap"},r.createElement("select",{name:"colours",value:a,onChange:function(e){"system"===e.target.value?(t?l():h(),u.e&&localStorage.removeItem("force-color-theme")):"force-light"===e.target.value?(h(),u.e&&localStorage.setItem("force-color-theme","force-light")):"force-dark"===e.target.value&&(l(),u.e&&localStorage.setItem("force-color-theme","force-dark")),c(e.target.value)}},r.createElement("option",{value:"system"},e("footer_site_colours_options_system")),r.createElement("option",{value:"force-light"},e("footer_site_colours_options_always_light")),r.createElement("option",{value:"force-dark"},e("footer_site_colours_options_always_dark"))))),r.createElement("label",null,r.createElement("p",null,e("footer_code_font"),":"),r.createElement("div",{className:"switch-wrap"},r.createElement("select",{name:"font",value:p,onChange:function(e){var t,n;localStorage.setItem("force-font",e.target.value),t=e.target.value,(n=f)&&document.documentElement.classList.remove("font-"+n),document.documentElement.classList.add("font-"+t),g(e.target.value)}},r.createElement("option",{value:"cascadia"},"Cascadia"),r.createElement("option",{value:"cascadia-ligatures"},"Cascadia (ligatures)"),r.createElement("option",{value:"consolas"},"Consolas"),r.createElement("option",{value:"dank-mono"},"Dank Mono"),r.createElement("option",{value:"fira-code"},"Fira Code"),r.createElement("option",{value:"jetbrains-mono"},"JetBrains Mono"),r.createElement("option",{value:"open-dyslexic"},"OpenDyslexic"),r.createElement("option",{value:"sf-mono"},"SF Mono"),r.createElement("option",{value:"source-code-pro"},"Source Code Pro"))))))},m=[{title:"Get Started",url:"/docs/handbook/overview"},{title:"Download",url:"/download"},{title:"Why Lingua Franca",url:"/"},{title:"Publications",url:"/publications-and-presentations"}],p=[{title:"Get Help",url:"/community"},{title:"GitHub Repo",url:"https://github.com/lf-lang/lingua-franca"},{title:"@thelflang",url:"https://twitter.com/thelflang"},{title:"Web Repo",url:"https://github.com/lf-lang/website-lingua-franca"},{title:"Zulip",url:"https://zulip.lf-lang.org"}],g=function(e){var t=m.filter((function(e){return!e.url.includes("#show-examples")})),o=(0,a.i)(e.lang);e.suppressDocRecommendations;return r.createElement("footer",{id:"site-footer",role:"contentinfo"},e.suppressCustomization?null:r.createElement(f,null),r.createElement("section",{id:"community"},r.createElement("article",{id:"logos"},r.createElement("a",{href:""},r.createElement("img",{id:"lf-logo",width:195,height:75,src:n(9808).Z,alt:"Lingua Franca Logo"})),r.createElement("p",null,"Made with ♥ in Berkeley, Dallas, Dresden, Kiel, and Seoul"),r.createElement("p",null,"© 2019-",(new Date).getFullYear()," The Lingua Franca Team",r.createElement("br",null))),r.createElement("article",{id:"using-lf"},r.createElement("h3",null,"Using Lingua Franca"),r.createElement("ul",null,t.map((function(e){return r.createElement("li",{key:e.url},r.createElement(o,{to:e.url},e.title))})))),r.createElement("article",{id:"community-links"},r.createElement("h3",null,"Community"),r.createElement("ul",null,p.map((function(e){var t=function(e){switch(e){case"https://github.com/lf-lang/website-lingua-franca":case"https://github.com/lf-lang/lingua-franca":return r.createElement("svg",{fill:"none",height:"12",viewBox:"0 0 12 12",width:"12",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{clipRule:"evenodd",d:"m6.03927.165405c-3.27055 0-5.922909 2.652005-5.922909 5.923645 0 2.61709 1.697089 4.83705 4.050909 5.62035.29636.0546.40436-.1284.40436-.2854 0-.1408-.00509-.5131-.008-1.0073-1.64763.3578-1.99527-.7942-1.99527-.7942-.26946-.68436-.65782-.86654-.65782-.86654-.53782-.36727.04073-.36001.04073-.36001.59454.04182.90727.61055.90727.61055.52836.90509 1.38655.64364 1.724.492.05382-.38254.20691-.64363.376-.79163-1.31527-.14946-2.69818-.65782-2.69818-2.92764 0-.64654.23091-1.17564.60982-1.58946-.06109-.14981-.26437-.75236.05818-1.56763 0 0 .49709-.15927 1.62872.60727.47237-.13163.97928-.19709 1.48291-.19964.50328.00255 1.00982.06801 1.48291.19964 1.13091-.76654 1.62727-.60727 1.62727-.60727.32328.81527.12001 1.41782.05928 1.56763.37964.41382.60873.94292.60873 1.58946 0 2.27564-1.38509 2.77636-2.70437 2.92291.21237.18291.40182.54436.40182 1.09672 0 .79204-.00727 1.43094-.00727 1.62514 0 .1585.10691.3429.40727.2851 2.35197-.7851 4.04767-3.00369 4.04767-5.62005 0-3.27164-2.6524-5.923645-5.92403-5.923645z",fill:"#ffffff",fillRule:"evenodd"}));case"https://twitter.com/thelflang":return r.createElement("svg",{fill:"none",height:"10",viewBox:"0 0 13 10",width:"13",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"m4.58519 10c4.62962 0 7.16291-3.83919 7.16291-7.16289 0-.10801 0-.21602-.0049-.32403.4909-.35348.918-.80024 1.2568-1.30591-.4517.20128-.9377.33384-1.4483.39766.5204-.30929.9181-.805148 1.1095-1.394284-.486.289658-1.026.495856-1.6004.608773-.4615-.490946-1.11448-.7953322-1.83617-.7953322-1.38938 0-2.51856 1.1291732-2.51856 2.5185532 0 .19638.02455.38785.06383.57441-2.09143-.1031-3.94721-1.10954-5.1893-2.631474-.21602.373119-.33876.805154-.33876 1.266644 0 .87388.44677 1.64467 1.11936 2.09634-.41239-.01473-.80024-.12765-1.13899-.31421v.03437c0 1.21754.86897 2.23871 2.01778 2.46946-.2111.05891-.43203.08837-.66277.08837-.16202 0-.31912-.01473-.47131-.04419.31911 1.00153 1.25191 1.72813 2.35163 1.74777-.86406.67751-1.94906 1.08008-3.12733 1.08008-.20128 0-.402571-.00982-.59895-.03436 1.10954.70696 2.43509 1.12425 3.85393 1.12425z",fill:"#ffffff"}));case"https://zulip.lf-lang.org":return r.createElement("svg",{fill:"none",viewBox:"-30 -10 55 55",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"M22.767 3.589c0 1.209-.543 2.283-1.37 2.934l-8.034 7.174c-.149.128-.343-.078-.235-.25l2.946-5.9c.083-.165-.024-.368-.194-.368H4.452c-1.77 0-3.219-1.615-3.219-3.59C1.233 1.616 2.682 0 4.452 0h15.096c1.77-.001 3.219 1.614 3.219 3.589zM4.452 24h15.096c1.77 0 3.219-1.616 3.219-3.59 0-1.974-1.449-3.59-3.219-3.59H8.12c-.17 0-.277-.202-.194-.367l2.946-5.9c.108-.172-.086-.378-.235-.25l-8.033 7.173c-.828.65-1.37 1.725-1.37 2.934 0 1.974 1.448 3.59 3.218 3.59z",fill:"#ffffff"}))}}(e.url),n=t?r.createElement("span",{className:"link-prefix"},t):null;return r.createElement("li",{key:e.url},r.createElement("a",{style:{position:"relative"},href:e.url},n,e.title))}))))))},d=n(7609),y=function(e){var t=Object.assign({},e.ogTags,{"og:title":e.title,"og:description":e.description,"twitter:site":"thelflang"});return r.createElement(r.Fragment,null,r.createElement(d.q,{title:e.title,titleTemplate:"%s"},r.createElement("meta",{name:"description",key:"description",content:e.description}),Object.keys(t).map((function(e){return r.createElement("meta",{key:e,property:e,content:t[e]})}))))},b=function(e){return r.createElement(r.Fragment,null,r.createElement(d.q,{htmlAttributes:{lang:e.lang}},r.createElement("script",{src:"https://polyfill.io/v3/polyfill.min.js?features=es2015%2CArray.prototype.forEach%2CNodeList.prototype.forEach"}),r.createElement("link",{rel:"preload",href:(0,o.withPrefix)("/css/docsearch.css"),as:"style"}),r.createElement("style",null,"\npre data-err {\n background:url(\"data:image/svg+xml,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%206%203'%20enable-background%3D'new%200%200%206%203'%20height%3D'3'%20width%3D'6'%3E%3Cg%20fill%3D'%23c94824'%3E%3Cpolygon%20points%3D'5.5%2C0%202.5%2C3%201.1%2C3%204.1%2C0'%2F%3E%3Cpolygon%20points%3D'4%2C0%206%2C2%206%2C0.6%205.4%2C0'%2F%3E%3Cpolygon%20points%3D'0%2C2%201%2C3%202.4%2C3%200%2C0.6'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E\") repeat-x 0 100%;\n padding-bottom: 3px;\n}")),r.createElement(y,e),r.createElement("div",{className:"ms-Fabric"},r.createElement(c,e),r.createElement("main",{role:"main"},e.children),r.createElement(g,e)))}},9931:function(e,t,n){"use strict";n.r(t),n.d(t,{comCopy:function(){return r}});var r={com_layout_title:"Lingua Franca Community",com_layout_description:"Connect with other LF Programmers online and offline.",com_connect_online:"Online",com_connect_online_description:"Tap into our online resources to learn more about Lingua Franca, provide feedback, connect with our developers, and find out about new updates.",com_online_publications_desc:"View our publications and presentations.",com_online_publications_tag:"",com_online_github_desc:"Found a bug, or want to provide feedback?",com_online_github_href:"Tell us on GitHub.",com_online_zulip_href:"Join the conversation on Zulip.",com_online_zulip_desc:"Have questions, or want to chat with other users?",com_online_twitter_href:"@thelflang",com_online_twitter_desc:"Stay up to date. Follow us on Twitter"}},636:function(e,t,n){"use strict";n.r(t),n.d(t,{docCopy:function(){return r}});var r={doc_layout_title:"Lingua Franca Documentation",doc_layout_description:"Find Lingua Franca starter projects: from Python to C to C++ to TypeScript.",doc_headline:"Learning Resources",doc_headline_handbook_blurb:"The Lingua Franca language reference"}},4158:function(e,t,n){"use strict";n.r(t),n.d(t,{dtCopy:function(){return r}});var r={dt_s_page_title:"Search for typed packages",dt_s_title:"Type Search",dt_s_subtitle:"Find npm packages that have type declarations, either bundled or on Definitely Typed.",dt_s_match:"match",dt_s_matchs:"matches",dt_s_match_exact:"Exact Match",dt_s_popular_on_dt:"Popular on Definitely Typed",dt_s_downloads_short:"DLs",dt_s_downloads_via:"Via",dt_s_module:"Module",dt_s_last_update:"Last Updated",dt_s_install:"Install",dt_s_no_results:"No results found for",dt_s_no_results_try:"Try another search?",dt_s_copy:"copy",dt_s_copied:"copied"}},5437:function(e,t,n){"use strict";n.r(t),n.d(t,{lang:function(){return b},messages:function(){return y}});var r,o,i=n(5542),a=n(2784),s=n(6563);!function(e){e.formatDate="FormattedDate",e.formatTime="FormattedTime",e.formatNumber="FormattedNumber",e.formatList="FormattedList",e.formatDisplayName="FormattedDisplayName"}(r||(r={})),function(e){e.formatDate="FormattedDateParts",e.formatTime="FormattedTimeParts",e.formatNumber="FormattedNumberParts",e.formatList="FormattedListParts"}(o||(o={}));var c=function(e){var t=(0,s.Z)(),n=e.value,r=e.children,o=(0,i._T)(e,["value","children"]);return r(t.formatNumberToParts(n,o))};c.displayName="FormattedNumberParts";function u(e){var t=function(t){var n=(0,s.Z)(),r=t.value,o=t.children,a=(0,i._T)(t,["value","children"]),c="string"==typeof r?new Date(r||0):r;return o("formatDate"===e?n.formatDateToParts(c,a):n.formatTimeToParts(c,a))};return t.displayName=o[e],t}function l(e){var t=function(t){var n=(0,s.Z)(),r=t.value,o=t.children,c=(0,i._T)(t,["value","children"]),u=n[e](r,c);if("function"==typeof o)return o(u);var l=n.textComponent||a.Fragment;return a.createElement(l,null,u)};return t.displayName=r[e],t}c.displayName="FormattedNumberParts";l("formatDate"),l("formatTime"),l("formatNumber"),l("formatList"),l("formatDisplayName"),u("formatDate"),u("formatTime");var h=n(6801),f=n(636),m=n(2990),p=n(9931),g=n(5636),d=n(519),y=Object.assign({},h.navCopy,f.docCopy,p.comCopy,g.handbookCopy,m.indexCopy,d.footerCopy),b=y},519:function(e,t,n){"use strict";n.r(t),n.d(t,{footerCopy:function(){return r}});var r={footer_customize:"Customize",footer_site_colours:"Color Mode",footer_code_font:"Code Font",footer_site_colours_options_system:"System",footer_site_colours_options_always_light:"Always Light",footer_site_colours_options_always_dark:"Always Dark"}},5636:function(e,t,n){"use strict";n.r(t),n.d(t,{handbookCopy:function(){return r}});var r={handb_prev:"Previous",handb_next:"Next",handb_on_this_page:"On this page",handb_like_dislike_title:"Is this page helpful?",handb_like_desc:"Yes",handb_dislike_desc:"No",handb_thanks:"Thanks for the feedback",handb_deprecated_title:"This page has been deprecated",handb_deprecated_subtitle:"This handbook page has been replaced, ",handb_deprecated_subtitle_link:"go to the new page",handb_deprecated_subtitle_action:"Go to new page",handb_experimental_title:"This page contains experimental documentation",handb_experimental_subtitle:"The contents are about a work in progress topic."}},2990:function(e,t,n){"use strict";n.r(t),n.d(t,{indexCopy:function(){return r}});var r={index_2_headline:"Lingua Franca is a polyglot coordination language for reactive, concurrent, and time-sensitive applications.",index_2_byline:"Lingua Franca",index_2_summary:"Lingua Franca (LF) is a polyglot coordination language built to bring deterministic reactive concurrency and time to mainstream target programming languages (currently C, C++, Python, TypeScript, and Rust). LF is supported by a runtime system that is capable of concurrent and distributed execution of reactive programs that are deployable on the Cloud, the Edge, and even on bare-iron embedded platforms.",index_2_detail:"A Lingua Franca program specifies the interactions between components called reactors. The logic of each reactor is written in plain target code. A code generator synthesizes one or more programs in the target language, which are then compiled using standard tool chains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.",index_2_cta_install:"Download Lingua Franca",index_2_cta_install_subtitle:"Version",index_2_cta_install_fallback:"Latest stable release",index_2_cta_download:"On your computer",index_2_cta_download_subtitle:"via Github",index_2_what_is:"What is Lingua Franca?",index_2_what_is_lf:"Reactor-oriented",index_2_what_is_lf_copy:"Reactors are reactive and composable concurrent software components with inputs, outputs, and local state.",index_2_trust:"Concurrent",index_2_trust_copy:"Reactions to events are concurrent unless there is an explicit dependency between them.",index_2_scale:"Deterministic",index_2_scale_copy:"Lingua Franca programs are deterministic by default and therefore easy to test.",index_2_started_title:"Get Started",index_2_started_handbook:"Handbook",index_2_started_handbook_blurb:"Learn the language",index_2_install:"Install Lingua Franca"}},6801:function(e,t,n){"use strict";n.r(t),n.d(t,{navCopy:function(){return r}});var r={skip_to_content:"Skip to main content",nav_documentation:"Documentation",nav_documentation_short:"Docs",nav_download:"Download",nav_community:"Community",nav_handbook:"Handbook",nav_tools:"Tools",nav_search_placeholder:"Search Docs",nav_search_aria:"Search the Lingua Franca site",nav_this_page_in_your_lang:"This page is available in your language",nav_this_page_in_your_lang_open:"Open",nav_this_page_in_your_lang_no_more:"Don't show again"}},9577:function(e,t,n){"use strict";n.r(t),n.d(t,{inYourLanguage:function(){return r}});var r={en:{shorthand:"In En",body:"This page is available in English",open:"Go",cancel:"Don't ask again"}}},2419:function(e,t,n){"use strict";function r(e){return function(t,n){return e.formatMessage({id:t},n)}}n.d(t,{D:function(){return r}})},8349:function(e,t,n){"use strict";n.d(t,{kG:function(){return r}});function r(e,t,n){if(void 0===n&&(n=Error),!e)throw new n(t)}},8770:function(e,t,n){"use strict";function r(e,t){var n=t&&t.cache?t.cache:l,r=t&&t.serializer?t.serializer:c;return(t&&t.strategy?t.strategy:s)(e,{cache:n,serializer:r})}function o(e,t,n,r){var o,i=null==(o=r)||"number"==typeof o||"boolean"==typeof o?r:n(r),a=t.get(i);return void 0===a&&(a=e.call(this,r),t.set(i,a)),a}function i(e,t,n){var r=Array.prototype.slice.call(arguments,3),o=n(r),i=t.get(o);return void 0===i&&(i=e.apply(this,r),t.set(o,i)),i}function a(e,t,n,r,o){return n.bind(t,e,r,o)}function s(e,t){return a(e,this,1===e.length?o:i,t.cache.create(),t.serializer)}n.d(t,{A:function(){return h},Z:function(){return r}});var c=function(){return JSON.stringify(arguments)};function u(){this.cache=Object.create(null)}u.prototype.get=function(e){return this.cache[e]},u.prototype.set=function(e,t){this.cache[e]=t};var l={create:function(){return new u}},h={variadic:function(e,t){return a(e,this,i,t.cache.create(),t.serializer)},monadic:function(e,t){return a(e,this,o,t.cache.create(),t.serializer)}}},7846:function(e,t,n){"use strict";n.d(t,{wD:function(){return i},VG:function(){return c},rp:function(){return l},Ii:function(){return y},O4:function(){return s},uf:function(){return u},Wh:function(){return d},Jo:function(){return m},yx:function(){return p},Wi:function(){return f},HI:function(){return g},pe:function(){return h},Qc:function(){return ie}});var r,o=n(5542);!function(e){e[e.EXPECT_ARGUMENT_CLOSING_BRACE=1]="EXPECT_ARGUMENT_CLOSING_BRACE",e[e.EMPTY_ARGUMENT=2]="EMPTY_ARGUMENT",e[e.MALFORMED_ARGUMENT=3]="MALFORMED_ARGUMENT",e[e.EXPECT_ARGUMENT_TYPE=4]="EXPECT_ARGUMENT_TYPE",e[e.INVALID_ARGUMENT_TYPE=5]="INVALID_ARGUMENT_TYPE",e[e.EXPECT_ARGUMENT_STYLE=6]="EXPECT_ARGUMENT_STYLE",e[e.INVALID_NUMBER_SKELETON=7]="INVALID_NUMBER_SKELETON",e[e.INVALID_DATE_TIME_SKELETON=8]="INVALID_DATE_TIME_SKELETON",e[e.EXPECT_NUMBER_SKELETON=9]="EXPECT_NUMBER_SKELETON",e[e.EXPECT_DATE_TIME_SKELETON=10]="EXPECT_DATE_TIME_SKELETON",e[e.UNCLOSED_QUOTE_IN_ARGUMENT_STYLE=11]="UNCLOSED_QUOTE_IN_ARGUMENT_STYLE",e[e.EXPECT_SELECT_ARGUMENT_OPTIONS=12]="EXPECT_SELECT_ARGUMENT_OPTIONS",e[e.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE=13]="EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE",e[e.INVALID_PLURAL_ARGUMENT_OFFSET_VALUE=14]="INVALID_PLURAL_ARGUMENT_OFFSET_VALUE",e[e.EXPECT_SELECT_ARGUMENT_SELECTOR=15]="EXPECT_SELECT_ARGUMENT_SELECTOR",e[e.EXPECT_PLURAL_ARGUMENT_SELECTOR=16]="EXPECT_PLURAL_ARGUMENT_SELECTOR",e[e.EXPECT_SELECT_ARGUMENT_SELECTOR_FRAGMENT=17]="EXPECT_SELECT_ARGUMENT_SELECTOR_FRAGMENT",e[e.EXPECT_PLURAL_ARGUMENT_SELECTOR_FRAGMENT=18]="EXPECT_PLURAL_ARGUMENT_SELECTOR_FRAGMENT",e[e.INVALID_PLURAL_ARGUMENT_SELECTOR=19]="INVALID_PLURAL_ARGUMENT_SELECTOR",e[e.DUPLICATE_PLURAL_ARGUMENT_SELECTOR=20]="DUPLICATE_PLURAL_ARGUMENT_SELECTOR",e[e.DUPLICATE_SELECT_ARGUMENT_SELECTOR=21]="DUPLICATE_SELECT_ARGUMENT_SELECTOR",e[e.MISSING_OTHER_CLAUSE=22]="MISSING_OTHER_CLAUSE",e[e.INVALID_TAG=23]="INVALID_TAG",e[e.INVALID_TAG_NAME=25]="INVALID_TAG_NAME",e[e.UNMATCHED_CLOSING_TAG=26]="UNMATCHED_CLOSING_TAG",e[e.UNCLOSED_TAG=27]="UNCLOSED_TAG"}(r||(r={}));var i,a;n(1804),n(1715),n(8827);function s(e){return e.type===i.literal}function c(e){return e.type===i.argument}function u(e){return e.type===i.number}function l(e){return e.type===i.date}function h(e){return e.type===i.time}function f(e){return e.type===i.select}function m(e){return e.type===i.plural}function p(e){return e.type===i.pound}function g(e){return e.type===i.tag}function d(e){return!(!e||"object"!=typeof e||e.type!==a.number)}function y(e){return!(!e||"object"!=typeof e||e.type!==a.dateTime)}!function(e){e[e.literal=0]="literal",e[e.argument=1]="argument",e[e.number=2]="number",e[e.date=3]="date",e[e.time=4]="time",e[e.select=5]="select",e[e.plural=6]="plural",e[e.pound=7]="pound",e[e.tag=8]="tag"}(i||(i={})),function(e){e[e.number=0]="number",e[e.dateTime=1]="dateTime"}(a||(a={}));var b=/[ \xA0\u1680\u2000-\u200A\u202F\u205F\u3000]/,I=/(?:[Eec]{1,6}|G{1,5}|[Qq]{1,5}|(?:[yYur]+|U{1,5})|[ML]{1,5}|d{1,2}|D{1,3}|F{1}|[abB]{1,5}|[hkHK]{1,2}|w{1,2}|W{1}|m{1,2}|s{1,2}|[zZOvVxX]{1,4})(?=([^']*'[^']*')*[^']*$)/g;function v(e){var t={};return e.replace(I,(function(e){var n=e.length;switch(e[0]){case"G":t.era=4===n?"long":5===n?"narrow":"short";break;case"y":t.year=2===n?"2-digit":"numeric";break;case"Y":case"u":case"U":case"r":throw new RangeError("`Y/u/U/r` (year) patterns are not supported, use `y` instead");case"q":case"Q":throw new RangeError("`q/Q` (quarter) patterns are not supported");case"M":case"L":t.month=["numeric","2-digit","short","long","narrow"][n-1];break;case"w":case"W":throw new RangeError("`w/W` (week) patterns are not supported");case"d":t.day=["numeric","2-digit"][n-1];break;case"D":case"F":case"g":throw new RangeError("`D/F/g` (day) patterns are not supported, use `d` instead");case"E":t.weekday=4===n?"short":5===n?"narrow":"short";break;case"e":if(n<4)throw new RangeError("`e..eee` (weekday) patterns are not supported");t.weekday=["short","long","narrow","short"][n-4];break;case"c":if(n<4)throw new RangeError("`c..ccc` (weekday) patterns are not supported");t.weekday=["short","long","narrow","short"][n-4];break;case"a":t.hour12=!0;break;case"b":case"B":throw new RangeError("`b/B` (period) patterns are not supported, use `a` instead");case"h":t.hourCycle="h12",t.hour=["numeric","2-digit"][n-1];break;case"H":t.hourCycle="h23",t.hour=["numeric","2-digit"][n-1];break;case"K":t.hourCycle="h11",t.hour=["numeric","2-digit"][n-1];break;case"k":t.hourCycle="h24",t.hour=["numeric","2-digit"][n-1];break;case"j":case"J":case"C":throw new RangeError("`j/J/C` (hour) patterns are not supported, use `h/H/K/k` instead");case"m":t.minute=["numeric","2-digit"][n-1];break;case"s":t.second=["numeric","2-digit"][n-1];break;case"S":case"A":throw new RangeError("`S/A` (second) patterns are not supported, use `s` instead");case"z":t.timeZoneName=n<4?"short":"long";break;case"Z":case"O":case"v":case"V":case"X":case"x":throw new RangeError("`Z/O/v/V/X/x` (timeZone) patterns are not supported, use `z` instead")}return""})),t}var A=/[\t-\r \x85\u200E\u200F\u2028\u2029]/i;var E=/^\.(?:(0+)(\*)?|(#+)|(0+)(#+))$/g,C=/^(@+)?(\+|#+)?[rs]?$/g,T=/(\*)(0+)|(#+)(0+)|(0+)/g,_=/^(0+)$/;function w(e){var t={};return"r"===e[e.length-1]?t.roundingPriority="morePrecision":"s"===e[e.length-1]&&(t.roundingPriority="lessPrecision"),e.replace(C,(function(e,n,r){return"string"!=typeof r?(t.minimumSignificantDigits=n.length,t.maximumSignificantDigits=n.length):"+"===r?t.minimumSignificantDigits=n.length:"#"===n[0]?t.maximumSignificantDigits=n.length:(t.minimumSignificantDigits=n.length,t.maximumSignificantDigits=n.length+("string"==typeof r?r.length:0)),""})),t}function M(e){switch(e){case"sign-auto":return{signDisplay:"auto"};case"sign-accounting":case"()":return{currencySign:"accounting"};case"sign-always":case"+!":return{signDisplay:"always"};case"sign-accounting-always":case"()!":return{signDisplay:"always",currencySign:"accounting"};case"sign-except-zero":case"+?":return{signDisplay:"exceptZero"};case"sign-accounting-except-zero":case"()?":return{signDisplay:"exceptZero",currencySign:"accounting"};case"sign-never":case"+_":return{signDisplay:"never"}}}function N(e){var t;if("E"===e[0]&&"E"===e[1]?(t={notation:"engineering"},e=e.slice(2)):"E"===e[0]&&(t={notation:"scientific"},e=e.slice(1)),t){var n=e.slice(0,2);if("+!"===n?(t.signDisplay="always",e=e.slice(2)):"+?"===n&&(t.signDisplay="exceptZero",e=e.slice(2)),!_.test(e))throw new Error("Malformed concise eng/scientific notation");t.minimumIntegerDigits=e.length}return t}function L(e){var t=M(e);return t||{}}function S(e){for(var t={},n=0,r=e;n1)throw new RangeError("integer-width stems only accept a single optional option");i.options[0].replace(T,(function(e,n,r,o,i,a){if(n)t.minimumIntegerDigits=r.length;else{if(o&&i)throw new Error("We currently do not support maximum integer digits");if(a)throw new Error("We currently do not support exact integer digits")}return""}));continue}if(_.test(i.stem))t.minimumIntegerDigits=i.stem.length;else if(E.test(i.stem)){if(i.options.length>1)throw new RangeError("Fraction-precision stems only accept a single optional option");i.stem.replace(E,(function(e,n,r,o,i,a){return"*"===r?t.minimumFractionDigits=n.length:o&&"#"===o[0]?t.maximumFractionDigits=o.length:i&&a?(t.minimumFractionDigits=i.length,t.maximumFractionDigits=i.length+a.length):(t.minimumFractionDigits=n.length,t.maximumFractionDigits=n.length),""}));var a=i.options[0];"w"===a?t=(0,o.pi)((0,o.pi)({},t),{trailingZeroDisplay:"stripIfInteger"}):a&&(t=(0,o.pi)((0,o.pi)({},t),w(a)))}else if(C.test(i.stem))t=(0,o.pi)((0,o.pi)({},t),w(i.stem));else{var s=M(i.stem);s&&(t=(0,o.pi)((0,o.pi)({},t),s));var c=N(i.stem);c&&(t=(0,o.pi)((0,o.pi)({},t),c))}}return t}var O,D={AX:["H"],BQ:["H"],CP:["H"],CZ:["H"],DK:["H"],FI:["H"],ID:["H"],IS:["H"],ML:["H"],NE:["H"],RU:["H"],SE:["H"],SJ:["H"],SK:["H"],AS:["h","H"],BT:["h","H"],DJ:["h","H"],ER:["h","H"],GH:["h","H"],IN:["h","H"],LS:["h","H"],PG:["h","H"],PW:["h","H"],SO:["h","H"],TO:["h","H"],VU:["h","H"],WS:["h","H"],"001":["H","h"],AL:["h","H","hB"],TD:["h","H","hB"],"ca-ES":["H","h","hB"],CF:["H","h","hB"],CM:["H","h","hB"],"fr-CA":["H","h","hB"],"gl-ES":["H","h","hB"],"it-CH":["H","h","hB"],"it-IT":["H","h","hB"],LU:["H","h","hB"],NP:["H","h","hB"],PF:["H","h","hB"],SC:["H","h","hB"],SM:["H","h","hB"],SN:["H","h","hB"],TF:["H","h","hB"],VA:["H","h","hB"],CY:["h","H","hb","hB"],GR:["h","H","hb","hB"],CO:["h","H","hB","hb"],DO:["h","H","hB","hb"],KP:["h","H","hB","hb"],KR:["h","H","hB","hb"],NA:["h","H","hB","hb"],PA:["h","H","hB","hb"],PR:["h","H","hB","hb"],VE:["h","H","hB","hb"],AC:["H","h","hb","hB"],AI:["H","h","hb","hB"],BW:["H","h","hb","hB"],BZ:["H","h","hb","hB"],CC:["H","h","hb","hB"],CK:["H","h","hb","hB"],CX:["H","h","hb","hB"],DG:["H","h","hb","hB"],FK:["H","h","hb","hB"],GB:["H","h","hb","hB"],GG:["H","h","hb","hB"],GI:["H","h","hb","hB"],IE:["H","h","hb","hB"],IM:["H","h","hb","hB"],IO:["H","h","hb","hB"],JE:["H","h","hb","hB"],LT:["H","h","hb","hB"],MK:["H","h","hb","hB"],MN:["H","h","hb","hB"],MS:["H","h","hb","hB"],NF:["H","h","hb","hB"],NG:["H","h","hb","hB"],NR:["H","h","hb","hB"],NU:["H","h","hb","hB"],PN:["H","h","hb","hB"],SH:["H","h","hb","hB"],SX:["H","h","hb","hB"],TA:["H","h","hb","hB"],ZA:["H","h","hb","hB"],"af-ZA":["H","h","hB","hb"],AR:["H","h","hB","hb"],CL:["H","h","hB","hb"],CR:["H","h","hB","hb"],CU:["H","h","hB","hb"],EA:["H","h","hB","hb"],"es-BO":["H","h","hB","hb"],"es-BR":["H","h","hB","hb"],"es-EC":["H","h","hB","hb"],"es-ES":["H","h","hB","hb"],"es-GQ":["H","h","hB","hb"],"es-PE":["H","h","hB","hb"],GT:["H","h","hB","hb"],HN:["H","h","hB","hb"],IC:["H","h","hB","hb"],KG:["H","h","hB","hb"],KM:["H","h","hB","hb"],LK:["H","h","hB","hb"],MA:["H","h","hB","hb"],MX:["H","h","hB","hb"],NI:["H","h","hB","hb"],PY:["H","h","hB","hb"],SV:["H","h","hB","hb"],UY:["H","h","hB","hb"],JP:["H","h","K"],AD:["H","hB"],AM:["H","hB"],AO:["H","hB"],AT:["H","hB"],AW:["H","hB"],BE:["H","hB"],BF:["H","hB"],BJ:["H","hB"],BL:["H","hB"],BR:["H","hB"],CG:["H","hB"],CI:["H","hB"],CV:["H","hB"],DE:["H","hB"],EE:["H","hB"],FR:["H","hB"],GA:["H","hB"],GF:["H","hB"],GN:["H","hB"],GP:["H","hB"],GW:["H","hB"],HR:["H","hB"],IL:["H","hB"],IT:["H","hB"],KZ:["H","hB"],MC:["H","hB"],MD:["H","hB"],MF:["H","hB"],MQ:["H","hB"],MZ:["H","hB"],NC:["H","hB"],NL:["H","hB"],PM:["H","hB"],PT:["H","hB"],RE:["H","hB"],RO:["H","hB"],SI:["H","hB"],SR:["H","hB"],ST:["H","hB"],TG:["H","hB"],TR:["H","hB"],WF:["H","hB"],YT:["H","hB"],BD:["h","hB","H"],PK:["h","hB","H"],AZ:["H","hB","h"],BA:["H","hB","h"],BG:["H","hB","h"],CH:["H","hB","h"],GE:["H","hB","h"],LI:["H","hB","h"],ME:["H","hB","h"],RS:["H","hB","h"],UA:["H","hB","h"],UZ:["H","hB","h"],XK:["H","hB","h"],AG:["h","hb","H","hB"],AU:["h","hb","H","hB"],BB:["h","hb","H","hB"],BM:["h","hb","H","hB"],BS:["h","hb","H","hB"],CA:["h","hb","H","hB"],DM:["h","hb","H","hB"],"en-001":["h","hb","H","hB"],FJ:["h","hb","H","hB"],FM:["h","hb","H","hB"],GD:["h","hb","H","hB"],GM:["h","hb","H","hB"],GU:["h","hb","H","hB"],GY:["h","hb","H","hB"],JM:["h","hb","H","hB"],KI:["h","hb","H","hB"],KN:["h","hb","H","hB"],KY:["h","hb","H","hB"],LC:["h","hb","H","hB"],LR:["h","hb","H","hB"],MH:["h","hb","H","hB"],MP:["h","hb","H","hB"],MW:["h","hb","H","hB"],NZ:["h","hb","H","hB"],SB:["h","hb","H","hB"],SG:["h","hb","H","hB"],SL:["h","hb","H","hB"],SS:["h","hb","H","hB"],SZ:["h","hb","H","hB"],TC:["h","hb","H","hB"],TT:["h","hb","H","hB"],UM:["h","hb","H","hB"],US:["h","hb","H","hB"],VC:["h","hb","H","hB"],VG:["h","hb","H","hB"],VI:["h","hb","H","hB"],ZM:["h","hb","H","hB"],BO:["H","hB","h","hb"],EC:["H","hB","h","hb"],ES:["H","hB","h","hb"],GQ:["H","hB","h","hb"],PE:["H","hB","h","hb"],AE:["h","hB","hb","H"],"ar-001":["h","hB","hb","H"],BH:["h","hB","hb","H"],DZ:["h","hB","hb","H"],EG:["h","hB","hb","H"],EH:["h","hB","hb","H"],HK:["h","hB","hb","H"],IQ:["h","hB","hb","H"],JO:["h","hB","hb","H"],KW:["h","hB","hb","H"],LB:["h","hB","hb","H"],LY:["h","hB","hb","H"],MO:["h","hB","hb","H"],MR:["h","hB","hb","H"],OM:["h","hB","hb","H"],PH:["h","hB","hb","H"],PS:["h","hB","hb","H"],QA:["h","hB","hb","H"],SA:["h","hB","hb","H"],SD:["h","hB","hb","H"],SY:["h","hB","hb","H"],TN:["h","hB","hb","H"],YE:["h","hB","hb","H"],AF:["H","hb","hB","h"],LA:["H","hb","hB","h"],CN:["H","hB","hb","h"],LV:["H","hB","hb","h"],TL:["H","hB","hb","h"],"zu-ZA":["H","hB","hb","h"],CD:["hB","H"],IR:["hB","H"],"hi-IN":["hB","h","H"],"kn-IN":["hB","h","H"],"ml-IN":["hB","h","H"],"te-IN":["hB","h","H"],KH:["hB","h","H","hb"],"ta-IN":["hB","h","hb","H"],BN:["hb","hB","h","H"],MY:["hb","hB","h","H"],ET:["hB","hb","h","H"],"gu-IN":["hB","hb","h","H"],"mr-IN":["hB","hb","h","H"],"pa-IN":["hB","hb","h","H"],TW:["hB","hb","h","H"],KE:["hB","hb","H","h"],MM:["hB","hb","H","h"],TZ:["hB","hb","H","h"],UG:["hB","hb","H","h"]};function P(e){var t=e.hourCycle;if(void 0===t&&e.hourCycles&&e.hourCycles.length&&(t=e.hourCycles[0]),t)switch(t){case"h24":return"k";case"h23":return"H";case"h12":return"h";case"h11":return"K";default:throw new Error("Invalid hourCycle")}var n,r=e.language;return"root"!==r&&(n=e.maximize().region),(D[n||""]||D[r||""]||D["".concat(r,"-001")]||D["001"])[0]}var B=new RegExp("^".concat(b.source,"*")),R=new RegExp("".concat(b.source,"*$"));function H(e,t){return{start:e,end:t}}var j=!!String.prototype.startsWith,k=!!String.fromCodePoint,F=!!Object.fromEntries,x=!!String.prototype.codePointAt,G=!!String.prototype.trimStart,Z=!!String.prototype.trimEnd,U=!!Number.isSafeInteger?Number.isSafeInteger:function(e){return"number"==typeof e&&isFinite(e)&&Math.floor(e)===e&&Math.abs(e)<=9007199254740991},z=!0;try{z="a"===(null===(O=$("([^\\p{White_Space}\\p{Pattern_Syntax}]*)","yu").exec("a"))||void 0===O?void 0:O[0])}catch(ae){z=!1}var Y,W=j?function(e,t,n){return e.startsWith(t,n)}:function(e,t,n){return e.slice(n,n+t.length)===t},V=k?String.fromCodePoint:function(){for(var e=[],t=0;ti;){if((n=e[i++])>1114111)throw RangeError(n+" is not a valid code point");r+=n<65536?String.fromCharCode(n):String.fromCharCode(55296+((n-=65536)>>10),n%1024+56320)}return r},X=F?Object.fromEntries:function(e){for(var t={},n=0,r=e;n=n)){var r,o=e.charCodeAt(t);return o<55296||o>56319||t+1===n||(r=e.charCodeAt(t+1))<56320||r>57343?o:r-56320+(o-55296<<10)+65536}},Q=G?function(e){return e.trimStart()}:function(e){return e.replace(B,"")},J=Z?function(e){return e.trimEnd()}:function(e){return e.replace(R,"")};function $(e,t){return new RegExp(e,t)}if(z){var q=$("([^\\p{White_Space}\\p{Pattern_Syntax}]*)","yu");Y=function(e,t){var n;return q.lastIndex=t,null!==(n=q.exec(e)[1])&&void 0!==n?n:""}}else Y=function(e,t){for(var n=[];;){var r=K(e,t);if(void 0===r||ne(r)||re(r))break;n.push(r),t+=r>=65536?2:1}return V.apply(void 0,n)};var ee=function(){function e(e,t){void 0===t&&(t={}),this.message=e,this.position={offset:0,line:1,column:1},this.ignoreTag=!!t.ignoreTag,this.locale=t.locale,this.requiresOtherClause=!!t.requiresOtherClause,this.shouldParseSkeletons=!!t.shouldParseSkeletons}return e.prototype.parse=function(){if(0!==this.offset())throw Error("parser can only be used once");return this.parseMessage(0,"",!1)},e.prototype.parseMessage=function(e,t,n){for(var o=[];!this.isEOF();){var a=this.char();if(123===a){if((s=this.parseArgument(e,n)).err)return s;o.push(s.val)}else{if(125===a&&e>0)break;if(35!==a||"plural"!==t&&"selectordinal"!==t){if(60===a&&!this.ignoreTag&&47===this.peek()){if(n)break;return this.error(r.UNMATCHED_CLOSING_TAG,H(this.clonePosition(),this.clonePosition()))}if(60===a&&!this.ignoreTag&&te(this.peek()||0)){if((s=this.parseTag(e,t)).err)return s;o.push(s.val)}else{var s;if((s=this.parseLiteral(e,t)).err)return s;o.push(s.val)}}else{var c=this.clonePosition();this.bump(),o.push({type:i.pound,location:H(c,this.clonePosition())})}}}return{val:o,err:null}},e.prototype.parseTag=function(e,t){var n=this.clonePosition();this.bump();var o=this.parseTagName();if(this.bumpSpace(),this.bumpIf("/>"))return{val:{type:i.literal,value:"<".concat(o,"/>"),location:H(n,this.clonePosition())},err:null};if(this.bumpIf(">")){var a=this.parseMessage(e+1,t,!0);if(a.err)return a;var s=a.val,c=this.clonePosition();if(this.bumpIf("")?{val:{type:i.tag,value:o,children:s,location:H(n,this.clonePosition())},err:null}:this.error(r.INVALID_TAG,H(c,this.clonePosition())))}return this.error(r.UNCLOSED_TAG,H(n,this.clonePosition()))}return this.error(r.INVALID_TAG,H(n,this.clonePosition()))},e.prototype.parseTagName=function(){var e,t=this.offset();for(this.bump();!this.isEOF()&&(45===(e=this.char())||46===e||e>=48&&e<=57||95===e||e>=97&&e<=122||e>=65&&e<=90||183==e||e>=192&&e<=214||e>=216&&e<=246||e>=248&&e<=893||e>=895&&e<=8191||e>=8204&&e<=8205||e>=8255&&e<=8256||e>=8304&&e<=8591||e>=11264&&e<=12271||e>=12289&&e<=55295||e>=63744&&e<=64975||e>=65008&&e<=65533||e>=65536&&e<=983039);)this.bump();return this.message.slice(t,this.offset())},e.prototype.parseLiteral=function(e,t){for(var n=this.clonePosition(),r="";;){var o=this.tryParseQuote(t);if(o)r+=o;else{var a=this.tryParseUnquoted(e,t);if(a)r+=a;else{var s=this.tryParseLeftAngleBracket();if(!s)break;r+=s}}}var c=H(n,this.clonePosition());return{val:{type:i.literal,value:r,location:c},err:null}},e.prototype.tryParseLeftAngleBracket=function(){return this.isEOF()||60!==this.char()||!this.ignoreTag&&(te(e=this.peek()||0)||47===e)?null:(this.bump(),"<");var e},e.prototype.tryParseQuote=function(e){if(this.isEOF()||39!==this.char())return null;switch(this.peek()){case 39:return this.bump(),this.bump(),"'";case 123:case 60:case 62:case 125:break;case 35:if("plural"===e||"selectordinal"===e)break;return null;default:return null}this.bump();var t=[this.char()];for(this.bump();!this.isEOF();){var n=this.char();if(39===n){if(39!==this.peek()){this.bump();break}t.push(39),this.bump()}else t.push(n);this.bump()}return V.apply(void 0,t)},e.prototype.tryParseUnquoted=function(e,t){if(this.isEOF())return null;var n=this.char();return 60===n||123===n||35===n&&("plural"===t||"selectordinal"===t)||125===n&&e>0?null:(this.bump(),V(n))},e.prototype.parseArgument=function(e,t){var n=this.clonePosition();if(this.bump(),this.bumpSpace(),this.isEOF())return this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition()));if(125===this.char())return this.bump(),this.error(r.EMPTY_ARGUMENT,H(n,this.clonePosition()));var o=this.parseIdentifierIfPossible().value;if(!o)return this.error(r.MALFORMED_ARGUMENT,H(n,this.clonePosition()));if(this.bumpSpace(),this.isEOF())return this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition()));switch(this.char()){case 125:return this.bump(),{val:{type:i.argument,value:o,location:H(n,this.clonePosition())},err:null};case 44:return this.bump(),this.bumpSpace(),this.isEOF()?this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition())):this.parseArgumentOptions(e,t,o,n);default:return this.error(r.MALFORMED_ARGUMENT,H(n,this.clonePosition()))}},e.prototype.parseIdentifierIfPossible=function(){var e=this.clonePosition(),t=this.offset(),n=Y(this.message,t),r=t+n.length;return this.bumpTo(r),{value:n,location:H(e,this.clonePosition())}},e.prototype.parseArgumentOptions=function(e,t,n,s){var c,u=this.clonePosition(),l=this.parseIdentifierIfPossible().value,h=this.clonePosition();switch(l){case"":return this.error(r.EXPECT_ARGUMENT_TYPE,H(u,h));case"number":case"date":case"time":this.bumpSpace();var f=null;if(this.bumpIf(",")){this.bumpSpace();var m=this.clonePosition();if((E=this.parseSimpleArgStyleIfPossible()).err)return E;if(0===(y=J(E.val)).length)return this.error(r.EXPECT_ARGUMENT_STYLE,H(this.clonePosition(),this.clonePosition()));f={style:y,styleLocation:H(m,this.clonePosition())}}if((C=this.tryParseArgumentClose(s)).err)return C;var p=H(s,this.clonePosition());if(f&&W(null==f?void 0:f.style,"::",0)){var g=Q(f.style.slice(2));if("number"===l)return(E=this.parseNumberSkeletonFromString(g,f.styleLocation)).err?E:{val:{type:i.number,value:n,location:p,style:E.val},err:null};if(0===g.length)return this.error(r.EXPECT_DATE_TIME_SKELETON,p);var d=g;this.locale&&(d=function(e,t){for(var n="",r=0;r>1),c=P(t);for("H"!=c&&"k"!=c||(s=0);s-- >0;)n+="a";for(;a-- >0;)n=c+n}else n+="J"===o?"H":o}return n}(g,this.locale));var y={type:a.dateTime,pattern:d,location:f.styleLocation,parsedOptions:this.shouldParseSkeletons?v(d):{}};return{val:{type:"date"===l?i.date:i.time,value:n,location:p,style:y},err:null}}return{val:{type:"number"===l?i.number:"date"===l?i.date:i.time,value:n,location:p,style:null!==(c=null==f?void 0:f.style)&&void 0!==c?c:null},err:null};case"plural":case"selectordinal":case"select":var b=this.clonePosition();if(this.bumpSpace(),!this.bumpIf(","))return this.error(r.EXPECT_SELECT_ARGUMENT_OPTIONS,H(b,(0,o.pi)({},b)));this.bumpSpace();var I=this.parseIdentifierIfPossible(),A=0;if("select"!==l&&"offset"===I.value){if(!this.bumpIf(":"))return this.error(r.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE,H(this.clonePosition(),this.clonePosition()));var E;if(this.bumpSpace(),(E=this.tryParseDecimalInteger(r.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE,r.INVALID_PLURAL_ARGUMENT_OFFSET_VALUE)).err)return E;this.bumpSpace(),I=this.parseIdentifierIfPossible(),A=E.val}var C,T=this.tryParsePluralOrSelectOptions(e,l,t,I);if(T.err)return T;if((C=this.tryParseArgumentClose(s)).err)return C;var _=H(s,this.clonePosition());return"select"===l?{val:{type:i.select,value:n,options:X(T.val),location:_},err:null}:{val:{type:i.plural,value:n,options:X(T.val),offset:A,pluralType:"plural"===l?"cardinal":"ordinal",location:_},err:null};default:return this.error(r.INVALID_ARGUMENT_TYPE,H(u,h))}},e.prototype.tryParseArgumentClose=function(e){return this.isEOF()||125!==this.char()?this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(e,this.clonePosition())):(this.bump(),{val:!0,err:null})},e.prototype.parseSimpleArgStyleIfPossible=function(){for(var e=0,t=this.clonePosition();!this.isEOF();){switch(this.char()){case 39:this.bump();var n=this.clonePosition();if(!this.bumpUntil("'"))return this.error(r.UNCLOSED_QUOTE_IN_ARGUMENT_STYLE,H(n,this.clonePosition()));this.bump();break;case 123:e+=1,this.bump();break;case 125:if(!(e>0))return{val:this.message.slice(t.offset,this.offset()),err:null};e-=1;break;default:this.bump()}}return{val:this.message.slice(t.offset,this.offset()),err:null}},e.prototype.parseNumberSkeletonFromString=function(e,t){var n=[];try{n=function(e){if(0===e.length)throw new Error("Number skeleton cannot be empty");for(var t=[],n=0,r=e.split(A).filter((function(e){return e.length>0}));n=48&&a<=57))break;o=!0,i=10*i+(a-48),this.bump()}var s=H(r,this.clonePosition());return o?U(i*=n)?{val:i,err:null}:this.error(t,s):this.error(e,s)},e.prototype.offset=function(){return this.position.offset},e.prototype.isEOF=function(){return this.offset()===this.message.length},e.prototype.clonePosition=function(){return{offset:this.position.offset,line:this.position.line,column:this.position.column}},e.prototype.char=function(){var e=this.position.offset;if(e>=this.message.length)throw Error("out of bound");var t=K(this.message,e);if(void 0===t)throw Error("Offset ".concat(e," is at invalid UTF-16 code unit boundary"));return t},e.prototype.error=function(e,t){return{val:null,err:{kind:e,message:this.message,location:t}}},e.prototype.bump=function(){if(!this.isEOF()){var e=this.char();10===e?(this.position.line+=1,this.position.column=1,this.position.offset+=1):(this.position.column+=1,this.position.offset+=e<65536?1:2)}},e.prototype.bumpIf=function(e){if(W(this.message,e,this.offset())){for(var t=0;t=0?(this.bumpTo(n),!0):(this.bumpTo(this.message.length),!1)},e.prototype.bumpTo=function(e){if(this.offset()>e)throw Error("targetOffset ".concat(e," must be greater than or equal to the current offset ").concat(this.offset()));for(e=Math.min(e,this.message.length);;){var t=this.offset();if(t===e)break;if(t>e)throw Error("targetOffset ".concat(e," is at invalid UTF-16 code unit boundary"));if(this.bump(),this.isEOF())break}},e.prototype.bumpSpace=function(){for(;!this.isEOF()&&ne(this.char());)this.bump()},e.prototype.peek=function(){if(this.isEOF())return null;var e=this.char(),t=this.offset(),n=this.message.charCodeAt(t+(e>=65536?2:1));return null!=n?n:null},e}();function te(e){return e>=97&&e<=122||e>=65&&e<=90}function ne(e){return e>=9&&e<=13||32===e||133===e||e>=8206&&e<=8207||8232===e||8233===e}function re(e){return e>=33&&e<=35||36===e||e>=37&&e<=39||40===e||41===e||42===e||43===e||44===e||45===e||e>=46&&e<=47||e>=58&&e<=59||e>=60&&e<=62||e>=63&&e<=64||91===e||92===e||93===e||94===e||96===e||123===e||124===e||125===e||126===e||161===e||e>=162&&e<=165||166===e||167===e||169===e||171===e||172===e||174===e||176===e||177===e||182===e||187===e||191===e||215===e||247===e||e>=8208&&e<=8213||e>=8214&&e<=8215||8216===e||8217===e||8218===e||e>=8219&&e<=8220||8221===e||8222===e||8223===e||e>=8224&&e<=8231||e>=8240&&e<=8248||8249===e||8250===e||e>=8251&&e<=8254||e>=8257&&e<=8259||8260===e||8261===e||8262===e||e>=8263&&e<=8273||8274===e||8275===e||e>=8277&&e<=8286||e>=8592&&e<=8596||e>=8597&&e<=8601||e>=8602&&e<=8603||e>=8604&&e<=8607||8608===e||e>=8609&&e<=8610||8611===e||e>=8612&&e<=8613||8614===e||e>=8615&&e<=8621||8622===e||e>=8623&&e<=8653||e>=8654&&e<=8655||e>=8656&&e<=8657||8658===e||8659===e||8660===e||e>=8661&&e<=8691||e>=8692&&e<=8959||e>=8960&&e<=8967||8968===e||8969===e||8970===e||8971===e||e>=8972&&e<=8991||e>=8992&&e<=8993||e>=8994&&e<=9e3||9001===e||9002===e||e>=9003&&e<=9083||9084===e||e>=9085&&e<=9114||e>=9115&&e<=9139||e>=9140&&e<=9179||e>=9180&&e<=9185||e>=9186&&e<=9254||e>=9255&&e<=9279||e>=9280&&e<=9290||e>=9291&&e<=9311||e>=9472&&e<=9654||9655===e||e>=9656&&e<=9664||9665===e||e>=9666&&e<=9719||e>=9720&&e<=9727||e>=9728&&e<=9838||9839===e||e>=9840&&e<=10087||10088===e||10089===e||10090===e||10091===e||10092===e||10093===e||10094===e||10095===e||10096===e||10097===e||10098===e||10099===e||10100===e||10101===e||e>=10132&&e<=10175||e>=10176&&e<=10180||10181===e||10182===e||e>=10183&&e<=10213||10214===e||10215===e||10216===e||10217===e||10218===e||10219===e||10220===e||10221===e||10222===e||10223===e||e>=10224&&e<=10239||e>=10240&&e<=10495||e>=10496&&e<=10626||10627===e||10628===e||10629===e||10630===e||10631===e||10632===e||10633===e||10634===e||10635===e||10636===e||10637===e||10638===e||10639===e||10640===e||10641===e||10642===e||10643===e||10644===e||10645===e||10646===e||10647===e||10648===e||e>=10649&&e<=10711||10712===e||10713===e||10714===e||10715===e||e>=10716&&e<=10747||10748===e||10749===e||e>=10750&&e<=11007||e>=11008&&e<=11055||e>=11056&&e<=11076||e>=11077&&e<=11078||e>=11079&&e<=11084||e>=11085&&e<=11123||e>=11124&&e<=11125||e>=11126&&e<=11157||11158===e||e>=11159&&e<=11263||e>=11776&&e<=11777||11778===e||11779===e||11780===e||11781===e||e>=11782&&e<=11784||11785===e||11786===e||11787===e||11788===e||11789===e||e>=11790&&e<=11798||11799===e||e>=11800&&e<=11801||11802===e||11803===e||11804===e||11805===e||e>=11806&&e<=11807||11808===e||11809===e||11810===e||11811===e||11812===e||11813===e||11814===e||11815===e||11816===e||11817===e||e>=11818&&e<=11822||11823===e||e>=11824&&e<=11833||e>=11834&&e<=11835||e>=11836&&e<=11839||11840===e||11841===e||11842===e||e>=11843&&e<=11855||e>=11856&&e<=11857||11858===e||e>=11859&&e<=11903||e>=12289&&e<=12291||12296===e||12297===e||12298===e||12299===e||12300===e||12301===e||12302===e||12303===e||12304===e||12305===e||e>=12306&&e<=12307||12308===e||12309===e||12310===e||12311===e||12312===e||12313===e||12314===e||12315===e||12316===e||12317===e||e>=12318&&e<=12319||12320===e||12336===e||64830===e||64831===e||e>=65093&&e<=65094}function oe(e){e.forEach((function(e){if(delete e.location,f(e)||m(e))for(var t in e.options)delete e.options[t].location,oe(e.options[t].value);else u(e)&&d(e.style)||(l(e)||h(e))&&y(e.style)?delete e.style.location:g(e)&&oe(e.children)}))}function ie(e,t){void 0===t&&(t={}),t=(0,o.pi)({shouldParseSkeletons:!0,requiresOtherClause:!0},t);var n=new ee(e,t).parse();if(n.err){var i=SyntaxError(r[n.err.kind]);throw i.location=n.err.location,i.originalMessage=n.err.message,i}return(null==t?void 0:t.captureLocation)||oe(n.val),n.val}},7309:function(e,t,n){"use strict";n.d(t,{$6:function(){return h},OV:function(){return s},Qe:function(){return u},Rw:function(){return i},X9:function(){return l},bc:function(){return r},gb:function(){return c},wI:function(){return a}});var r,o=n(5542);!function(e){e.FORMAT_ERROR="FORMAT_ERROR",e.UNSUPPORTED_FORMATTER="UNSUPPORTED_FORMATTER",e.INVALID_CONFIG="INVALID_CONFIG",e.MISSING_DATA="MISSING_DATA",e.MISSING_TRANSLATION="MISSING_TRANSLATION"}(r||(r={}));var i=function(e){function t(n,r,o){var i=this,a=o?o instanceof Error?o:new Error(String(o)):void 0;return(i=e.call(this,"[@formatjs/intl Error ".concat(n,"] ").concat(r,"\n").concat(a?"\n".concat(a.message,"\n").concat(a.stack):""))||this).code=n,"function"==typeof Error.captureStackTrace&&Error.captureStackTrace(i,t),i}return(0,o.ZT)(t,e),t}(Error),a=function(e){function t(t,n){return e.call(this,r.UNSUPPORTED_FORMATTER,t,n)||this}return(0,o.ZT)(t,e),t}(i),s=function(e){function t(t,n){return e.call(this,r.INVALID_CONFIG,t,n)||this}return(0,o.ZT)(t,e),t}(i),c=function(e){function t(t,n){return e.call(this,r.MISSING_DATA,t,n)||this}return(0,o.ZT)(t,e),t}(i),u=function(e){function t(t,n,o){return e.call(this,r.FORMAT_ERROR,"".concat(t,"\nLocale: ").concat(n,"\n"),o)||this}return(0,o.ZT)(t,e),t}(i),l=function(e){function t(t,n,r,o){var i=e.call(this,"".concat(t,"\nMessageID: ").concat(null==r?void 0:r.id,"\nDefault Message: ").concat(null==r?void 0:r.defaultMessage,"\nDescription: ").concat(null==r?void 0:r.description,"\n"),n,o)||this;return i.descriptor=r,i}return(0,o.ZT)(t,e),t}(u),h=function(e){function t(t,n){var o=e.call(this,r.MISSING_TRANSLATION,'Missing message: "'.concat(t.id,'" for locale "').concat(n,'", using ').concat(t.defaultMessage?"default message (".concat("string"==typeof t.defaultMessage?t.defaultMessage:t.defaultMessage.map((function(e){var t;return null!==(t=e.value)&&void 0!==t?t:JSON.stringify(e)})).join(),")"):"id"," as fallback."))||this;return o.descriptor=t,o}return(0,o.ZT)(t,e),t}(i)},3167:function(e,t,n){"use strict";n.d(t,{L6:function(){return s},Sn:function(){return u},TB:function(){return f},Z0:function(){return c},ax:function(){return h}});var r=n(5542),o=n(1875),i=n(8770),a=n(7309);function s(e,t,n){return void 0===n&&(n={}),t.reduce((function(t,r){return r in e?t[r]=e[r]:r in n&&(t[r]=n[r]),t}),{})}var c={formats:{},messages:{},timeZone:void 0,defaultLocale:"en",defaultFormats:{},fallbackOnEmptyString:!0,onError:function(e){0},onWarn:function(e){0}};function u(){return{dateTime:{},number:{},message:{},relativeTime:{},pluralRules:{},list:{},displayNames:{}}}function l(e){return{create:function(){return{get:function(t){return e[t]},set:function(t,n){e[t]=n}}}}}function h(e){void 0===e&&(e={dateTime:{},number:{},message:{},relativeTime:{},pluralRules:{},list:{},displayNames:{}});var t=Intl.RelativeTimeFormat,n=Intl.ListFormat,a=Intl.DisplayNames,s=(0,i.Z)((function(){for(var e,t=[],n=0;n0?new Intl.Locale(t[0]):new Intl.Locale("string"==typeof e?e:e[0])}},e.__parse=o.Qc,e.formats={number:{integer:{maximumFractionDigits:0},currency:{style:"currency"},percent:{style:"percent"}},date:{short:{month:"numeric",day:"numeric",year:"2-digit"},medium:{month:"short",day:"numeric",year:"numeric"},long:{month:"long",day:"numeric",year:"numeric"},full:{weekday:"long",month:"long",day:"numeric",year:"numeric"}},time:{short:{hour:"numeric",minute:"numeric"},medium:{hour:"numeric",minute:"numeric",second:"numeric"},long:{hour:"numeric",minute:"numeric",second:"numeric",timeZoneName:"short"},full:{hour:"numeric",minute:"numeric",second:"numeric",timeZoneName:"short"}}},e}()},1694:function(e,t,n){"use strict";n.d(t,{C8:function(){return a},HR:function(){return c},YR:function(){return s},jK:function(){return r},u_:function(){return i}});var r,o=n(5542);!function(e){e.MISSING_VALUE="MISSING_VALUE",e.INVALID_VALUE="INVALID_VALUE",e.MISSING_INTL_API="MISSING_INTL_API"}(r||(r={}));var i=function(e){function t(t,n,r){var o=e.call(this,t)||this;return o.code=n,o.originalMessage=r,o}return(0,o.ZT)(t,e),t.prototype.toString=function(){return"[formatjs Error: ".concat(this.code,"] ").concat(this.message)},t}(Error),a=function(e){function t(t,n,o,i){return e.call(this,'Invalid values for "'.concat(t,'": "').concat(n,'". Options are "').concat(Object.keys(o).join('", "'),'"'),r.INVALID_VALUE,i)||this}return(0,o.ZT)(t,e),t}(i),s=function(e){function t(t,n,o){return e.call(this,'Value for "'.concat(t,'" must be of type ').concat(n),r.INVALID_VALUE,o)||this}return(0,o.ZT)(t,e),t}(i),c=function(e){function t(t,n){return e.call(this,'The intl string context variable "'.concat(t,'" was not provided to the string "').concat(n,'"'),r.MISSING_VALUE,n)||this}return(0,o.ZT)(t,e),t}(i)},3528:function(e,t,n){"use strict";n.d(t,{FK:function(){return s},Gt:function(){return a},du:function(){return r}});var r,o=n(7846),i=n(1694);function a(e){return"function"==typeof e}function s(e,t,n,c,u,l,h){if(1===e.length&&(0,o.O4)(e[0]))return[{type:r.literal,value:e[0].value}];for(var f=[],m=0,p=e;m=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n},W=function(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t},V=function(e){var t=!(arguments.length>1&&void 0!==arguments[1])||arguments[1];return!1===t?String(e):String(e).replace(/&/g,"&").replace(//g,">").replace(/"/g,""").replace(/'/g,"'")},X=function(e){var t=q(e,v.TITLE),n=q(e,j);if(n&&t)return n.replace(/%s/g,(function(){return Array.isArray(t)?t.join(""):t}));var r=q(e,P);return t||r||void 0},K=function(e){return q(e,H)||function(){}},Q=function(e,t){return t.filter((function(t){return void 0!==t[e]})).map((function(t){return t[e]})).reduce((function(e,t){return z({},e,t)}),{})},J=function(e,t){return t.filter((function(e){return void 0!==e[v.BASE]})).map((function(e){return e[v.BASE]})).reverse().reduce((function(t,n){if(!t.length)for(var r=Object.keys(n),o=0;o=0;n--){var r=e[n];if(r.hasOwnProperty(t))return r[t]}return null},ee=(r=Date.now(),function(e){var t=Date.now();t-r>16?(r=t,e(t)):setTimeout((function(){ee(e)}),0)}),te=function(e){return clearTimeout(e)},ne="undefined"!=typeof window?window.requestAnimationFrame&&window.requestAnimationFrame.bind(window)||window.webkitRequestAnimationFrame||window.mozRequestAnimationFrame||ee:n.g.requestAnimationFrame||ee,re="undefined"!=typeof window?window.cancelAnimationFrame||window.webkitCancelAnimationFrame||window.mozCancelAnimationFrame||te:n.g.cancelAnimationFrame||te,oe=function(e){return d&&"function"==typeof d.warn&&d.warn(e)},ie=null,ae=function(e,t){var n=e.baseTag,r=e.bodyAttributes,o=e.htmlAttributes,i=e.linkTags,a=e.metaTags,s=e.noscriptTags,c=e.onChangeClientState,u=e.scriptTags,l=e.styleTags,h=e.title,f=e.titleAttributes;ue(v.BODY,r),ue(v.HTML,o),ce(h,f);var m={baseTag:le(v.BASE,n),linkTags:le(v.LINK,i),metaTags:le(v.META,a),noscriptTags:le(v.NOSCRIPT,s),scriptTags:le(v.SCRIPT,u),styleTags:le(v.STYLE,l)},p={},g={};Object.keys(m).forEach((function(e){var t=m[e],n=t.newTags,r=t.oldTags;n.length&&(p[e]=n),r.length&&(g[e]=m[e].oldTags)})),t&&t(),c(e,p,g)},se=function(e){return Array.isArray(e)?e.join(""):e},ce=function(e,t){void 0!==e&&document.title!==e&&(document.title=se(e)),ue(v.TITLE,t)},ue=function(e,t){var n=document.getElementsByTagName(e)[0];if(n){for(var r=n.getAttribute(x),o=r?r.split(","):[],i=[].concat(o),a=Object.keys(t),s=0;s=0;h--)n.removeAttribute(i[h]);o.length===i.length?n.removeAttribute(x):n.getAttribute(x)!==a.join(",")&&n.setAttribute(x,a.join(","))}},le=function(e,t){var n=document.head||document.querySelector(v.HEAD),r=n.querySelectorAll(e+"["+"data-react-helmet]"),o=Array.prototype.slice.call(r),i=[],a=void 0;return t&&t.length&&t.forEach((function(t){var n=document.createElement(e);for(var r in t)if(t.hasOwnProperty(r))if(r===_)n.innerHTML=t.innerHTML;else if(r===E)n.styleSheet?n.styleSheet.cssText=t.cssText:n.appendChild(document.createTextNode(t.cssText));else{var s=void 0===t[r]?"":t[r];n.setAttribute(r,s)}n.setAttribute(x,"true"),o.some((function(e,t){return a=t,n.isEqualNode(e)}))?o.splice(a,1):i.push(n)})),o.forEach((function(e){return e.parentNode.removeChild(e)})),i.forEach((function(e){return n.appendChild(e)})),{oldTags:o,newTags:i}},he=function(e){return Object.keys(e).reduce((function(t,n){var r=void 0!==e[n]?n+'="'+e[n]+'"':""+n;return t?t+" "+r:r}),"")},fe=function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};return Object.keys(e).reduce((function(t,n){return t[D[n]||n]=e[n],t}),t)},me=function(e,t,n){switch(e){case v.TITLE:return{toComponent:function(){return e=t.title,n=t.titleAttributes,(r={key:e})[x]=!0,o=fe(n,r),[m.createElement(v.TITLE,o,e)];var e,n,r,o},toString:function(){return function(e,t,n,r){var o=he(n),i=se(t);return o?"<"+e+' data-react-helmet="true" '+o+">"+V(i,r)+"":"<"+e+' data-react-helmet="true">'+V(i,r)+""}(e,t.title,t.titleAttributes,n)}};case y:case b:return{toComponent:function(){return fe(t)},toString:function(){return he(t)}};default:return{toComponent:function(){return function(e,t){return t.map((function(t,n){var r,o=((r={key:n})[x]=!0,r);return Object.keys(t).forEach((function(e){var n=D[e]||e;if(n===_||n===E){var r=t.innerHTML||t.cssText;o.dangerouslySetInnerHTML={__html:r}}else o[n]=t[e]})),m.createElement(e,o)}))}(e,t)},toString:function(){return function(e,t,n){return t.reduce((function(t,r){var o=Object.keys(r).filter((function(e){return!(e===_||e===E)})).reduce((function(e,t){var o=void 0===r[t]?t:t+'="'+V(r[t],n)+'"';return e?e+" "+o:o}),""),i=r.innerHTML||r.cssText||"",a=-1===F.indexOf(e);return t+"<"+e+' data-react-helmet="true" '+o+(a?"/>":">"+i+"")}),"")}(e,t,n)}}}},pe=function(e){var t=e.baseTag,n=e.bodyAttributes,r=e.encode,o=e.htmlAttributes,i=e.linkTags,a=e.metaTags,s=e.noscriptTags,c=e.scriptTags,u=e.styleTags,l=e.title,h=void 0===l?"":l,f=e.titleAttributes;return{base:me(v.BASE,t,r),bodyAttributes:me(y,n,r),htmlAttributes:me(b,o,r),link:me(v.LINK,i,r),meta:me(v.META,a,r),noscript:me(v.NOSCRIPT,s,r),script:me(v.SCRIPT,c,r),style:me(v.STYLE,u,r),title:me(v.TITLE,{title:h,titleAttributes:f},r)}},ge=l()((function(e){return{baseTag:J([C,O],e),bodyAttributes:Q(y,e),defer:q(e,B),encode:q(e,R),htmlAttributes:Q(b,e),linkTags:$(v.LINK,[L,C],e),metaTags:$(v.META,[M,A,T,N,w],e),noscriptTags:$(v.NOSCRIPT,[_],e),onChangeClientState:K(e),scriptTags:$(v.SCRIPT,[S,_],e),styleTags:$(v.STYLE,[E],e),title:X(e),titleAttributes:Q(I,e)}}),(function(e){ie&&re(ie),e.defer?ie=ne((function(){ae(e,(function(){ie=null}))})):(ae(e),ie=null)}),pe)((function(){return null})),de=(o=ge,a=i=function(e){function t(){return Z(this,t),W(this,e.apply(this,arguments))}return function(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}(t,e),t.prototype.shouldComponentUpdate=function(e){return!f()(this.props,e)},t.prototype.mapNestedChildrenToProps=function(e,t){if(!t)return null;switch(e.type){case v.SCRIPT:case v.NOSCRIPT:return{innerHTML:t};case v.STYLE:return{cssText:t}}throw new Error("<"+e.type+" /> elements are self-closing and can not contain children. Refer to our API for more information.")},t.prototype.flattenArrayTypeChildren=function(e){var t,n=e.child,r=e.arrayTypeChildren,o=e.newChildProps,i=e.nestedChildren;return z({},r,((t={})[n.type]=[].concat(r[n.type]||[],[z({},o,this.mapNestedChildrenToProps(n,i))]),t))},t.prototype.mapObjectTypeChildren=function(e){var t,n,r=e.child,o=e.newProps,i=e.newChildProps,a=e.nestedChildren;switch(r.type){case v.TITLE:return z({},o,((t={})[r.type]=a,t.titleAttributes=z({},i),t));case v.BODY:return z({},o,{bodyAttributes:z({},i)});case v.HTML:return z({},o,{htmlAttributes:z({},i)})}return z({},o,((n={})[r.type]=z({},i),n))},t.prototype.mapArrayTypeChildrenToProps=function(e,t){var n=z({},t);return Object.keys(e).forEach((function(t){var r;n=z({},n,((r={})[t]=e[t],r))})),n},t.prototype.warnOnInvalidChildren=function(e,t){return!0},t.prototype.mapChildrenToProps=function(e,t){var n=this,r={};return m.Children.forEach(e,(function(e){if(e&&e.props){var o=e.props,i=o.children,a=function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};return Object.keys(e).reduce((function(t,n){return t[k[n]||n]=e[n],t}),t)}(Y(o,["children"]));switch(n.warnOnInvalidChildren(e,i),e.type){case v.LINK:case v.META:case v.NOSCRIPT:case v.SCRIPT:case v.STYLE:r=n.flattenArrayTypeChildren({child:e,arrayTypeChildren:r,newChildProps:a,nestedChildren:i});break;default:t=n.mapObjectTypeChildren({child:e,newProps:t,newChildProps:a,nestedChildren:i})}}})),t=this.mapArrayTypeChildrenToProps(r,t)},t.prototype.render=function(){var e=this.props,t=e.children,n=Y(e,["children"]),r=z({},n);return t&&(r=this.mapChildrenToProps(t,r)),m.createElement(o,r)},U(t,null,[{key:"canUseDOM",set:function(e){o.canUseDOM=e}}]),t}(m.Component),i.propTypes={base:c().object,bodyAttributes:c().object,children:c().oneOfType([c().arrayOf(c().node),c().node]),defaultTitle:c().string,defer:c().bool,encodeSpecialCharacters:c().bool,htmlAttributes:c().object,link:c().arrayOf(c().object),meta:c().arrayOf(c().object),noscript:c().arrayOf(c().object),onChangeClientState:c().func,script:c().arrayOf(c().object),style:c().arrayOf(c().object),title:c().string,titleAttributes:c().object,titleTemplate:c().string},i.defaultProps={defer:!0,encodeSpecialCharacters:!0},i.peek=o.peek,i.rewind=function(){var e=o.rewind();return e||(e=pe({baseTag:[],bodyAttributes:{},encodeSpecialCharacters:!0,htmlAttributes:{},linkTags:[],metaTags:[],noscriptTags:[],scriptTags:[],styleTags:[],title:"",titleAttributes:{}})),e},a);de.renderStatic=de.rewind,t.Z=de},6911:function(e,t,n){"use strict";n.d(t,{_y:function(){return a},zt:function(){return i}});var r=n(2784);n(9703);var o=r.createContext(null),i=(o.Consumer,o.Provider),a=o},6563:function(e,t,n){"use strict";n.d(t,{Z:function(){return a}});var r=n(2784),o=n(6911),i=n(3087);function a(){var e=r.useContext(o._y);return(0,i.lq)(e),e}},3087:function(e,t,n){"use strict";n.d(t,{Z0:function(){return c},dt:function(){return u},lq:function(){return s},wU:function(){return l}});var r=n(5542),o=n(2784),i=n(8349),a=n(3167);function s(e){(0,i.kG)(e,"[React Intl] Could not find required `intl` object. needs to exist in the component ancestry.")}var c=(0,r.pi)((0,r.pi)({},a.Z0),{textComponent:o.Fragment});function u(e){return function(t){return e(o.Children.toArray(t))}}function l(e,t){if(e===t)return!0;if(!e||!t)return!1;var n=Object.keys(e),r=Object.keys(t),o=n.length;if(r.length!==o)return!1;for(var i=0;i1&&window.pageYOffset>n;n=window.pageYOffset,r?(e.classList.add("down"),e.classList.remove("up"),null==t||t.classList.add("hidden")):(e.classList.add("up"),e.classList.remove("down"),null==t||t.classList.remove("hidden"))}};document.removeEventListener("scroll",r,{capture:!0,passive:!0}),document.addEventListener("scroll",r,{capture:!0,passive:!0})}(),window.docsearch&&u(),!document.getElementById("algolia-search")){var e=document.createElement("script");e.id="algolia-search";var t=document.createElement("link");e.src=(0,o.withPrefix)("/js/docsearch.js"),e.async=!0,e.onload=function(){var e;window.docsearch&&(u(),t.rel="stylesheet",t.href=(0,o.withPrefix)("/css/docsearch.css"),t.type="text/css",document.body.appendChild(t),null===(e=document.getElementById("search-form"))||void 0===e||e.classList.add("search-enabled"))},document.body.appendChild(e)}}),[]),r.createElement("header",{dir:"ltr"},r.createElement("a",{className:"skip-to-main",href:"#site-content",tabIndex:0},t("skip_to_content")),r.createElement("div",{id:"top-menu",className:"up"},r.createElement("div",{className:"left below-small"},r.createElement(c,{id:"home-page-logo",to:"/","aria-label":"Lingua Franca Home Page"},r.createElement("picture",null,r.createElement("source",{media:"(min-width: 600px)",srcSet:n(2285).Z}),r.createElement("img",{src:n(6732).Z}))),r.createElement("nav",{role:"navigation"},r.createElement("ul",null,r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/download"},t("nav_download"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/docs/"},t("nav_documentation_short"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/docs/handbook/overview"},t("nav_handbook"))),r.createElement("li",{className:"nav-item"},r.createElement(c,{to:"/community"},t("nav_community")))))),r.createElement("div",{className:"right above-small"},r.createElement("div",{className:"search-section"},r.createElement("div",{className:"nav-item"},r.createElement("form",{id:"search-form",className:"search top-nav",role:"search"},r.createElement("svg",{fill:"none",height:"16",viewBox:"0 0 16 6",width:"20",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"m10.5 0c.5052 0 .9922.0651042 1.4609.195312.4688.130209.9063.315105 1.3125.554688.4063.239583.7761.52865 1.1094.86719.3386.33333.6276.70312.8672 1.10937s.4245.84375.5547 1.3125.1953.95573.1953 1.46094-.0651.99219-.1953 1.46094-.3151.90625-.5547 1.3125-.5286.77864-.8672 1.11718c-.3333.33334-.7031.61978-1.1094.85938-.4062.2396-.8437.4245-1.3125.5547-.4687.1302-.9557.1953-1.4609.1953-.65104 0-1.27604-.1094-1.875-.3281-.59375-.2188-1.14062-.5339-1.64062-.94534l-6.132818 6.12504c-.098958.0989-.216145.1484-.351562.1484s-.252604-.0495-.351562-.1484c-.0989588-.099-.148438-.2162-.148438-.3516s.0494792-.2526.148438-.3516l6.125002-6.13278c-.41146-.5-.72656-1.04687-.94532-1.64062-.21874-.59896-.32812-1.22396-.32812-1.875 0-.50521.0651-.99219.19531-1.46094s.31511-.90625.55469-1.3125.52604-.77604.85938-1.10937c.33854-.33854.71093-.627607 1.11718-.86719s.84375-.424479 1.3125-.554688c.46875-.1302078.95573-.195312 1.46094-.195312zm0 10c.6198 0 1.2031-.11719 1.75-.35156.5469-.23959 1.0234-.5625 1.4297-.96875.4062-.40625.7265-.88281.9609-1.42969.2396-.54688.3594-1.13021.3594-1.75s-.1198-1.20312-.3594-1.75c-.2344-.54688-.5547-1.02344-.9609-1.42969-.4063-.40625-.8828-.72656-1.4297-.96093-.5469-.23959-1.1302-.35938-1.75-.35938-.61979 0-1.20312.11979-1.75.35938-.54688.23437-1.02344.55468-1.42969.96093s-.72916.88281-.96875 1.42969c-.23437.54688-.35156 1.13021-.35156 1.75s.11719 1.20312.35156 1.75c.23959.54688.5625 1.02344.96875 1.42969s.88281.72916 1.42969.96875c.54688.23437 1.13021.35156 1.75.35156z",fill:"#fff"})),r.createElement("span",null,r.createElement("input",{id:"search-box-top",type:"search",placeholder:t("nav_search_placeholder"),"aria-label":t("nav_search_aria")})),r.createElement("input",{type:"submit",style:{display:"none"}})))))),r.createElement("div",{id:"site-content"}))},u=n(8760),l=function(){document.documentElement.classList.remove("light-theme"),document.documentElement.classList.add("dark-theme")},h=function(){document.documentElement.classList.remove("dark-theme"),document.documentElement.classList.add("light-theme")},f=function(){var e=(0,s.D)((0,i.Z)()),t="undefined"!=typeof window&&window.matchMedia&&window.matchMedia("(prefers-color-scheme: dark)").matches,n=u.e&&localStorage.getItem("force-color-theme")||"system",o=(0,r.useState)(n),a=o[0],c=o[1],f=u.e&&localStorage.getItem("force-font")||"cascadia",m=(0,r.useState)(f),p=m[0],g=m[1];return r.createElement("section",{id:"customize"},r.createElement("article",null,r.createElement("h3",null,e("footer_customize")),r.createElement("label",null,r.createElement("p",null,e("footer_site_colours"),":"),r.createElement("div",{className:"switch-wrap"},r.createElement("select",{name:"colours",value:a,onChange:function(e){"system"===e.target.value?(t?l():h(),u.e&&localStorage.removeItem("force-color-theme")):"force-light"===e.target.value?(h(),u.e&&localStorage.setItem("force-color-theme","force-light")):"force-dark"===e.target.value&&(l(),u.e&&localStorage.setItem("force-color-theme","force-dark")),c(e.target.value)}},r.createElement("option",{value:"system"},e("footer_site_colours_options_system")),r.createElement("option",{value:"force-light"},e("footer_site_colours_options_always_light")),r.createElement("option",{value:"force-dark"},e("footer_site_colours_options_always_dark"))))),r.createElement("label",null,r.createElement("p",null,e("footer_code_font"),":"),r.createElement("div",{className:"switch-wrap"},r.createElement("select",{name:"font",value:p,onChange:function(e){var t,n;localStorage.setItem("force-font",e.target.value),t=e.target.value,(n=f)&&document.documentElement.classList.remove("font-"+n),document.documentElement.classList.add("font-"+t),g(e.target.value)}},r.createElement("option",{value:"cascadia"},"Cascadia"),r.createElement("option",{value:"cascadia-ligatures"},"Cascadia (ligatures)"),r.createElement("option",{value:"consolas"},"Consolas"),r.createElement("option",{value:"dank-mono"},"Dank Mono"),r.createElement("option",{value:"fira-code"},"Fira Code"),r.createElement("option",{value:"jetbrains-mono"},"JetBrains Mono"),r.createElement("option",{value:"open-dyslexic"},"OpenDyslexic"),r.createElement("option",{value:"sf-mono"},"SF Mono"),r.createElement("option",{value:"source-code-pro"},"Source Code Pro"))))))},m=[{title:"Get Started",url:"/docs/handbook/overview"},{title:"Download",url:"/download"},{title:"Why Lingua Franca",url:"/"},{title:"Publications",url:"/publications-and-presentations"}],p=[{title:"Get Help",url:"/community"},{title:"GitHub Repo",url:"https://github.com/lf-lang/lingua-franca"},{title:"@thelflang",url:"https://twitter.com/thelflang"},{title:"Web Repo",url:"https://github.com/lf-lang/website-lingua-franca"},{title:"Zulip",url:"https://zulip.lf-lang.org"}],g=function(e){var t=m.filter((function(e){return!e.url.includes("#show-examples")})),o=(0,a.i)(e.lang);e.suppressDocRecommendations;return r.createElement("footer",{id:"site-footer",role:"contentinfo"},e.suppressCustomization?null:r.createElement(f,null),r.createElement("section",{id:"community"},r.createElement("article",{id:"logos"},r.createElement("a",{href:""},r.createElement("img",{id:"lf-logo",width:195,height:75,src:n(9808).Z,alt:"Lingua Franca Logo"})),r.createElement("p",null,"Made with ♥ in Berkeley, Dallas, Dresden, Kiel, and Seoul"),r.createElement("p",null,"© 2019-",(new Date).getFullYear()," The Lingua Franca Team",r.createElement("br",null))),r.createElement("article",{id:"using-lf"},r.createElement("h3",null,"Using Lingua Franca"),r.createElement("ul",null,t.map((function(e){return r.createElement("li",{key:e.url},r.createElement(o,{to:e.url},e.title))})))),r.createElement("article",{id:"community-links"},r.createElement("h3",null,"Community"),r.createElement("ul",null,p.map((function(e){var t=function(e){switch(e){case"https://github.com/lf-lang/website-lingua-franca":case"https://github.com/lf-lang/lingua-franca":return r.createElement("svg",{fill:"none",height:"12",viewBox:"0 0 12 12",width:"12",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{clipRule:"evenodd",d:"m6.03927.165405c-3.27055 0-5.922909 2.652005-5.922909 5.923645 0 2.61709 1.697089 4.83705 4.050909 5.62035.29636.0546.40436-.1284.40436-.2854 0-.1408-.00509-.5131-.008-1.0073-1.64763.3578-1.99527-.7942-1.99527-.7942-.26946-.68436-.65782-.86654-.65782-.86654-.53782-.36727.04073-.36001.04073-.36001.59454.04182.90727.61055.90727.61055.52836.90509 1.38655.64364 1.724.492.05382-.38254.20691-.64363.376-.79163-1.31527-.14946-2.69818-.65782-2.69818-2.92764 0-.64654.23091-1.17564.60982-1.58946-.06109-.14981-.26437-.75236.05818-1.56763 0 0 .49709-.15927 1.62872.60727.47237-.13163.97928-.19709 1.48291-.19964.50328.00255 1.00982.06801 1.48291.19964 1.13091-.76654 1.62727-.60727 1.62727-.60727.32328.81527.12001 1.41782.05928 1.56763.37964.41382.60873.94292.60873 1.58946 0 2.27564-1.38509 2.77636-2.70437 2.92291.21237.18291.40182.54436.40182 1.09672 0 .79204-.00727 1.43094-.00727 1.62514 0 .1585.10691.3429.40727.2851 2.35197-.7851 4.04767-3.00369 4.04767-5.62005 0-3.27164-2.6524-5.923645-5.92403-5.923645z",fill:"#ffffff",fillRule:"evenodd"}));case"https://twitter.com/thelflang":return r.createElement("svg",{fill:"none",height:"10",viewBox:"0 0 13 10",width:"13",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"m4.58519 10c4.62962 0 7.16291-3.83919 7.16291-7.16289 0-.10801 0-.21602-.0049-.32403.4909-.35348.918-.80024 1.2568-1.30591-.4517.20128-.9377.33384-1.4483.39766.5204-.30929.9181-.805148 1.1095-1.394284-.486.289658-1.026.495856-1.6004.608773-.4615-.490946-1.11448-.7953322-1.83617-.7953322-1.38938 0-2.51856 1.1291732-2.51856 2.5185532 0 .19638.02455.38785.06383.57441-2.09143-.1031-3.94721-1.10954-5.1893-2.631474-.21602.373119-.33876.805154-.33876 1.266644 0 .87388.44677 1.64467 1.11936 2.09634-.41239-.01473-.80024-.12765-1.13899-.31421v.03437c0 1.21754.86897 2.23871 2.01778 2.46946-.2111.05891-.43203.08837-.66277.08837-.16202 0-.31912-.01473-.47131-.04419.31911 1.00153 1.25191 1.72813 2.35163 1.74777-.86406.67751-1.94906 1.08008-3.12733 1.08008-.20128 0-.402571-.00982-.59895-.03436 1.10954.70696 2.43509 1.12425 3.85393 1.12425z",fill:"#ffffff"}));case"https://zulip.lf-lang.org":return r.createElement("svg",{fill:"none",viewBox:"-30 -10 55 55",xmlns:"http://www.w3.org/2000/svg"},r.createElement("path",{d:"M22.767 3.589c0 1.209-.543 2.283-1.37 2.934l-8.034 7.174c-.149.128-.343-.078-.235-.25l2.946-5.9c.083-.165-.024-.368-.194-.368H4.452c-1.77 0-3.219-1.615-3.219-3.59C1.233 1.616 2.682 0 4.452 0h15.096c1.77-.001 3.219 1.614 3.219 3.589zM4.452 24h15.096c1.77 0 3.219-1.616 3.219-3.59 0-1.974-1.449-3.59-3.219-3.59H8.12c-.17 0-.277-.202-.194-.367l2.946-5.9c.108-.172-.086-.378-.235-.25l-8.033 7.173c-.828.65-1.37 1.725-1.37 2.934 0 1.974 1.448 3.59 3.218 3.59z",fill:"#ffffff"}))}}(e.url),n=t?r.createElement("span",{className:"link-prefix"},t):null;return r.createElement("li",{key:e.url},r.createElement("a",{style:{position:"relative"},href:e.url},n,e.title))}))))))},d=n(7609),y=function(e){var t=Object.assign({},e.ogTags,{"og:title":e.title,"og:description":e.description,"twitter:site":"thelflang"});return r.createElement(r.Fragment,null,r.createElement(d.q,{title:e.title,titleTemplate:"%s"},r.createElement("meta",{name:"description",key:"description",content:e.description}),Object.keys(t).map((function(e){return r.createElement("meta",{key:e,property:e,content:t[e]})}))))},b=function(e){return r.createElement(r.Fragment,null,r.createElement(d.q,{htmlAttributes:{lang:e.lang}},r.createElement("script",{src:"https://polyfill.io/v3/polyfill.min.js?features=es2015%2CArray.prototype.forEach%2CNodeList.prototype.forEach"}),r.createElement("link",{rel:"preload",href:(0,o.withPrefix)("/css/docsearch.css"),as:"style"}),r.createElement("style",null,"\npre data-err {\n background:url(\"data:image/svg+xml,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%206%203'%20enable-background%3D'new%200%200%206%203'%20height%3D'3'%20width%3D'6'%3E%3Cg%20fill%3D'%23c94824'%3E%3Cpolygon%20points%3D'5.5%2C0%202.5%2C3%201.1%2C3%204.1%2C0'%2F%3E%3Cpolygon%20points%3D'4%2C0%206%2C2%206%2C0.6%205.4%2C0'%2F%3E%3Cpolygon%20points%3D'0%2C2%201%2C3%202.4%2C3%200%2C0.6'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E\") repeat-x 0 100%;\n padding-bottom: 3px;\n}")),r.createElement(y,e),r.createElement("div",{className:"ms-Fabric"},r.createElement(c,e),r.createElement("main",{role:"main"},e.children),r.createElement(g,e)))}},9931:function(e,t,n){"use strict";n.r(t),n.d(t,{comCopy:function(){return r}});var r={com_layout_title:"Lingua Franca Community",com_layout_description:"Connect with other LF Programmers online and offline.",com_connect_online:"Online",com_connect_online_description:"Tap into our online resources to learn more about Lingua Franca, provide feedback, connect with our developers, and find out about new updates.",com_online_publications_desc:"View our publications and presentations.",com_online_publications_tag:"",com_online_github_desc:"Found a bug, or want to provide feedback?",com_online_github_href:"Tell us on GitHub.",com_online_zulip_href:"Join the conversation on Zulip.",com_online_zulip_desc:"Have questions, or want to chat with other users?",com_online_twitter_href:"@thelflang",com_online_twitter_desc:"Stay up to date. Follow us on Twitter"}},636:function(e,t,n){"use strict";n.r(t),n.d(t,{docCopy:function(){return r}});var r={doc_layout_title:"Lingua Franca Documentation",doc_layout_description:"Find Lingua Franca starter projects: from Python to C to C++ to TypeScript.",doc_headline:"Learning Resources",doc_headline_handbook_blurb:"The Lingua Franca language reference"}},4158:function(e,t,n){"use strict";n.r(t),n.d(t,{dtCopy:function(){return r}});var r={dt_s_page_title:"Search for typed packages",dt_s_title:"Type Search",dt_s_subtitle:"Find npm packages that have type declarations, either bundled or on Definitely Typed.",dt_s_match:"match",dt_s_matchs:"matches",dt_s_match_exact:"Exact Match",dt_s_popular_on_dt:"Popular on Definitely Typed",dt_s_downloads_short:"DLs",dt_s_downloads_via:"Via",dt_s_module:"Module",dt_s_last_update:"Last Updated",dt_s_install:"Install",dt_s_no_results:"No results found for",dt_s_no_results_try:"Try another search?",dt_s_copy:"copy",dt_s_copied:"copied"}},5437:function(e,t,n){"use strict";n.r(t),n.d(t,{lang:function(){return b},messages:function(){return y}});var r,o,i=n(5542),a=n(2784),s=n(6563);!function(e){e.formatDate="FormattedDate",e.formatTime="FormattedTime",e.formatNumber="FormattedNumber",e.formatList="FormattedList",e.formatDisplayName="FormattedDisplayName"}(r||(r={})),function(e){e.formatDate="FormattedDateParts",e.formatTime="FormattedTimeParts",e.formatNumber="FormattedNumberParts",e.formatList="FormattedListParts"}(o||(o={}));var c=function(e){var t=(0,s.Z)(),n=e.value,r=e.children,o=(0,i._T)(e,["value","children"]);return r(t.formatNumberToParts(n,o))};c.displayName="FormattedNumberParts";function u(e){var t=function(t){var n=(0,s.Z)(),r=t.value,o=t.children,a=(0,i._T)(t,["value","children"]),c="string"==typeof r?new Date(r||0):r;return o("formatDate"===e?n.formatDateToParts(c,a):n.formatTimeToParts(c,a))};return t.displayName=o[e],t}function l(e){var t=function(t){var n=(0,s.Z)(),r=t.value,o=t.children,c=(0,i._T)(t,["value","children"]),u=n[e](r,c);if("function"==typeof o)return o(u);var l=n.textComponent||a.Fragment;return a.createElement(l,null,u)};return t.displayName=r[e],t}c.displayName="FormattedNumberParts";l("formatDate"),l("formatTime"),l("formatNumber"),l("formatList"),l("formatDisplayName"),u("formatDate"),u("formatTime");var h=n(6801),f=n(636),m=n(2990),p=n(9931),g=n(5636),d=n(519),y=Object.assign({},h.navCopy,f.docCopy,p.comCopy,g.handbookCopy,m.indexCopy,d.footerCopy),b=y},519:function(e,t,n){"use strict";n.r(t),n.d(t,{footerCopy:function(){return r}});var r={footer_customize:"Customize",footer_site_colours:"Color Mode",footer_code_font:"Code Font",footer_site_colours_options_system:"System",footer_site_colours_options_always_light:"Always Light",footer_site_colours_options_always_dark:"Always Dark"}},5636:function(e,t,n){"use strict";n.r(t),n.d(t,{handbookCopy:function(){return r}});var r={handb_prev:"Previous",handb_next:"Next",handb_on_this_page:"On this page",handb_like_dislike_title:"Is this page helpful?",handb_like_desc:"Yes",handb_dislike_desc:"No",handb_thanks:"Thanks for the feedback",handb_deprecated_title:"This page has been deprecated",handb_deprecated_subtitle:"This handbook page has been replaced, ",handb_deprecated_subtitle_link:"go to the new page",handb_deprecated_subtitle_action:"Go to new page",handb_experimental_title:"This page contains experimental documentation",handb_experimental_subtitle:"The contents are about a work in progress topic."}},2990:function(e,t,n){"use strict";n.r(t),n.d(t,{indexCopy:function(){return r}});var r={index_2_headline:"Lingua Franca is a polyglot coordination language for reactive, concurrent, and time-sensitive applications.",index_2_byline:"Lingua Franca",index_2_summary:"Lingua Franca (LF) is a polyglot coordination language built to bring deterministic reactive concurrency and time to mainstream target programming languages (currently C, C++, Python, TypeScript, and Rust). LF is supported by a runtime system that is capable of concurrent and distributed execution of reactive programs that are deployable on the Cloud, the Edge, and even on bare-iron embedded platforms.",index_2_detail:"A Lingua Franca program specifies the interactions between components called reactors. The logic of each reactor is written in plain target code. A code generator synthesizes one or more programs in the target language, which are then compiled using standard tool chains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.",index_2_cta_install:"Download Lingua Franca",index_2_cta_install_subtitle:"Version",index_2_cta_install_fallback:"Latest stable release",index_2_cta_download:"On your computer",index_2_cta_download_subtitle:"via Github",index_2_what_is:"What is Lingua Franca?",index_2_what_is_lf:"Reactor-oriented",index_2_what_is_lf_copy:"Reactors are reactive and composable concurrent software components with inputs, outputs, and local state.",index_2_trust:"Concurrent",index_2_trust_copy:"Reactions to events are concurrent unless there is an explicit dependency between them.",index_2_scale:"Deterministic",index_2_scale_copy:"Lingua Franca programs are deterministic by default and therefore easy to test.",index_2_started_title:"Get Started",index_2_started_handbook:"Handbook",index_2_started_handbook_blurb:"Learn the language",index_2_install:"Install Lingua Franca"}},6801:function(e,t,n){"use strict";n.r(t),n.d(t,{navCopy:function(){return r}});var r={skip_to_content:"Skip to main content",nav_documentation:"Documentation",nav_documentation_short:"Docs",nav_download:"Download",nav_community:"Community",nav_handbook:"Handbook",nav_tools:"Tools",nav_search_placeholder:"Search Docs",nav_search_aria:"Search the Lingua Franca site",nav_this_page_in_your_lang:"This page is available in your language",nav_this_page_in_your_lang_open:"Open",nav_this_page_in_your_lang_no_more:"Don't show again"}},9577:function(e,t,n){"use strict";n.r(t),n.d(t,{inYourLanguage:function(){return r}});var r={en:{shorthand:"In En",body:"This page is available in English",open:"Go",cancel:"Don't ask again"}}},2419:function(e,t,n){"use strict";function r(e){return function(t,n){return e.formatMessage({id:t},n)}}n.d(t,{D:function(){return r}})},8349:function(e,t,n){"use strict";n.d(t,{kG:function(){return r}});function r(e,t,n){if(void 0===n&&(n=Error),!e)throw new n(t)}},8770:function(e,t,n){"use strict";function r(e,t){var n=t&&t.cache?t.cache:l,r=t&&t.serializer?t.serializer:c;return(t&&t.strategy?t.strategy:s)(e,{cache:n,serializer:r})}function o(e,t,n,r){var o,i=null==(o=r)||"number"==typeof o||"boolean"==typeof o?r:n(r),a=t.get(i);return void 0===a&&(a=e.call(this,r),t.set(i,a)),a}function i(e,t,n){var r=Array.prototype.slice.call(arguments,3),o=n(r),i=t.get(o);return void 0===i&&(i=e.apply(this,r),t.set(o,i)),i}function a(e,t,n,r,o){return n.bind(t,e,r,o)}function s(e,t){return a(e,this,1===e.length?o:i,t.cache.create(),t.serializer)}n.d(t,{A:function(){return h},Z:function(){return r}});var c=function(){return JSON.stringify(arguments)};function u(){this.cache=Object.create(null)}u.prototype.get=function(e){return this.cache[e]},u.prototype.set=function(e,t){this.cache[e]=t};var l={create:function(){return new u}},h={variadic:function(e,t){return a(e,this,i,t.cache.create(),t.serializer)},monadic:function(e,t){return a(e,this,o,t.cache.create(),t.serializer)}}},7846:function(e,t,n){"use strict";n.d(t,{wD:function(){return i},VG:function(){return c},rp:function(){return l},Ii:function(){return y},O4:function(){return s},uf:function(){return u},Wh:function(){return d},Jo:function(){return m},yx:function(){return p},Wi:function(){return f},HI:function(){return g},pe:function(){return h},Qc:function(){return ie}});var r,o=n(5542);!function(e){e[e.EXPECT_ARGUMENT_CLOSING_BRACE=1]="EXPECT_ARGUMENT_CLOSING_BRACE",e[e.EMPTY_ARGUMENT=2]="EMPTY_ARGUMENT",e[e.MALFORMED_ARGUMENT=3]="MALFORMED_ARGUMENT",e[e.EXPECT_ARGUMENT_TYPE=4]="EXPECT_ARGUMENT_TYPE",e[e.INVALID_ARGUMENT_TYPE=5]="INVALID_ARGUMENT_TYPE",e[e.EXPECT_ARGUMENT_STYLE=6]="EXPECT_ARGUMENT_STYLE",e[e.INVALID_NUMBER_SKELETON=7]="INVALID_NUMBER_SKELETON",e[e.INVALID_DATE_TIME_SKELETON=8]="INVALID_DATE_TIME_SKELETON",e[e.EXPECT_NUMBER_SKELETON=9]="EXPECT_NUMBER_SKELETON",e[e.EXPECT_DATE_TIME_SKELETON=10]="EXPECT_DATE_TIME_SKELETON",e[e.UNCLOSED_QUOTE_IN_ARGUMENT_STYLE=11]="UNCLOSED_QUOTE_IN_ARGUMENT_STYLE",e[e.EXPECT_SELECT_ARGUMENT_OPTIONS=12]="EXPECT_SELECT_ARGUMENT_OPTIONS",e[e.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE=13]="EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE",e[e.INVALID_PLURAL_ARGUMENT_OFFSET_VALUE=14]="INVALID_PLURAL_ARGUMENT_OFFSET_VALUE",e[e.EXPECT_SELECT_ARGUMENT_SELECTOR=15]="EXPECT_SELECT_ARGUMENT_SELECTOR",e[e.EXPECT_PLURAL_ARGUMENT_SELECTOR=16]="EXPECT_PLURAL_ARGUMENT_SELECTOR",e[e.EXPECT_SELECT_ARGUMENT_SELECTOR_FRAGMENT=17]="EXPECT_SELECT_ARGUMENT_SELECTOR_FRAGMENT",e[e.EXPECT_PLURAL_ARGUMENT_SELECTOR_FRAGMENT=18]="EXPECT_PLURAL_ARGUMENT_SELECTOR_FRAGMENT",e[e.INVALID_PLURAL_ARGUMENT_SELECTOR=19]="INVALID_PLURAL_ARGUMENT_SELECTOR",e[e.DUPLICATE_PLURAL_ARGUMENT_SELECTOR=20]="DUPLICATE_PLURAL_ARGUMENT_SELECTOR",e[e.DUPLICATE_SELECT_ARGUMENT_SELECTOR=21]="DUPLICATE_SELECT_ARGUMENT_SELECTOR",e[e.MISSING_OTHER_CLAUSE=22]="MISSING_OTHER_CLAUSE",e[e.INVALID_TAG=23]="INVALID_TAG",e[e.INVALID_TAG_NAME=25]="INVALID_TAG_NAME",e[e.UNMATCHED_CLOSING_TAG=26]="UNMATCHED_CLOSING_TAG",e[e.UNCLOSED_TAG=27]="UNCLOSED_TAG"}(r||(r={}));var i,a;n(1804),n(1715),n(8827);function s(e){return e.type===i.literal}function c(e){return e.type===i.argument}function u(e){return e.type===i.number}function l(e){return e.type===i.date}function h(e){return e.type===i.time}function f(e){return e.type===i.select}function m(e){return e.type===i.plural}function p(e){return e.type===i.pound}function g(e){return e.type===i.tag}function d(e){return!(!e||"object"!=typeof e||e.type!==a.number)}function y(e){return!(!e||"object"!=typeof e||e.type!==a.dateTime)}!function(e){e[e.literal=0]="literal",e[e.argument=1]="argument",e[e.number=2]="number",e[e.date=3]="date",e[e.time=4]="time",e[e.select=5]="select",e[e.plural=6]="plural",e[e.pound=7]="pound",e[e.tag=8]="tag"}(i||(i={})),function(e){e[e.number=0]="number",e[e.dateTime=1]="dateTime"}(a||(a={}));var b=/[ \xA0\u1680\u2000-\u200A\u202F\u205F\u3000]/,I=/(?:[Eec]{1,6}|G{1,5}|[Qq]{1,5}|(?:[yYur]+|U{1,5})|[ML]{1,5}|d{1,2}|D{1,3}|F{1}|[abB]{1,5}|[hkHK]{1,2}|w{1,2}|W{1}|m{1,2}|s{1,2}|[zZOvVxX]{1,4})(?=([^']*'[^']*')*[^']*$)/g;function v(e){var t={};return e.replace(I,(function(e){var n=e.length;switch(e[0]){case"G":t.era=4===n?"long":5===n?"narrow":"short";break;case"y":t.year=2===n?"2-digit":"numeric";break;case"Y":case"u":case"U":case"r":throw new RangeError("`Y/u/U/r` (year) patterns are not supported, use `y` instead");case"q":case"Q":throw new RangeError("`q/Q` (quarter) patterns are not supported");case"M":case"L":t.month=["numeric","2-digit","short","long","narrow"][n-1];break;case"w":case"W":throw new RangeError("`w/W` (week) patterns are not supported");case"d":t.day=["numeric","2-digit"][n-1];break;case"D":case"F":case"g":throw new RangeError("`D/F/g` (day) patterns are not supported, use `d` instead");case"E":t.weekday=4===n?"short":5===n?"narrow":"short";break;case"e":if(n<4)throw new RangeError("`e..eee` (weekday) patterns are not supported");t.weekday=["short","long","narrow","short"][n-4];break;case"c":if(n<4)throw new RangeError("`c..ccc` (weekday) patterns are not supported");t.weekday=["short","long","narrow","short"][n-4];break;case"a":t.hour12=!0;break;case"b":case"B":throw new RangeError("`b/B` (period) patterns are not supported, use `a` instead");case"h":t.hourCycle="h12",t.hour=["numeric","2-digit"][n-1];break;case"H":t.hourCycle="h23",t.hour=["numeric","2-digit"][n-1];break;case"K":t.hourCycle="h11",t.hour=["numeric","2-digit"][n-1];break;case"k":t.hourCycle="h24",t.hour=["numeric","2-digit"][n-1];break;case"j":case"J":case"C":throw new RangeError("`j/J/C` (hour) patterns are not supported, use `h/H/K/k` instead");case"m":t.minute=["numeric","2-digit"][n-1];break;case"s":t.second=["numeric","2-digit"][n-1];break;case"S":case"A":throw new RangeError("`S/A` (second) patterns are not supported, use `s` instead");case"z":t.timeZoneName=n<4?"short":"long";break;case"Z":case"O":case"v":case"V":case"X":case"x":throw new RangeError("`Z/O/v/V/X/x` (timeZone) patterns are not supported, use `z` instead")}return""})),t}var A=/[\t-\r \x85\u200E\u200F\u2028\u2029]/i;var E=/^\.(?:(0+)(\*)?|(#+)|(0+)(#+))$/g,C=/^(@+)?(\+|#+)?[rs]?$/g,T=/(\*)(0+)|(#+)(0+)|(0+)/g,_=/^(0+)$/;function w(e){var t={};return"r"===e[e.length-1]?t.roundingPriority="morePrecision":"s"===e[e.length-1]&&(t.roundingPriority="lessPrecision"),e.replace(C,(function(e,n,r){return"string"!=typeof r?(t.minimumSignificantDigits=n.length,t.maximumSignificantDigits=n.length):"+"===r?t.minimumSignificantDigits=n.length:"#"===n[0]?t.maximumSignificantDigits=n.length:(t.minimumSignificantDigits=n.length,t.maximumSignificantDigits=n.length+("string"==typeof r?r.length:0)),""})),t}function M(e){switch(e){case"sign-auto":return{signDisplay:"auto"};case"sign-accounting":case"()":return{currencySign:"accounting"};case"sign-always":case"+!":return{signDisplay:"always"};case"sign-accounting-always":case"()!":return{signDisplay:"always",currencySign:"accounting"};case"sign-except-zero":case"+?":return{signDisplay:"exceptZero"};case"sign-accounting-except-zero":case"()?":return{signDisplay:"exceptZero",currencySign:"accounting"};case"sign-never":case"+_":return{signDisplay:"never"}}}function N(e){var t;if("E"===e[0]&&"E"===e[1]?(t={notation:"engineering"},e=e.slice(2)):"E"===e[0]&&(t={notation:"scientific"},e=e.slice(1)),t){var n=e.slice(0,2);if("+!"===n?(t.signDisplay="always",e=e.slice(2)):"+?"===n&&(t.signDisplay="exceptZero",e=e.slice(2)),!_.test(e))throw new Error("Malformed concise eng/scientific notation");t.minimumIntegerDigits=e.length}return t}function L(e){var t=M(e);return t||{}}function S(e){for(var t={},n=0,r=e;n1)throw new RangeError("integer-width stems only accept a single optional option");i.options[0].replace(T,(function(e,n,r,o,i,a){if(n)t.minimumIntegerDigits=r.length;else{if(o&&i)throw new Error("We currently do not support maximum integer digits");if(a)throw new Error("We currently do not support exact integer digits")}return""}));continue}if(_.test(i.stem))t.minimumIntegerDigits=i.stem.length;else if(E.test(i.stem)){if(i.options.length>1)throw new RangeError("Fraction-precision stems only accept a single optional option");i.stem.replace(E,(function(e,n,r,o,i,a){return"*"===r?t.minimumFractionDigits=n.length:o&&"#"===o[0]?t.maximumFractionDigits=o.length:i&&a?(t.minimumFractionDigits=i.length,t.maximumFractionDigits=i.length+a.length):(t.minimumFractionDigits=n.length,t.maximumFractionDigits=n.length),""}));var a=i.options[0];"w"===a?t=(0,o.pi)((0,o.pi)({},t),{trailingZeroDisplay:"stripIfInteger"}):a&&(t=(0,o.pi)((0,o.pi)({},t),w(a)))}else if(C.test(i.stem))t=(0,o.pi)((0,o.pi)({},t),w(i.stem));else{var s=M(i.stem);s&&(t=(0,o.pi)((0,o.pi)({},t),s));var c=N(i.stem);c&&(t=(0,o.pi)((0,o.pi)({},t),c))}}return t}var O,D={AX:["H"],BQ:["H"],CP:["H"],CZ:["H"],DK:["H"],FI:["H"],ID:["H"],IS:["H"],ML:["H"],NE:["H"],RU:["H"],SE:["H"],SJ:["H"],SK:["H"],AS:["h","H"],BT:["h","H"],DJ:["h","H"],ER:["h","H"],GH:["h","H"],IN:["h","H"],LS:["h","H"],PG:["h","H"],PW:["h","H"],SO:["h","H"],TO:["h","H"],VU:["h","H"],WS:["h","H"],"001":["H","h"],AL:["h","H","hB"],TD:["h","H","hB"],"ca-ES":["H","h","hB"],CF:["H","h","hB"],CM:["H","h","hB"],"fr-CA":["H","h","hB"],"gl-ES":["H","h","hB"],"it-CH":["H","h","hB"],"it-IT":["H","h","hB"],LU:["H","h","hB"],NP:["H","h","hB"],PF:["H","h","hB"],SC:["H","h","hB"],SM:["H","h","hB"],SN:["H","h","hB"],TF:["H","h","hB"],VA:["H","h","hB"],CY:["h","H","hb","hB"],GR:["h","H","hb","hB"],CO:["h","H","hB","hb"],DO:["h","H","hB","hb"],KP:["h","H","hB","hb"],KR:["h","H","hB","hb"],NA:["h","H","hB","hb"],PA:["h","H","hB","hb"],PR:["h","H","hB","hb"],VE:["h","H","hB","hb"],AC:["H","h","hb","hB"],AI:["H","h","hb","hB"],BW:["H","h","hb","hB"],BZ:["H","h","hb","hB"],CC:["H","h","hb","hB"],CK:["H","h","hb","hB"],CX:["H","h","hb","hB"],DG:["H","h","hb","hB"],FK:["H","h","hb","hB"],GB:["H","h","hb","hB"],GG:["H","h","hb","hB"],GI:["H","h","hb","hB"],IE:["H","h","hb","hB"],IM:["H","h","hb","hB"],IO:["H","h","hb","hB"],JE:["H","h","hb","hB"],LT:["H","h","hb","hB"],MK:["H","h","hb","hB"],MN:["H","h","hb","hB"],MS:["H","h","hb","hB"],NF:["H","h","hb","hB"],NG:["H","h","hb","hB"],NR:["H","h","hb","hB"],NU:["H","h","hb","hB"],PN:["H","h","hb","hB"],SH:["H","h","hb","hB"],SX:["H","h","hb","hB"],TA:["H","h","hb","hB"],ZA:["H","h","hb","hB"],"af-ZA":["H","h","hB","hb"],AR:["H","h","hB","hb"],CL:["H","h","hB","hb"],CR:["H","h","hB","hb"],CU:["H","h","hB","hb"],EA:["H","h","hB","hb"],"es-BO":["H","h","hB","hb"],"es-BR":["H","h","hB","hb"],"es-EC":["H","h","hB","hb"],"es-ES":["H","h","hB","hb"],"es-GQ":["H","h","hB","hb"],"es-PE":["H","h","hB","hb"],GT:["H","h","hB","hb"],HN:["H","h","hB","hb"],IC:["H","h","hB","hb"],KG:["H","h","hB","hb"],KM:["H","h","hB","hb"],LK:["H","h","hB","hb"],MA:["H","h","hB","hb"],MX:["H","h","hB","hb"],NI:["H","h","hB","hb"],PY:["H","h","hB","hb"],SV:["H","h","hB","hb"],UY:["H","h","hB","hb"],JP:["H","h","K"],AD:["H","hB"],AM:["H","hB"],AO:["H","hB"],AT:["H","hB"],AW:["H","hB"],BE:["H","hB"],BF:["H","hB"],BJ:["H","hB"],BL:["H","hB"],BR:["H","hB"],CG:["H","hB"],CI:["H","hB"],CV:["H","hB"],DE:["H","hB"],EE:["H","hB"],FR:["H","hB"],GA:["H","hB"],GF:["H","hB"],GN:["H","hB"],GP:["H","hB"],GW:["H","hB"],HR:["H","hB"],IL:["H","hB"],IT:["H","hB"],KZ:["H","hB"],MC:["H","hB"],MD:["H","hB"],MF:["H","hB"],MQ:["H","hB"],MZ:["H","hB"],NC:["H","hB"],NL:["H","hB"],PM:["H","hB"],PT:["H","hB"],RE:["H","hB"],RO:["H","hB"],SI:["H","hB"],SR:["H","hB"],ST:["H","hB"],TG:["H","hB"],TR:["H","hB"],WF:["H","hB"],YT:["H","hB"],BD:["h","hB","H"],PK:["h","hB","H"],AZ:["H","hB","h"],BA:["H","hB","h"],BG:["H","hB","h"],CH:["H","hB","h"],GE:["H","hB","h"],LI:["H","hB","h"],ME:["H","hB","h"],RS:["H","hB","h"],UA:["H","hB","h"],UZ:["H","hB","h"],XK:["H","hB","h"],AG:["h","hb","H","hB"],AU:["h","hb","H","hB"],BB:["h","hb","H","hB"],BM:["h","hb","H","hB"],BS:["h","hb","H","hB"],CA:["h","hb","H","hB"],DM:["h","hb","H","hB"],"en-001":["h","hb","H","hB"],FJ:["h","hb","H","hB"],FM:["h","hb","H","hB"],GD:["h","hb","H","hB"],GM:["h","hb","H","hB"],GU:["h","hb","H","hB"],GY:["h","hb","H","hB"],JM:["h","hb","H","hB"],KI:["h","hb","H","hB"],KN:["h","hb","H","hB"],KY:["h","hb","H","hB"],LC:["h","hb","H","hB"],LR:["h","hb","H","hB"],MH:["h","hb","H","hB"],MP:["h","hb","H","hB"],MW:["h","hb","H","hB"],NZ:["h","hb","H","hB"],SB:["h","hb","H","hB"],SG:["h","hb","H","hB"],SL:["h","hb","H","hB"],SS:["h","hb","H","hB"],SZ:["h","hb","H","hB"],TC:["h","hb","H","hB"],TT:["h","hb","H","hB"],UM:["h","hb","H","hB"],US:["h","hb","H","hB"],VC:["h","hb","H","hB"],VG:["h","hb","H","hB"],VI:["h","hb","H","hB"],ZM:["h","hb","H","hB"],BO:["H","hB","h","hb"],EC:["H","hB","h","hb"],ES:["H","hB","h","hb"],GQ:["H","hB","h","hb"],PE:["H","hB","h","hb"],AE:["h","hB","hb","H"],"ar-001":["h","hB","hb","H"],BH:["h","hB","hb","H"],DZ:["h","hB","hb","H"],EG:["h","hB","hb","H"],EH:["h","hB","hb","H"],HK:["h","hB","hb","H"],IQ:["h","hB","hb","H"],JO:["h","hB","hb","H"],KW:["h","hB","hb","H"],LB:["h","hB","hb","H"],LY:["h","hB","hb","H"],MO:["h","hB","hb","H"],MR:["h","hB","hb","H"],OM:["h","hB","hb","H"],PH:["h","hB","hb","H"],PS:["h","hB","hb","H"],QA:["h","hB","hb","H"],SA:["h","hB","hb","H"],SD:["h","hB","hb","H"],SY:["h","hB","hb","H"],TN:["h","hB","hb","H"],YE:["h","hB","hb","H"],AF:["H","hb","hB","h"],LA:["H","hb","hB","h"],CN:["H","hB","hb","h"],LV:["H","hB","hb","h"],TL:["H","hB","hb","h"],"zu-ZA":["H","hB","hb","h"],CD:["hB","H"],IR:["hB","H"],"hi-IN":["hB","h","H"],"kn-IN":["hB","h","H"],"ml-IN":["hB","h","H"],"te-IN":["hB","h","H"],KH:["hB","h","H","hb"],"ta-IN":["hB","h","hb","H"],BN:["hb","hB","h","H"],MY:["hb","hB","h","H"],ET:["hB","hb","h","H"],"gu-IN":["hB","hb","h","H"],"mr-IN":["hB","hb","h","H"],"pa-IN":["hB","hb","h","H"],TW:["hB","hb","h","H"],KE:["hB","hb","H","h"],MM:["hB","hb","H","h"],TZ:["hB","hb","H","h"],UG:["hB","hb","H","h"]};function P(e){var t=e.hourCycle;if(void 0===t&&e.hourCycles&&e.hourCycles.length&&(t=e.hourCycles[0]),t)switch(t){case"h24":return"k";case"h23":return"H";case"h12":return"h";case"h11":return"K";default:throw new Error("Invalid hourCycle")}var n,r=e.language;return"root"!==r&&(n=e.maximize().region),(D[n||""]||D[r||""]||D["".concat(r,"-001")]||D["001"])[0]}var B=new RegExp("^".concat(b.source,"*")),R=new RegExp("".concat(b.source,"*$"));function H(e,t){return{start:e,end:t}}var j=!!String.prototype.startsWith,k=!!String.fromCodePoint,F=!!Object.fromEntries,x=!!String.prototype.codePointAt,G=!!String.prototype.trimStart,Z=!!String.prototype.trimEnd,U=!!Number.isSafeInteger?Number.isSafeInteger:function(e){return"number"==typeof e&&isFinite(e)&&Math.floor(e)===e&&Math.abs(e)<=9007199254740991},z=!0;try{z="a"===(null===(O=$("([^\\p{White_Space}\\p{Pattern_Syntax}]*)","yu").exec("a"))||void 0===O?void 0:O[0])}catch(ae){z=!1}var Y,W=j?function(e,t,n){return e.startsWith(t,n)}:function(e,t,n){return e.slice(n,n+t.length)===t},V=k?String.fromCodePoint:function(){for(var e=[],t=0;ti;){if((n=e[i++])>1114111)throw RangeError(n+" is not a valid code point");r+=n<65536?String.fromCharCode(n):String.fromCharCode(55296+((n-=65536)>>10),n%1024+56320)}return r},X=F?Object.fromEntries:function(e){for(var t={},n=0,r=e;n=n)){var r,o=e.charCodeAt(t);return o<55296||o>56319||t+1===n||(r=e.charCodeAt(t+1))<56320||r>57343?o:r-56320+(o-55296<<10)+65536}},Q=G?function(e){return e.trimStart()}:function(e){return e.replace(B,"")},J=Z?function(e){return e.trimEnd()}:function(e){return e.replace(R,"")};function $(e,t){return new RegExp(e,t)}if(z){var q=$("([^\\p{White_Space}\\p{Pattern_Syntax}]*)","yu");Y=function(e,t){var n;return q.lastIndex=t,null!==(n=q.exec(e)[1])&&void 0!==n?n:""}}else Y=function(e,t){for(var n=[];;){var r=K(e,t);if(void 0===r||ne(r)||re(r))break;n.push(r),t+=r>=65536?2:1}return V.apply(void 0,n)};var ee=function(){function e(e,t){void 0===t&&(t={}),this.message=e,this.position={offset:0,line:1,column:1},this.ignoreTag=!!t.ignoreTag,this.locale=t.locale,this.requiresOtherClause=!!t.requiresOtherClause,this.shouldParseSkeletons=!!t.shouldParseSkeletons}return e.prototype.parse=function(){if(0!==this.offset())throw Error("parser can only be used once");return this.parseMessage(0,"",!1)},e.prototype.parseMessage=function(e,t,n){for(var o=[];!this.isEOF();){var a=this.char();if(123===a){if((s=this.parseArgument(e,n)).err)return s;o.push(s.val)}else{if(125===a&&e>0)break;if(35!==a||"plural"!==t&&"selectordinal"!==t){if(60===a&&!this.ignoreTag&&47===this.peek()){if(n)break;return this.error(r.UNMATCHED_CLOSING_TAG,H(this.clonePosition(),this.clonePosition()))}if(60===a&&!this.ignoreTag&&te(this.peek()||0)){if((s=this.parseTag(e,t)).err)return s;o.push(s.val)}else{var s;if((s=this.parseLiteral(e,t)).err)return s;o.push(s.val)}}else{var c=this.clonePosition();this.bump(),o.push({type:i.pound,location:H(c,this.clonePosition())})}}}return{val:o,err:null}},e.prototype.parseTag=function(e,t){var n=this.clonePosition();this.bump();var o=this.parseTagName();if(this.bumpSpace(),this.bumpIf("/>"))return{val:{type:i.literal,value:"<".concat(o,"/>"),location:H(n,this.clonePosition())},err:null};if(this.bumpIf(">")){var a=this.parseMessage(e+1,t,!0);if(a.err)return a;var s=a.val,c=this.clonePosition();if(this.bumpIf("")?{val:{type:i.tag,value:o,children:s,location:H(n,this.clonePosition())},err:null}:this.error(r.INVALID_TAG,H(c,this.clonePosition())))}return this.error(r.UNCLOSED_TAG,H(n,this.clonePosition()))}return this.error(r.INVALID_TAG,H(n,this.clonePosition()))},e.prototype.parseTagName=function(){var e,t=this.offset();for(this.bump();!this.isEOF()&&(45===(e=this.char())||46===e||e>=48&&e<=57||95===e||e>=97&&e<=122||e>=65&&e<=90||183==e||e>=192&&e<=214||e>=216&&e<=246||e>=248&&e<=893||e>=895&&e<=8191||e>=8204&&e<=8205||e>=8255&&e<=8256||e>=8304&&e<=8591||e>=11264&&e<=12271||e>=12289&&e<=55295||e>=63744&&e<=64975||e>=65008&&e<=65533||e>=65536&&e<=983039);)this.bump();return this.message.slice(t,this.offset())},e.prototype.parseLiteral=function(e,t){for(var n=this.clonePosition(),r="";;){var o=this.tryParseQuote(t);if(o)r+=o;else{var a=this.tryParseUnquoted(e,t);if(a)r+=a;else{var s=this.tryParseLeftAngleBracket();if(!s)break;r+=s}}}var c=H(n,this.clonePosition());return{val:{type:i.literal,value:r,location:c},err:null}},e.prototype.tryParseLeftAngleBracket=function(){return this.isEOF()||60!==this.char()||!this.ignoreTag&&(te(e=this.peek()||0)||47===e)?null:(this.bump(),"<");var e},e.prototype.tryParseQuote=function(e){if(this.isEOF()||39!==this.char())return null;switch(this.peek()){case 39:return this.bump(),this.bump(),"'";case 123:case 60:case 62:case 125:break;case 35:if("plural"===e||"selectordinal"===e)break;return null;default:return null}this.bump();var t=[this.char()];for(this.bump();!this.isEOF();){var n=this.char();if(39===n){if(39!==this.peek()){this.bump();break}t.push(39),this.bump()}else t.push(n);this.bump()}return V.apply(void 0,t)},e.prototype.tryParseUnquoted=function(e,t){if(this.isEOF())return null;var n=this.char();return 60===n||123===n||35===n&&("plural"===t||"selectordinal"===t)||125===n&&e>0?null:(this.bump(),V(n))},e.prototype.parseArgument=function(e,t){var n=this.clonePosition();if(this.bump(),this.bumpSpace(),this.isEOF())return this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition()));if(125===this.char())return this.bump(),this.error(r.EMPTY_ARGUMENT,H(n,this.clonePosition()));var o=this.parseIdentifierIfPossible().value;if(!o)return this.error(r.MALFORMED_ARGUMENT,H(n,this.clonePosition()));if(this.bumpSpace(),this.isEOF())return this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition()));switch(this.char()){case 125:return this.bump(),{val:{type:i.argument,value:o,location:H(n,this.clonePosition())},err:null};case 44:return this.bump(),this.bumpSpace(),this.isEOF()?this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(n,this.clonePosition())):this.parseArgumentOptions(e,t,o,n);default:return this.error(r.MALFORMED_ARGUMENT,H(n,this.clonePosition()))}},e.prototype.parseIdentifierIfPossible=function(){var e=this.clonePosition(),t=this.offset(),n=Y(this.message,t),r=t+n.length;return this.bumpTo(r),{value:n,location:H(e,this.clonePosition())}},e.prototype.parseArgumentOptions=function(e,t,n,s){var c,u=this.clonePosition(),l=this.parseIdentifierIfPossible().value,h=this.clonePosition();switch(l){case"":return this.error(r.EXPECT_ARGUMENT_TYPE,H(u,h));case"number":case"date":case"time":this.bumpSpace();var f=null;if(this.bumpIf(",")){this.bumpSpace();var m=this.clonePosition();if((E=this.parseSimpleArgStyleIfPossible()).err)return E;if(0===(y=J(E.val)).length)return this.error(r.EXPECT_ARGUMENT_STYLE,H(this.clonePosition(),this.clonePosition()));f={style:y,styleLocation:H(m,this.clonePosition())}}if((C=this.tryParseArgumentClose(s)).err)return C;var p=H(s,this.clonePosition());if(f&&W(null==f?void 0:f.style,"::",0)){var g=Q(f.style.slice(2));if("number"===l)return(E=this.parseNumberSkeletonFromString(g,f.styleLocation)).err?E:{val:{type:i.number,value:n,location:p,style:E.val},err:null};if(0===g.length)return this.error(r.EXPECT_DATE_TIME_SKELETON,p);var d=g;this.locale&&(d=function(e,t){for(var n="",r=0;r>1),c=P(t);for("H"!=c&&"k"!=c||(s=0);s-- >0;)n+="a";for(;a-- >0;)n=c+n}else n+="J"===o?"H":o}return n}(g,this.locale));var y={type:a.dateTime,pattern:d,location:f.styleLocation,parsedOptions:this.shouldParseSkeletons?v(d):{}};return{val:{type:"date"===l?i.date:i.time,value:n,location:p,style:y},err:null}}return{val:{type:"number"===l?i.number:"date"===l?i.date:i.time,value:n,location:p,style:null!==(c=null==f?void 0:f.style)&&void 0!==c?c:null},err:null};case"plural":case"selectordinal":case"select":var b=this.clonePosition();if(this.bumpSpace(),!this.bumpIf(","))return this.error(r.EXPECT_SELECT_ARGUMENT_OPTIONS,H(b,(0,o.pi)({},b)));this.bumpSpace();var I=this.parseIdentifierIfPossible(),A=0;if("select"!==l&&"offset"===I.value){if(!this.bumpIf(":"))return this.error(r.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE,H(this.clonePosition(),this.clonePosition()));var E;if(this.bumpSpace(),(E=this.tryParseDecimalInteger(r.EXPECT_PLURAL_ARGUMENT_OFFSET_VALUE,r.INVALID_PLURAL_ARGUMENT_OFFSET_VALUE)).err)return E;this.bumpSpace(),I=this.parseIdentifierIfPossible(),A=E.val}var C,T=this.tryParsePluralOrSelectOptions(e,l,t,I);if(T.err)return T;if((C=this.tryParseArgumentClose(s)).err)return C;var _=H(s,this.clonePosition());return"select"===l?{val:{type:i.select,value:n,options:X(T.val),location:_},err:null}:{val:{type:i.plural,value:n,options:X(T.val),offset:A,pluralType:"plural"===l?"cardinal":"ordinal",location:_},err:null};default:return this.error(r.INVALID_ARGUMENT_TYPE,H(u,h))}},e.prototype.tryParseArgumentClose=function(e){return this.isEOF()||125!==this.char()?this.error(r.EXPECT_ARGUMENT_CLOSING_BRACE,H(e,this.clonePosition())):(this.bump(),{val:!0,err:null})},e.prototype.parseSimpleArgStyleIfPossible=function(){for(var e=0,t=this.clonePosition();!this.isEOF();){switch(this.char()){case 39:this.bump();var n=this.clonePosition();if(!this.bumpUntil("'"))return this.error(r.UNCLOSED_QUOTE_IN_ARGUMENT_STYLE,H(n,this.clonePosition()));this.bump();break;case 123:e+=1,this.bump();break;case 125:if(!(e>0))return{val:this.message.slice(t.offset,this.offset()),err:null};e-=1;break;default:this.bump()}}return{val:this.message.slice(t.offset,this.offset()),err:null}},e.prototype.parseNumberSkeletonFromString=function(e,t){var n=[];try{n=function(e){if(0===e.length)throw new Error("Number skeleton cannot be empty");for(var t=[],n=0,r=e.split(A).filter((function(e){return e.length>0}));n=48&&a<=57))break;o=!0,i=10*i+(a-48),this.bump()}var s=H(r,this.clonePosition());return o?U(i*=n)?{val:i,err:null}:this.error(t,s):this.error(e,s)},e.prototype.offset=function(){return this.position.offset},e.prototype.isEOF=function(){return this.offset()===this.message.length},e.prototype.clonePosition=function(){return{offset:this.position.offset,line:this.position.line,column:this.position.column}},e.prototype.char=function(){var e=this.position.offset;if(e>=this.message.length)throw Error("out of bound");var t=K(this.message,e);if(void 0===t)throw Error("Offset ".concat(e," is at invalid UTF-16 code unit boundary"));return t},e.prototype.error=function(e,t){return{val:null,err:{kind:e,message:this.message,location:t}}},e.prototype.bump=function(){if(!this.isEOF()){var e=this.char();10===e?(this.position.line+=1,this.position.column=1,this.position.offset+=1):(this.position.column+=1,this.position.offset+=e<65536?1:2)}},e.prototype.bumpIf=function(e){if(W(this.message,e,this.offset())){for(var t=0;t=0?(this.bumpTo(n),!0):(this.bumpTo(this.message.length),!1)},e.prototype.bumpTo=function(e){if(this.offset()>e)throw Error("targetOffset ".concat(e," must be greater than or equal to the current offset ").concat(this.offset()));for(e=Math.min(e,this.message.length);;){var t=this.offset();if(t===e)break;if(t>e)throw Error("targetOffset ".concat(e," is at invalid UTF-16 code unit boundary"));if(this.bump(),this.isEOF())break}},e.prototype.bumpSpace=function(){for(;!this.isEOF()&&ne(this.char());)this.bump()},e.prototype.peek=function(){if(this.isEOF())return null;var e=this.char(),t=this.offset(),n=this.message.charCodeAt(t+(e>=65536?2:1));return null!=n?n:null},e}();function te(e){return e>=97&&e<=122||e>=65&&e<=90}function ne(e){return e>=9&&e<=13||32===e||133===e||e>=8206&&e<=8207||8232===e||8233===e}function re(e){return e>=33&&e<=35||36===e||e>=37&&e<=39||40===e||41===e||42===e||43===e||44===e||45===e||e>=46&&e<=47||e>=58&&e<=59||e>=60&&e<=62||e>=63&&e<=64||91===e||92===e||93===e||94===e||96===e||123===e||124===e||125===e||126===e||161===e||e>=162&&e<=165||166===e||167===e||169===e||171===e||172===e||174===e||176===e||177===e||182===e||187===e||191===e||215===e||247===e||e>=8208&&e<=8213||e>=8214&&e<=8215||8216===e||8217===e||8218===e||e>=8219&&e<=8220||8221===e||8222===e||8223===e||e>=8224&&e<=8231||e>=8240&&e<=8248||8249===e||8250===e||e>=8251&&e<=8254||e>=8257&&e<=8259||8260===e||8261===e||8262===e||e>=8263&&e<=8273||8274===e||8275===e||e>=8277&&e<=8286||e>=8592&&e<=8596||e>=8597&&e<=8601||e>=8602&&e<=8603||e>=8604&&e<=8607||8608===e||e>=8609&&e<=8610||8611===e||e>=8612&&e<=8613||8614===e||e>=8615&&e<=8621||8622===e||e>=8623&&e<=8653||e>=8654&&e<=8655||e>=8656&&e<=8657||8658===e||8659===e||8660===e||e>=8661&&e<=8691||e>=8692&&e<=8959||e>=8960&&e<=8967||8968===e||8969===e||8970===e||8971===e||e>=8972&&e<=8991||e>=8992&&e<=8993||e>=8994&&e<=9e3||9001===e||9002===e||e>=9003&&e<=9083||9084===e||e>=9085&&e<=9114||e>=9115&&e<=9139||e>=9140&&e<=9179||e>=9180&&e<=9185||e>=9186&&e<=9254||e>=9255&&e<=9279||e>=9280&&e<=9290||e>=9291&&e<=9311||e>=9472&&e<=9654||9655===e||e>=9656&&e<=9664||9665===e||e>=9666&&e<=9719||e>=9720&&e<=9727||e>=9728&&e<=9838||9839===e||e>=9840&&e<=10087||10088===e||10089===e||10090===e||10091===e||10092===e||10093===e||10094===e||10095===e||10096===e||10097===e||10098===e||10099===e||10100===e||10101===e||e>=10132&&e<=10175||e>=10176&&e<=10180||10181===e||10182===e||e>=10183&&e<=10213||10214===e||10215===e||10216===e||10217===e||10218===e||10219===e||10220===e||10221===e||10222===e||10223===e||e>=10224&&e<=10239||e>=10240&&e<=10495||e>=10496&&e<=10626||10627===e||10628===e||10629===e||10630===e||10631===e||10632===e||10633===e||10634===e||10635===e||10636===e||10637===e||10638===e||10639===e||10640===e||10641===e||10642===e||10643===e||10644===e||10645===e||10646===e||10647===e||10648===e||e>=10649&&e<=10711||10712===e||10713===e||10714===e||10715===e||e>=10716&&e<=10747||10748===e||10749===e||e>=10750&&e<=11007||e>=11008&&e<=11055||e>=11056&&e<=11076||e>=11077&&e<=11078||e>=11079&&e<=11084||e>=11085&&e<=11123||e>=11124&&e<=11125||e>=11126&&e<=11157||11158===e||e>=11159&&e<=11263||e>=11776&&e<=11777||11778===e||11779===e||11780===e||11781===e||e>=11782&&e<=11784||11785===e||11786===e||11787===e||11788===e||11789===e||e>=11790&&e<=11798||11799===e||e>=11800&&e<=11801||11802===e||11803===e||11804===e||11805===e||e>=11806&&e<=11807||11808===e||11809===e||11810===e||11811===e||11812===e||11813===e||11814===e||11815===e||11816===e||11817===e||e>=11818&&e<=11822||11823===e||e>=11824&&e<=11833||e>=11834&&e<=11835||e>=11836&&e<=11839||11840===e||11841===e||11842===e||e>=11843&&e<=11855||e>=11856&&e<=11857||11858===e||e>=11859&&e<=11903||e>=12289&&e<=12291||12296===e||12297===e||12298===e||12299===e||12300===e||12301===e||12302===e||12303===e||12304===e||12305===e||e>=12306&&e<=12307||12308===e||12309===e||12310===e||12311===e||12312===e||12313===e||12314===e||12315===e||12316===e||12317===e||e>=12318&&e<=12319||12320===e||12336===e||64830===e||64831===e||e>=65093&&e<=65094}function oe(e){e.forEach((function(e){if(delete e.location,f(e)||m(e))for(var t in e.options)delete e.options[t].location,oe(e.options[t].value);else u(e)&&d(e.style)||(l(e)||h(e))&&y(e.style)?delete e.style.location:g(e)&&oe(e.children)}))}function ie(e,t){void 0===t&&(t={}),t=(0,o.pi)({shouldParseSkeletons:!0,requiresOtherClause:!0},t);var n=new ee(e,t).parse();if(n.err){var i=SyntaxError(r[n.err.kind]);throw i.location=n.err.location,i.originalMessage=n.err.message,i}return(null==t?void 0:t.captureLocation)||oe(n.val),n.val}},7309:function(e,t,n){"use strict";n.d(t,{$6:function(){return h},OV:function(){return s},Qe:function(){return u},Rw:function(){return i},X9:function(){return l},bc:function(){return r},gb:function(){return c},wI:function(){return a}});var r,o=n(5542);!function(e){e.FORMAT_ERROR="FORMAT_ERROR",e.UNSUPPORTED_FORMATTER="UNSUPPORTED_FORMATTER",e.INVALID_CONFIG="INVALID_CONFIG",e.MISSING_DATA="MISSING_DATA",e.MISSING_TRANSLATION="MISSING_TRANSLATION"}(r||(r={}));var i=function(e){function t(n,r,o){var i=this,a=o?o instanceof Error?o:new Error(String(o)):void 0;return(i=e.call(this,"[@formatjs/intl Error ".concat(n,"] ").concat(r,"\n").concat(a?"\n".concat(a.message,"\n").concat(a.stack):""))||this).code=n,"function"==typeof Error.captureStackTrace&&Error.captureStackTrace(i,t),i}return(0,o.ZT)(t,e),t}(Error),a=function(e){function t(t,n){return e.call(this,r.UNSUPPORTED_FORMATTER,t,n)||this}return(0,o.ZT)(t,e),t}(i),s=function(e){function t(t,n){return e.call(this,r.INVALID_CONFIG,t,n)||this}return(0,o.ZT)(t,e),t}(i),c=function(e){function t(t,n){return e.call(this,r.MISSING_DATA,t,n)||this}return(0,o.ZT)(t,e),t}(i),u=function(e){function t(t,n,o){return e.call(this,r.FORMAT_ERROR,"".concat(t,"\nLocale: ").concat(n,"\n"),o)||this}return(0,o.ZT)(t,e),t}(i),l=function(e){function t(t,n,r,o){var i=e.call(this,"".concat(t,"\nMessageID: ").concat(null==r?void 0:r.id,"\nDefault Message: ").concat(null==r?void 0:r.defaultMessage,"\nDescription: ").concat(null==r?void 0:r.description,"\n"),n,o)||this;return i.descriptor=r,i}return(0,o.ZT)(t,e),t}(u),h=function(e){function t(t,n){var o=e.call(this,r.MISSING_TRANSLATION,'Missing message: "'.concat(t.id,'" for locale "').concat(n,'", using ').concat(t.defaultMessage?"default message (".concat("string"==typeof t.defaultMessage?t.defaultMessage:t.defaultMessage.map((function(e){var t;return null!==(t=e.value)&&void 0!==t?t:JSON.stringify(e)})).join(),")"):"id"," as fallback."))||this;return o.descriptor=t,o}return(0,o.ZT)(t,e),t}(i)},3167:function(e,t,n){"use strict";n.d(t,{L6:function(){return s},Sn:function(){return u},TB:function(){return f},Z0:function(){return c},ax:function(){return h}});var r=n(5542),o=n(1875),i=n(8770),a=n(7309);function s(e,t,n){return void 0===n&&(n={}),t.reduce((function(t,r){return r in e?t[r]=e[r]:r in n&&(t[r]=n[r]),t}),{})}var c={formats:{},messages:{},timeZone:void 0,defaultLocale:"en",defaultFormats:{},fallbackOnEmptyString:!0,onError:function(e){0},onWarn:function(e){0}};function u(){return{dateTime:{},number:{},message:{},relativeTime:{},pluralRules:{},list:{},displayNames:{}}}function l(e){return{create:function(){return{get:function(t){return e[t]},set:function(t,n){e[t]=n}}}}}function h(e){void 0===e&&(e={dateTime:{},number:{},message:{},relativeTime:{},pluralRules:{},list:{},displayNames:{}});var t=Intl.RelativeTimeFormat,n=Intl.ListFormat,a=Intl.DisplayNames,s=(0,i.Z)((function(){for(var e,t=[],n=0;n0?new Intl.Locale(t[0]):new Intl.Locale("string"==typeof e?e:e[0])}},e.__parse=o.Qc,e.formats={number:{integer:{maximumFractionDigits:0},currency:{style:"currency"},percent:{style:"percent"}},date:{short:{month:"numeric",day:"numeric",year:"2-digit"},medium:{month:"short",day:"numeric",year:"numeric"},long:{month:"long",day:"numeric",year:"numeric"},full:{weekday:"long",month:"long",day:"numeric",year:"numeric"}},time:{short:{hour:"numeric",minute:"numeric"},medium:{hour:"numeric",minute:"numeric",second:"numeric"},long:{hour:"numeric",minute:"numeric",second:"numeric",timeZoneName:"short"},full:{hour:"numeric",minute:"numeric",second:"numeric",timeZoneName:"short"}}},e}()},1694:function(e,t,n){"use strict";n.d(t,{C8:function(){return a},HR:function(){return c},YR:function(){return s},jK:function(){return r},u_:function(){return i}});var r,o=n(5542);!function(e){e.MISSING_VALUE="MISSING_VALUE",e.INVALID_VALUE="INVALID_VALUE",e.MISSING_INTL_API="MISSING_INTL_API"}(r||(r={}));var i=function(e){function t(t,n,r){var o=e.call(this,t)||this;return o.code=n,o.originalMessage=r,o}return(0,o.ZT)(t,e),t.prototype.toString=function(){return"[formatjs Error: ".concat(this.code,"] ").concat(this.message)},t}(Error),a=function(e){function t(t,n,o,i){return e.call(this,'Invalid values for "'.concat(t,'": "').concat(n,'". Options are "').concat(Object.keys(o).join('", "'),'"'),r.INVALID_VALUE,i)||this}return(0,o.ZT)(t,e),t}(i),s=function(e){function t(t,n,o){return e.call(this,'Value for "'.concat(t,'" must be of type ').concat(n),r.INVALID_VALUE,o)||this}return(0,o.ZT)(t,e),t}(i),c=function(e){function t(t,n){return e.call(this,'The intl string context variable "'.concat(t,'" was not provided to the string "').concat(n,'"'),r.MISSING_VALUE,n)||this}return(0,o.ZT)(t,e),t}(i)},3528:function(e,t,n){"use strict";n.d(t,{FK:function(){return s},Gt:function(){return a},du:function(){return r}});var r,o=n(7846),i=n(1694);function a(e){return"function"==typeof e}function s(e,t,n,c,u,l,h){if(1===e.length&&(0,o.O4)(e[0]))return[{type:r.literal,value:e[0].value}];for(var f=[],m=0,p=e;m=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n},W=function(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t},V=function(e){var t=!(arguments.length>1&&void 0!==arguments[1])||arguments[1];return!1===t?String(e):String(e).replace(/&/g,"&").replace(//g,">").replace(/"/g,""").replace(/'/g,"'")},X=function(e){var t=q(e,v.TITLE),n=q(e,j);if(n&&t)return n.replace(/%s/g,(function(){return Array.isArray(t)?t.join(""):t}));var r=q(e,P);return t||r||void 0},K=function(e){return q(e,H)||function(){}},Q=function(e,t){return t.filter((function(t){return void 0!==t[e]})).map((function(t){return t[e]})).reduce((function(e,t){return z({},e,t)}),{})},J=function(e,t){return t.filter((function(e){return void 0!==e[v.BASE]})).map((function(e){return e[v.BASE]})).reverse().reduce((function(t,n){if(!t.length)for(var r=Object.keys(n),o=0;o=0;n--){var r=e[n];if(r.hasOwnProperty(t))return r[t]}return null},ee=(r=Date.now(),function(e){var t=Date.now();t-r>16?(r=t,e(t)):setTimeout((function(){ee(e)}),0)}),te=function(e){return clearTimeout(e)},ne="undefined"!=typeof window?window.requestAnimationFrame&&window.requestAnimationFrame.bind(window)||window.webkitRequestAnimationFrame||window.mozRequestAnimationFrame||ee:n.g.requestAnimationFrame||ee,re="undefined"!=typeof window?window.cancelAnimationFrame||window.webkitCancelAnimationFrame||window.mozCancelAnimationFrame||te:n.g.cancelAnimationFrame||te,oe=function(e){return d&&"function"==typeof d.warn&&d.warn(e)},ie=null,ae=function(e,t){var n=e.baseTag,r=e.bodyAttributes,o=e.htmlAttributes,i=e.linkTags,a=e.metaTags,s=e.noscriptTags,c=e.onChangeClientState,u=e.scriptTags,l=e.styleTags,h=e.title,f=e.titleAttributes;ue(v.BODY,r),ue(v.HTML,o),ce(h,f);var m={baseTag:le(v.BASE,n),linkTags:le(v.LINK,i),metaTags:le(v.META,a),noscriptTags:le(v.NOSCRIPT,s),scriptTags:le(v.SCRIPT,u),styleTags:le(v.STYLE,l)},p={},g={};Object.keys(m).forEach((function(e){var t=m[e],n=t.newTags,r=t.oldTags;n.length&&(p[e]=n),r.length&&(g[e]=m[e].oldTags)})),t&&t(),c(e,p,g)},se=function(e){return Array.isArray(e)?e.join(""):e},ce=function(e,t){void 0!==e&&document.title!==e&&(document.title=se(e)),ue(v.TITLE,t)},ue=function(e,t){var n=document.getElementsByTagName(e)[0];if(n){for(var r=n.getAttribute(x),o=r?r.split(","):[],i=[].concat(o),a=Object.keys(t),s=0;s=0;h--)n.removeAttribute(i[h]);o.length===i.length?n.removeAttribute(x):n.getAttribute(x)!==a.join(",")&&n.setAttribute(x,a.join(","))}},le=function(e,t){var n=document.head||document.querySelector(v.HEAD),r=n.querySelectorAll(e+"["+"data-react-helmet]"),o=Array.prototype.slice.call(r),i=[],a=void 0;return t&&t.length&&t.forEach((function(t){var n=document.createElement(e);for(var r in t)if(t.hasOwnProperty(r))if(r===_)n.innerHTML=t.innerHTML;else if(r===E)n.styleSheet?n.styleSheet.cssText=t.cssText:n.appendChild(document.createTextNode(t.cssText));else{var s=void 0===t[r]?"":t[r];n.setAttribute(r,s)}n.setAttribute(x,"true"),o.some((function(e,t){return a=t,n.isEqualNode(e)}))?o.splice(a,1):i.push(n)})),o.forEach((function(e){return e.parentNode.removeChild(e)})),i.forEach((function(e){return n.appendChild(e)})),{oldTags:o,newTags:i}},he=function(e){return Object.keys(e).reduce((function(t,n){var r=void 0!==e[n]?n+'="'+e[n]+'"':""+n;return t?t+" "+r:r}),"")},fe=function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};return Object.keys(e).reduce((function(t,n){return t[D[n]||n]=e[n],t}),t)},me=function(e,t,n){switch(e){case v.TITLE:return{toComponent:function(){return e=t.title,n=t.titleAttributes,(r={key:e})[x]=!0,o=fe(n,r),[m.createElement(v.TITLE,o,e)];var e,n,r,o},toString:function(){return function(e,t,n,r){var o=he(n),i=se(t);return o?"<"+e+' data-react-helmet="true" '+o+">"+V(i,r)+"":"<"+e+' data-react-helmet="true">'+V(i,r)+""}(e,t.title,t.titleAttributes,n)}};case y:case b:return{toComponent:function(){return fe(t)},toString:function(){return he(t)}};default:return{toComponent:function(){return function(e,t){return t.map((function(t,n){var r,o=((r={key:n})[x]=!0,r);return Object.keys(t).forEach((function(e){var n=D[e]||e;if(n===_||n===E){var r=t.innerHTML||t.cssText;o.dangerouslySetInnerHTML={__html:r}}else o[n]=t[e]})),m.createElement(e,o)}))}(e,t)},toString:function(){return function(e,t,n){return t.reduce((function(t,r){var o=Object.keys(r).filter((function(e){return!(e===_||e===E)})).reduce((function(e,t){var o=void 0===r[t]?t:t+'="'+V(r[t],n)+'"';return e?e+" "+o:o}),""),i=r.innerHTML||r.cssText||"",a=-1===F.indexOf(e);return t+"<"+e+' data-react-helmet="true" '+o+(a?"/>":">"+i+"")}),"")}(e,t,n)}}}},pe=function(e){var t=e.baseTag,n=e.bodyAttributes,r=e.encode,o=e.htmlAttributes,i=e.linkTags,a=e.metaTags,s=e.noscriptTags,c=e.scriptTags,u=e.styleTags,l=e.title,h=void 0===l?"":l,f=e.titleAttributes;return{base:me(v.BASE,t,r),bodyAttributes:me(y,n,r),htmlAttributes:me(b,o,r),link:me(v.LINK,i,r),meta:me(v.META,a,r),noscript:me(v.NOSCRIPT,s,r),script:me(v.SCRIPT,c,r),style:me(v.STYLE,u,r),title:me(v.TITLE,{title:h,titleAttributes:f},r)}},ge=l()((function(e){return{baseTag:J([C,O],e),bodyAttributes:Q(y,e),defer:q(e,B),encode:q(e,R),htmlAttributes:Q(b,e),linkTags:$(v.LINK,[L,C],e),metaTags:$(v.META,[M,A,T,N,w],e),noscriptTags:$(v.NOSCRIPT,[_],e),onChangeClientState:K(e),scriptTags:$(v.SCRIPT,[S,_],e),styleTags:$(v.STYLE,[E],e),title:X(e),titleAttributes:Q(I,e)}}),(function(e){ie&&re(ie),e.defer?ie=ne((function(){ae(e,(function(){ie=null}))})):(ae(e),ie=null)}),pe)((function(){return null})),de=(o=ge,a=i=function(e){function t(){return Z(this,t),W(this,e.apply(this,arguments))}return function(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}(t,e),t.prototype.shouldComponentUpdate=function(e){return!f()(this.props,e)},t.prototype.mapNestedChildrenToProps=function(e,t){if(!t)return null;switch(e.type){case v.SCRIPT:case v.NOSCRIPT:return{innerHTML:t};case v.STYLE:return{cssText:t}}throw new Error("<"+e.type+" /> elements are self-closing and can not contain children. Refer to our API for more information.")},t.prototype.flattenArrayTypeChildren=function(e){var t,n=e.child,r=e.arrayTypeChildren,o=e.newChildProps,i=e.nestedChildren;return z({},r,((t={})[n.type]=[].concat(r[n.type]||[],[z({},o,this.mapNestedChildrenToProps(n,i))]),t))},t.prototype.mapObjectTypeChildren=function(e){var t,n,r=e.child,o=e.newProps,i=e.newChildProps,a=e.nestedChildren;switch(r.type){case v.TITLE:return z({},o,((t={})[r.type]=a,t.titleAttributes=z({},i),t));case v.BODY:return z({},o,{bodyAttributes:z({},i)});case v.HTML:return z({},o,{htmlAttributes:z({},i)})}return z({},o,((n={})[r.type]=z({},i),n))},t.prototype.mapArrayTypeChildrenToProps=function(e,t){var n=z({},t);return Object.keys(e).forEach((function(t){var r;n=z({},n,((r={})[t]=e[t],r))})),n},t.prototype.warnOnInvalidChildren=function(e,t){return!0},t.prototype.mapChildrenToProps=function(e,t){var n=this,r={};return m.Children.forEach(e,(function(e){if(e&&e.props){var o=e.props,i=o.children,a=function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};return Object.keys(e).reduce((function(t,n){return t[k[n]||n]=e[n],t}),t)}(Y(o,["children"]));switch(n.warnOnInvalidChildren(e,i),e.type){case v.LINK:case v.META:case v.NOSCRIPT:case v.SCRIPT:case v.STYLE:r=n.flattenArrayTypeChildren({child:e,arrayTypeChildren:r,newChildProps:a,nestedChildren:i});break;default:t=n.mapObjectTypeChildren({child:e,newProps:t,newChildProps:a,nestedChildren:i})}}})),t=this.mapArrayTypeChildrenToProps(r,t)},t.prototype.render=function(){var e=this.props,t=e.children,n=Y(e,["children"]),r=z({},n);return t&&(r=this.mapChildrenToProps(t,r)),m.createElement(o,r)},U(t,null,[{key:"canUseDOM",set:function(e){o.canUseDOM=e}}]),t}(m.Component),i.propTypes={base:c().object,bodyAttributes:c().object,children:c().oneOfType([c().arrayOf(c().node),c().node]),defaultTitle:c().string,defer:c().bool,encodeSpecialCharacters:c().bool,htmlAttributes:c().object,link:c().arrayOf(c().object),meta:c().arrayOf(c().object),noscript:c().arrayOf(c().object),onChangeClientState:c().func,script:c().arrayOf(c().object),style:c().arrayOf(c().object),title:c().string,titleAttributes:c().object,titleTemplate:c().string},i.defaultProps={defer:!0,encodeSpecialCharacters:!0},i.peek=o.peek,i.rewind=function(){var e=o.rewind();return e||(e=pe({baseTag:[],bodyAttributes:{},encodeSpecialCharacters:!0,htmlAttributes:{},linkTags:[],metaTags:[],noscriptTags:[],scriptTags:[],styleTags:[],title:"",titleAttributes:{}})),e},a);de.renderStatic=de.rewind,t.Z=de},6911:function(e,t,n){"use strict";n.d(t,{_y:function(){return a},zt:function(){return i}});var r=n(2784);n(9703);var o=r.createContext(null),i=(o.Consumer,o.Provider),a=o},6563:function(e,t,n){"use strict";n.d(t,{Z:function(){return a}});var r=n(2784),o=n(6911),i=n(3087);function a(){var e=r.useContext(o._y);return(0,i.lq)(e),e}},3087:function(e,t,n){"use strict";n.d(t,{Z0:function(){return c},dt:function(){return u},lq:function(){return s},wU:function(){return l}});var r=n(5542),o=n(2784),i=n(8349),a=n(3167);function s(e){(0,i.kG)(e,"[React Intl] Could not find required `intl` object. needs to exist in the component ancestry.")}var c=(0,r.pi)((0,r.pi)({},a.Z0),{textComponent:o.Fragment});function u(e){return function(t){return e(o.Children.toArray(t))}}function l(e,t){if(e===t)return!0;if(!e||!t)return!1;var n=Object.keys(e),r=Object.keys(t),o=n.length;if(r.length!==o)return!1;for(var i=0;i for the\n // default locale.\n onError(new MissingTranslationError(messageDescriptor, locale));\n }\n\n if (defaultMessage) {\n try {\n var formatter = state.getMessageFormat(defaultMessage, defaultLocale, defaultFormats, opts);\n return formatter.format(values);\n } catch (e) {\n onError(new MessageFormatError(\"Error formatting default message for: \\\"\".concat(id, \"\\\", rendering default message verbatim\"), locale, messageDescriptor, e));\n return typeof defaultMessage === 'string' ? defaultMessage : id;\n }\n }\n\n return id;\n } // We have the translated message\n\n\n try {\n var formatter = state.getMessageFormat(message, locale, formats, __assign({\n formatters: state\n }, opts || {}));\n return formatter.format(values);\n } catch (e) {\n onError(new MessageFormatError(\"Error formatting message: \\\"\".concat(id, \"\\\", using \").concat(defaultMessage ? 'default message' : 'id', \" as fallback.\"), locale, messageDescriptor, e));\n }\n\n if (defaultMessage) {\n try {\n var formatter = state.getMessageFormat(defaultMessage, defaultLocale, defaultFormats, opts);\n return formatter.format(values);\n } catch (e) {\n onError(new MessageFormatError(\"Error formatting the default message for: \\\"\".concat(id, \"\\\", rendering message verbatim\"), locale, messageDescriptor, e));\n }\n }\n\n if (typeof message === 'string') {\n return message;\n }\n\n if (typeof defaultMessage === 'string') {\n return defaultMessage;\n }\n\n return id;\n};","import { getNamedFormat, filterProps } from './utils';\nimport { IntlError, IntlErrorCode } from './error';\nvar NUMBER_FORMAT_OPTIONS = ['style', 'currency', 'currencyDisplay', 'unit', 'unitDisplay', 'useGrouping', 'minimumIntegerDigits', 'minimumFractionDigits', 'maximumFractionDigits', 'minimumSignificantDigits', 'maximumSignificantDigits', // ES2020 NumberFormat\n'compactDisplay', 'currencyDisplay', 'currencySign', 'notation', 'signDisplay', 'unit', 'unitDisplay', 'numberingSystem'];\nexport function getFormatter(_a, getNumberFormat, options) {\n var locale = _a.locale,\n formats = _a.formats,\n onError = _a.onError;\n\n if (options === void 0) {\n options = {};\n }\n\n var format = options.format;\n var defaults = format && getNamedFormat(formats, 'number', format, onError) || {};\n var filteredOptions = filterProps(options, NUMBER_FORMAT_OPTIONS, defaults);\n return getNumberFormat(locale, filteredOptions);\n}\nexport function formatNumber(config, getNumberFormat, value, options) {\n if (options === void 0) {\n options = {};\n }\n\n try {\n return getFormatter(config, getNumberFormat, options).format(value);\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting number.', e));\n }\n\n return String(value);\n}\nexport function formatNumberToParts(config, getNumberFormat, value, options) {\n if (options === void 0) {\n options = {};\n }\n\n try {\n return getFormatter(config, getNumberFormat, options).formatToParts(value);\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting number.', e));\n }\n\n return [];\n}","import { getNamedFormat, filterProps } from './utils';\nimport { FormatError, ErrorCode } from 'intl-messageformat';\nimport { IntlFormatError } from './error';\nvar RELATIVE_TIME_FORMAT_OPTIONS = ['numeric', 'style'];\n\nfunction getFormatter(_a, getRelativeTimeFormat, options) {\n var locale = _a.locale,\n formats = _a.formats,\n onError = _a.onError;\n\n if (options === void 0) {\n options = {};\n }\n\n var format = options.format;\n var defaults = !!format && getNamedFormat(formats, 'relative', format, onError) || {};\n var filteredOptions = filterProps(options, RELATIVE_TIME_FORMAT_OPTIONS, defaults);\n return getRelativeTimeFormat(locale, filteredOptions);\n}\n\nexport function formatRelativeTime(config, getRelativeTimeFormat, value, unit, options) {\n if (options === void 0) {\n options = {};\n }\n\n if (!unit) {\n unit = 'second';\n }\n\n var RelativeTimeFormat = Intl.RelativeTimeFormat;\n\n if (!RelativeTimeFormat) {\n config.onError(new FormatError(\"Intl.RelativeTimeFormat is not available in this environment.\\nTry polyfilling it using \\\"@formatjs/intl-relativetimeformat\\\"\\n\", ErrorCode.MISSING_INTL_API));\n }\n\n try {\n return getFormatter(config, getRelativeTimeFormat, options).format(value, unit);\n } catch (e) {\n config.onError(new IntlFormatError('Error formatting relative time.', config.locale, e));\n }\n\n return String(value);\n}","import { __assign } from \"tslib\";\nimport { filterProps, getNamedFormat } from './utils';\nimport { IntlError, IntlErrorCode } from './error';\nvar DATE_TIME_FORMAT_OPTIONS = ['formatMatcher', 'timeZone', 'hour12', 'weekday', 'era', 'year', 'month', 'day', 'hour', 'minute', 'second', 'timeZoneName', 'hourCycle', 'dateStyle', 'timeStyle', 'calendar', // 'dayPeriod',\n'numberingSystem', 'fractionalSecondDigits'];\nexport function getFormatter(_a, type, getDateTimeFormat, options) {\n var locale = _a.locale,\n formats = _a.formats,\n onError = _a.onError,\n timeZone = _a.timeZone;\n\n if (options === void 0) {\n options = {};\n }\n\n var format = options.format;\n\n var defaults = __assign(__assign({}, timeZone && {\n timeZone: timeZone\n }), format && getNamedFormat(formats, type, format, onError));\n\n var filteredOptions = filterProps(options, DATE_TIME_FORMAT_OPTIONS, defaults);\n\n if (type === 'time' && !filteredOptions.hour && !filteredOptions.minute && !filteredOptions.second && !filteredOptions.timeStyle && !filteredOptions.dateStyle) {\n // Add default formatting options if hour, minute, or second isn't defined.\n filteredOptions = __assign(__assign({}, filteredOptions), {\n hour: 'numeric',\n minute: 'numeric'\n });\n }\n\n return getDateTimeFormat(locale, filteredOptions);\n}\nexport function formatDate(config, getDateTimeFormat) {\n var _a = [];\n\n for (var _i = 2; _i < arguments.length; _i++) {\n _a[_i - 2] = arguments[_i];\n }\n\n var value = _a[0],\n _b = _a[1],\n options = _b === void 0 ? {} : _b;\n var date = typeof value === 'string' ? new Date(value || 0) : value;\n\n try {\n return getFormatter(config, 'date', getDateTimeFormat, options).format(date);\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting date.', e));\n }\n\n return String(date);\n}\nexport function formatTime(config, getDateTimeFormat) {\n var _a = [];\n\n for (var _i = 2; _i < arguments.length; _i++) {\n _a[_i - 2] = arguments[_i];\n }\n\n var value = _a[0],\n _b = _a[1],\n options = _b === void 0 ? {} : _b;\n var date = typeof value === 'string' ? new Date(value || 0) : value;\n\n try {\n return getFormatter(config, 'time', getDateTimeFormat, options).format(date);\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting time.', e));\n }\n\n return String(date);\n}\nexport function formatDateTimeRange(config, getDateTimeFormat) {\n var _a = [];\n\n for (var _i = 2; _i < arguments.length; _i++) {\n _a[_i - 2] = arguments[_i];\n }\n\n var from = _a[0],\n to = _a[1],\n _b = _a[2],\n options = _b === void 0 ? {} : _b;\n var timeZone = config.timeZone,\n locale = config.locale,\n onError = config.onError;\n var filteredOptions = filterProps(options, DATE_TIME_FORMAT_OPTIONS, timeZone ? {\n timeZone: timeZone\n } : {});\n\n try {\n return getDateTimeFormat(locale, filteredOptions).formatRange(from, to);\n } catch (e) {\n onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting date time range.', e));\n }\n\n return String(from);\n}\nexport function formatDateToParts(config, getDateTimeFormat) {\n var _a = [];\n\n for (var _i = 2; _i < arguments.length; _i++) {\n _a[_i - 2] = arguments[_i];\n }\n\n var value = _a[0],\n _b = _a[1],\n options = _b === void 0 ? {} : _b;\n var date = typeof value === 'string' ? new Date(value || 0) : value;\n\n try {\n return getFormatter(config, 'date', getDateTimeFormat, options).formatToParts(date); // TODO: remove this when https://github.com/microsoft/TypeScript/pull/50402 is merged\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting date.', e));\n }\n\n return [];\n}\nexport function formatTimeToParts(config, getDateTimeFormat) {\n var _a = [];\n\n for (var _i = 2; _i < arguments.length; _i++) {\n _a[_i - 2] = arguments[_i];\n }\n\n var value = _a[0],\n _b = _a[1],\n options = _b === void 0 ? {} : _b;\n var date = typeof value === 'string' ? new Date(value || 0) : value;\n\n try {\n return getFormatter(config, 'time', getDateTimeFormat, options).formatToParts(date); // TODO: remove this when https://github.com/microsoft/TypeScript/pull/50402 is merged\n } catch (e) {\n config.onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting time.', e));\n }\n\n return [];\n}","import { filterProps } from './utils';\nimport { IntlFormatError } from './error';\nimport { ErrorCode, FormatError } from 'intl-messageformat';\nvar PLURAL_FORMAT_OPTIONS = ['type'];\nexport function formatPlural(_a, getPluralRules, value, options) {\n var locale = _a.locale,\n onError = _a.onError;\n\n if (options === void 0) {\n options = {};\n }\n\n if (!Intl.PluralRules) {\n onError(new FormatError(\"Intl.PluralRules is not available in this environment.\\nTry polyfilling it using \\\"@formatjs/intl-pluralrules\\\"\\n\", ErrorCode.MISSING_INTL_API));\n }\n\n var filteredOptions = filterProps(options, PLURAL_FORMAT_OPTIONS);\n\n try {\n return getPluralRules(locale, filteredOptions).select(value);\n } catch (e) {\n onError(new IntlFormatError('Error formatting plural.', locale, e));\n }\n\n return 'other';\n}","import { __assign } from \"tslib\";\nimport { filterProps } from './utils';\nimport { FormatError, ErrorCode } from 'intl-messageformat';\nimport { IntlError, IntlErrorCode } from './error';\nvar LIST_FORMAT_OPTIONS = ['type', 'style'];\nvar now = Date.now();\n\nfunction generateToken(i) {\n return \"\".concat(now, \"_\").concat(i, \"_\").concat(now);\n}\n\nexport function formatList(opts, getListFormat, values, options) {\n if (options === void 0) {\n options = {};\n }\n\n var results = formatListToParts(opts, getListFormat, values, options).reduce(function (all, el) {\n var val = el.value;\n\n if (typeof val !== 'string') {\n all.push(val);\n } else if (typeof all[all.length - 1] === 'string') {\n all[all.length - 1] += val;\n } else {\n all.push(val);\n }\n\n return all;\n }, []);\n return results.length === 1 ? results[0] : results.length === 0 ? '' : results;\n}\nexport function formatListToParts(_a, getListFormat, values, options) {\n var locale = _a.locale,\n onError = _a.onError;\n\n if (options === void 0) {\n options = {};\n }\n\n var ListFormat = Intl.ListFormat;\n\n if (!ListFormat) {\n onError(new FormatError(\"Intl.ListFormat is not available in this environment.\\nTry polyfilling it using \\\"@formatjs/intl-listformat\\\"\\n\", ErrorCode.MISSING_INTL_API));\n }\n\n var filteredOptions = filterProps(options, LIST_FORMAT_OPTIONS);\n\n try {\n var richValues_1 = {};\n var serializedValues = values.map(function (v, i) {\n if (typeof v === 'object') {\n var id = generateToken(i);\n richValues_1[id] = v;\n return id;\n }\n\n return String(v);\n });\n return getListFormat(locale, filteredOptions).formatToParts(serializedValues).map(function (part) {\n return part.type === 'literal' ? part : __assign(__assign({}, part), {\n value: richValues_1[part.value] || part.value\n });\n });\n } catch (e) {\n onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting list.', e));\n } // @ts-ignore\n\n\n return values;\n}","import { filterProps } from './utils';\nimport { FormatError, ErrorCode } from 'intl-messageformat';\nimport { IntlErrorCode, IntlError } from './error';\nvar DISPLAY_NAMES_OPTONS = ['style', 'type', 'fallback'];\nexport function formatDisplayName(_a, getDisplayNames, value, options) {\n var locale = _a.locale,\n onError = _a.onError;\n var DisplayNames = Intl.DisplayNames;\n\n if (!DisplayNames) {\n onError(new FormatError(\"Intl.DisplayNames is not available in this environment.\\nTry polyfilling it using \\\"@formatjs/intl-displaynames\\\"\\n\", ErrorCode.MISSING_INTL_API));\n }\n\n var filteredOptions = filterProps(options, DISPLAY_NAMES_OPTONS);\n\n try {\n return getDisplayNames(locale, filteredOptions).of(value);\n } catch (e) {\n onError(new IntlError(IntlErrorCode.FORMAT_ERROR, 'Error formatting display name.', e));\n }\n}","import { __assign } from \"tslib\";\nimport { createFormatters, DEFAULT_INTL_CONFIG } from './utils';\nimport { InvalidConfigError, MissingDataError } from './error';\nimport { formatNumber, formatNumberToParts } from './number';\nimport { formatRelativeTime } from './relativeTime';\nimport { formatDate, formatDateToParts, formatTime, formatTimeToParts, formatDateTimeRange } from './dateTime';\nimport { formatPlural } from './plural';\nimport { formatMessage } from './message';\nimport { formatList, formatListToParts } from './list';\nimport { formatDisplayName } from './displayName';\n\nfunction messagesContainString(messages) {\n var firstMessage = messages ? messages[Object.keys(messages)[0]] : undefined;\n return typeof firstMessage === 'string';\n}\n\nfunction verifyConfigMessages(config) {\n if (config.onWarn && config.defaultRichTextElements && messagesContainString(config.messages || {})) {\n config.onWarn(\"[@formatjs/intl] \\\"defaultRichTextElements\\\" was specified but \\\"message\\\" was not pre-compiled. \\nPlease consider using \\\"@formatjs/cli\\\" to pre-compile your messages for performance.\\nFor more details see https://formatjs.io/docs/getting-started/message-distribution\");\n }\n}\n/**\n * Create intl object\n * @param config intl config\n * @param cache cache for formatter instances to prevent memory leak\n */\n\n\nexport function createIntl(config, cache) {\n var formatters = createFormatters(cache);\n\n var resolvedConfig = __assign(__assign({}, DEFAULT_INTL_CONFIG), config);\n\n var locale = resolvedConfig.locale,\n defaultLocale = resolvedConfig.defaultLocale,\n onError = resolvedConfig.onError;\n\n if (!locale) {\n if (onError) {\n onError(new InvalidConfigError(\"\\\"locale\\\" was not configured, using \\\"\".concat(defaultLocale, \"\\\" as fallback. See https://formatjs.io/docs/react-intl/api#intlshape for more details\")));\n } // Since there's no registered locale data for `locale`, this will\n // fallback to the `defaultLocale` to make sure things can render.\n // The `messages` are overridden to the `defaultProps` empty object\n // to maintain referential equality across re-renders. It's assumed\n // each contains a `defaultMessage` prop.\n\n\n resolvedConfig.locale = resolvedConfig.defaultLocale || 'en';\n } else if (!Intl.NumberFormat.supportedLocalesOf(locale).length && onError) {\n onError(new MissingDataError(\"Missing locale data for locale: \\\"\".concat(locale, \"\\\" in Intl.NumberFormat. Using default locale: \\\"\").concat(defaultLocale, \"\\\" as fallback. See https://formatjs.io/docs/react-intl#runtime-requirements for more details\")));\n } else if (!Intl.DateTimeFormat.supportedLocalesOf(locale).length && onError) {\n onError(new MissingDataError(\"Missing locale data for locale: \\\"\".concat(locale, \"\\\" in Intl.DateTimeFormat. Using default locale: \\\"\").concat(defaultLocale, \"\\\" as fallback. See https://formatjs.io/docs/react-intl#runtime-requirements for more details\")));\n }\n\n verifyConfigMessages(resolvedConfig);\n return __assign(__assign({}, resolvedConfig), {\n formatters: formatters,\n formatNumber: formatNumber.bind(null, resolvedConfig, formatters.getNumberFormat),\n formatNumberToParts: formatNumberToParts.bind(null, resolvedConfig, formatters.getNumberFormat),\n formatRelativeTime: formatRelativeTime.bind(null, resolvedConfig, formatters.getRelativeTimeFormat),\n formatDate: formatDate.bind(null, resolvedConfig, formatters.getDateTimeFormat),\n formatDateToParts: formatDateToParts.bind(null, resolvedConfig, formatters.getDateTimeFormat),\n formatTime: formatTime.bind(null, resolvedConfig, formatters.getDateTimeFormat),\n formatDateTimeRange: formatDateTimeRange.bind(null, resolvedConfig, formatters.getDateTimeFormat),\n formatTimeToParts: formatTimeToParts.bind(null, resolvedConfig, formatters.getDateTimeFormat),\n formatPlural: formatPlural.bind(null, resolvedConfig, formatters.getPluralRules),\n // @ts-expect-error TODO: will get to this later\n formatMessage: formatMessage.bind(null, resolvedConfig, formatters),\n // @ts-expect-error TODO: will get to this later\n $t: formatMessage.bind(null, resolvedConfig, formatters),\n formatList: formatList.bind(null, resolvedConfig, formatters.getListFormat),\n formatListToParts: formatListToParts.bind(null, resolvedConfig, formatters.getListFormat),\n formatDisplayName: formatDisplayName.bind(null, resolvedConfig, formatters.getDisplayNames)\n });\n}","/*\n * Copyright 2015, Yahoo Inc.\n * Copyrights licensed under the New BSD License.\n * See the accompanying LICENSE file for terms.\n */\nimport { __assign, __extends, __rest, __spreadArray } from \"tslib\";\nimport * as React from 'react';\nimport { Provider } from './injectIntl';\nimport { DEFAULT_INTL_CONFIG, invariantIntlContext, assignUniqueKeysToParts, shallowEqual } from '../utils';\nimport { formatMessage as coreFormatMessage, createIntl as coreCreateIntl, createIntlCache } from '@formatjs/intl';\nimport { isFormatXMLElementFn } from 'intl-messageformat';\n\nfunction processIntlConfig(config) {\n return {\n locale: config.locale,\n timeZone: config.timeZone,\n fallbackOnEmptyString: config.fallbackOnEmptyString,\n formats: config.formats,\n textComponent: config.textComponent,\n messages: config.messages,\n defaultLocale: config.defaultLocale,\n defaultFormats: config.defaultFormats,\n onError: config.onError,\n onWarn: config.onWarn,\n wrapRichTextChunksInFragment: config.wrapRichTextChunksInFragment,\n defaultRichTextElements: config.defaultRichTextElements\n };\n}\n\nfunction assignUniqueKeysToFormatXMLElementFnArgument(values) {\n if (!values) {\n return values;\n }\n\n return Object.keys(values).reduce(function (acc, k) {\n var v = values[k];\n acc[k] = isFormatXMLElementFn(v) ? assignUniqueKeysToParts(v) : v;\n return acc;\n }, {});\n}\n\nvar formatMessage = function formatMessage(config, formatters, descriptor, rawValues) {\n var rest = [];\n\n for (var _i = 4; _i < arguments.length; _i++) {\n rest[_i - 4] = arguments[_i];\n }\n\n var values = assignUniqueKeysToFormatXMLElementFnArgument(rawValues);\n var chunks = coreFormatMessage.apply(void 0, __spreadArray([config, formatters, descriptor, values], rest, false));\n\n if (Array.isArray(chunks)) {\n return React.Children.toArray(chunks);\n }\n\n return chunks;\n};\n/**\n * Create intl object\n * @param config intl config\n * @param cache cache for formatter instances to prevent memory leak\n */\n\n\nexport var createIntl = function createIntl(_a, cache) {\n var rawDefaultRichTextElements = _a.defaultRichTextElements,\n config = __rest(_a, [\"defaultRichTextElements\"]);\n\n var defaultRichTextElements = assignUniqueKeysToFormatXMLElementFnArgument(rawDefaultRichTextElements);\n var coreIntl = coreCreateIntl(__assign(__assign(__assign({}, DEFAULT_INTL_CONFIG), config), {\n defaultRichTextElements: defaultRichTextElements\n }), cache);\n var resolvedConfig = {\n locale: coreIntl.locale,\n timeZone: coreIntl.timeZone,\n fallbackOnEmptyString: coreIntl.fallbackOnEmptyString,\n formats: coreIntl.formats,\n defaultLocale: coreIntl.defaultLocale,\n defaultFormats: coreIntl.defaultFormats,\n messages: coreIntl.messages,\n onError: coreIntl.onError,\n defaultRichTextElements: defaultRichTextElements\n };\n return __assign(__assign({}, coreIntl), {\n // @ts-expect-error fix this\n formatMessage: formatMessage.bind(null, resolvedConfig, coreIntl.formatters),\n // @ts-expect-error fix this\n $t: formatMessage.bind(null, resolvedConfig, coreIntl.formatters)\n });\n};\n\nvar IntlProvider =\n/** @class */\nfunction (_super) {\n __extends(IntlProvider, _super);\n\n function IntlProvider() {\n var _this = _super !== null && _super.apply(this, arguments) || this;\n\n _this.cache = createIntlCache();\n _this.state = {\n cache: _this.cache,\n intl: createIntl(processIntlConfig(_this.props), _this.cache),\n prevConfig: processIntlConfig(_this.props)\n };\n return _this;\n }\n\n IntlProvider.getDerivedStateFromProps = function (props, _a) {\n var prevConfig = _a.prevConfig,\n cache = _a.cache;\n var config = processIntlConfig(props);\n\n if (!shallowEqual(prevConfig, config)) {\n return {\n intl: createIntl(config, cache),\n prevConfig: config\n };\n }\n\n return null;\n };\n\n IntlProvider.prototype.render = function () {\n invariantIntlContext(this.state.intl);\n return React.createElement(Provider, {\n value: this.state.intl\n }, this.props.children);\n };\n\n IntlProvider.displayName = 'IntlProvider';\n IntlProvider.defaultProps = DEFAULT_INTL_CONFIG;\n return IntlProvider;\n}(React.PureComponent);\n\nexport default IntlProvider;","import * as React from \"react\"\nimport { IntlProvider } from 'react-intl';\n\ntype IntlProps = {\n locale: string\n children: any\n}\n\nexport const Intl = (props: IntlProps) => {\n const { children, locale } = props\n let messages = require(\"../copy/en/en\").lang\n try {\n messages = require(\"../copy/\" + locale + \"/\" + locale).lang\n } catch (error) {\n // NOOP\n }\n return (\n \n {children}\n \n )\n}\n","\n// Generated during bootstapping via pathsOnSiteTracker.ts\n \nexport const allFiles = [\"/docs/handbook/contributing\",\n\"/docs/handbook/eclipse-oomph\",\n\"/docs/handbook/intellij\",\n\"/docs/handbook/running-benchmarks\",\n\"/docs/handbook/website-development\",\n\"/docs/handbook/developer-setup\",\n\"/docs/handbook/regression-tests\",\n\"/docs/handbook/arduino\",\n\"/docs/handbook/proof-import\",\n\"/docs/handbook/logical-execution-time\",\n\"/docs/handbook/related-work\",\n\"/docs/handbook/timing-analysis\",\n\"/docs/handbook/tools\",\n\"/docs/handbook/language-specification\",\n\"/docs/handbook/generic-types-interfaces-inheritance\",\n\"/docs/handbook/import-system\",\n\"/docs/handbook/reactors-on-patmos\",\n\"/docs/handbook/features\",\n\"/docs/handbook/containerized-execution\",\n\"/docs/handbook/expressions\",\n\"/docs/handbook/security\",\n\"/docs/handbook/zephyr\",\n\"/docs/handbook/target-declaration\",\n\"/docs/handbook/code-extension\",\n\"/docs/handbook/tracing\",\n\"/docs/handbook/a-first-reactor\",\n\"/docs/handbook/actions\",\n\"/docs/handbook/causality-loops\",\n\"/docs/handbook/composing-reactors\",\n\"/docs/handbook/deadlines\",\n\"/docs/handbook/command-line-tools\",\n\"/docs/handbook/distributed-execution\",\n\"/docs/handbook/extending-reactors\",\n\"/docs/handbook/generics\",\n\"/docs/handbook/troubleshooting\",\n\"/docs/handbook/epoch-ide\",\n\"/docs/handbook/inputs-and-outputs\",\n\"/docs/handbook/methods\",\n\"/docs/handbook/modal-models\",\n\"/docs/handbook/multiports-and-banks\",\n\"/docs/handbook/overview\",\n\"/docs/handbook/parameters-and-state-variables\",\n\"/docs/handbook/preambles\",\n\"/docs/handbook/reaction-declarations\",\n\"/docs/handbook/reactions\",\n\"/docs/handbook/superdense-time\",\n\"/docs/handbook/termination\",\n\"/docs/handbook/time-and-timers\",\n\"/docs/handbook/tutorial-video\",\n\"/docs/handbook/target-language-details\",\n\"/community\",\n\"/download\",\n\"/empty\",\n\"/\",\n\"/publications-and-presentations\",\n\"/docs/\",\n\"/docs/handbook/\",]","import * as React from \"react\"\nimport { GatsbyLinkProps, Link } from \"gatsby\"\nimport { allFiles } from \"../__generated__/allPages\"\n\n/** \n * Creates a which supports gradual migration, you provide a link to the english page and\n * if the page supports the same version but in your language, it opts for that.\n */\nexport const createIntlLink = (currentLocale: string) => {\n const paths = allFiles\n\n return (linkProps: GatsbyLinkProps<{}>) => {\n let to = linkProps.to\n\n // /thing -> /ja/thing\n // This occurs when we want URL compat with old site\n\n const localeVersion = \"/\" + currentLocale + to\n if (currentLocale !== \"en\" && paths.includes(localeVersion)) {\n to = localeVersion\n }\n\n // This effectively needs to be duplicated in gatsby-config.js too\n const blocklistIncludes = [\"/play\", \"sandbox\", \"/dev\"]\n const blocklisted = blocklistIncludes.find(blocked => to.includes(blocked))\n\n if (blocklisted) {\n // @ts-ignore\n return \n } else {\n // @ts-ignore\n return \n }\n }\n}\n\n\n","import React, { useEffect } from \"react\"\nimport { withPrefix } from \"gatsby\"\n\nimport \"./TopNav.scss\"\nimport { useIntl } from \"react-intl\";\nimport { createIntlLink } from \"../IntlLink\";\nimport { setupStickyNavigation } from \"./stickyNavigation\";\n\nexport type Props = {\n lang: string\n}\n\nimport { navCopy } from \"../../copy/en/nav\"\nimport { createInternational } from \"../../lib/createInternational\"\n\nexport const SiteNav = (props: Props) => {\n const i = createInternational(useIntl())\n const IntlLink = createIntlLink(props.lang)\n const loadDocSearch = () => {\n const isDev = document.location.host.includes('localhost')\n let customHandleSelected;\n\n if (isDev) {\n customHandleSelected = (input, event, suggestion, datasetNumber, context) => {\n const urlToOpen = suggestion.url.replace(\"www.lf-lang.org\", \"localhost:8000\").replace(\"https\", \"http\")\n window.open(urlToOpen)\n }\n }\n\n // @ts-ignore - this comes from the script above\n docsearch({\n appId: \"U5IWIJW31Z\",\n apiKey: '0e594ee25f8bd7d73ae16a1af7076554',\n indexName: 'lf-lang',\n inputSelector: '.search input',\n handleSelected: customHandleSelected,\n });\n }\n // This extra bit of mis-direction ensures that non-essential code runs after\n // the page is loaded\n useEffect(() => {\n setupStickyNavigation()\n\n // @ts-ignore - this comes from the script above\n if (window.docsearch) {\n loadDocSearch();\n }\n if (document.getElementById(\"algolia-search\")) return\n\n const searchScript = document.createElement('script');\n searchScript.id = \"algolia-search\"\n const searchCSS = document.createElement('link');\n\n searchScript.src = withPrefix(\"/js/docsearch.js\");\n searchScript.async = true;\n searchScript.onload = () => {\n // @ts-ignore - this comes from the script above\n if (window.docsearch) {\n loadDocSearch();\n\n searchCSS.rel = 'stylesheet';\n searchCSS.href = withPrefix('/css/docsearch.css');\n searchCSS.type = 'text/css';\n document.body.appendChild(searchCSS);\n\n document.getElementById(\"search-form\")?.classList.add(\"search-enabled\")\n }\n }\n\n document.body.appendChild(searchScript);\n }, []);\n return (\n
\n {i(\"skip_to_content\")}\n\n
\n
\n\n \n \n \n \n \n \n\n \n\n
\n
\n
\n
\n
\n \n \n \n
\n
\n
\n
\n
\n\n
\n
\n )\n}\n","export function setupStickyNavigation() {\n const nav = document.getElementById(\"top-menu\")\n if (!nav) throw new Error(\"Didn't find a nav\")\n\n const sideButton = document.getElementById(\"small-device-button-sidebar\")\n let previousY = 9999\n\n const updateNav = () => {\n // iOS scrolls to make sure the viewport fits, don't hide the input then\n const hasKeyboardFocus =\n document.activeElement &&\n (document.activeElement.nodeName === \"INPUT\" ||\n document.activeElement.nodeName === \"TEXTAREA\")\n\n if (hasKeyboardFocus) {\n return\n }\n\n const showNav = () => {\n nav.classList.add(\"down\")\n nav.classList.remove(\"up\")\n sideButton?.classList.add(\"hidden\")\n }\n\n const hideNav = () => {\n nav.classList.add(\"up\")\n nav.classList.remove(\"down\")\n sideButton?.classList.remove(\"hidden\")\n }\n\n const goingUp = window.pageYOffset > 1 && window.pageYOffset > previousY\n previousY = window.pageYOffset\n\n if (goingUp) {\n showNav()\n } else {\n hideNav()\n }\n }\n\n // Non-blocking nav change\n document.removeEventListener(\"scroll\", updateNav, {\n capture: true,\n passive: true,\n } as any)\n\n document.addEventListener(\"scroll\", updateNav, {\n capture: true,\n passive: true,\n })\n}\n","import * as React from \"react\"\nimport { useState } from \"react\"\nimport { hasLocalStorage } from \"../../lib/hasLocalStorage\"\n\nimport { useIntl } from \"react-intl\";\n\nimport { footerCopy } from \"../../copy/en/footer\"\nimport { createInternational } from \"../../lib/createInternational\"\n\nconst makeDark = () => {\n document.documentElement.classList.remove(\"light-theme\")\n document.documentElement.classList.add(\"dark-theme\")\n}\n\nconst makeLight = () => {\n document.documentElement.classList.remove(\"dark-theme\")\n document.documentElement.classList.add(\"light-theme\")\n}\n\nconst switchFont = (newStyle: string, old?: string) => {\n if (old) document.documentElement.classList.remove(\"font-\" + old)\n document.documentElement.classList.add(\"font-\" + newStyle)\n}\n\nexport const Customize = () => {\n const i = createInternational(useIntl())\n const systemIsDark = typeof window !== 'undefined' && window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches\n const customThemeOverride = hasLocalStorage && localStorage.getItem(\"force-color-theme\") || \"system\"\n const [darkModeValue, setDarkMode] = useState(customThemeOverride)\n\n const customFontOverride = hasLocalStorage && localStorage.getItem(\"force-font\") || \"cascadia\"\n const [fontValue, setFont] = useState(customFontOverride)\n\n // Localstorage: force-dark, force-light, undefined \n // ->\n // CSS Body class: theme-dark, theme-light, theme-dark | theme-light \n\n const handleThemeChange = (event: React.ChangeEvent) => {\n if (event.target.value === \"system\") {\n systemIsDark ? makeDark() : makeLight()\n hasLocalStorage && localStorage.removeItem(\"force-color-theme\")\n } else if (event.target.value === \"force-light\") {\n makeLight()\n hasLocalStorage && localStorage.setItem(\"force-color-theme\", \"force-light\")\n } else if (event.target.value === \"force-dark\") {\n makeDark()\n hasLocalStorage && localStorage.setItem(\"force-color-theme\", \"force-dark\")\n }\n\n setDarkMode(event.target.value)\n }\n\n\n // Localstorage: undefined, cascadia, cascadia-ligatures, consolas, ...\n // ->\n // CSS Body class: font-cascadia, font-cascadia, font-cascadia-ligatures | font-consolas, ...\n\n const handleFontChange = (event: React.ChangeEvent) => {\n localStorage.setItem(\"force-font\", event.target.value)\n switchFont(event.target.value, customFontOverride)\n setFont(event.target.value)\n\n }\n\n\n return (\n
\n
\n

{i(\"footer_customize\")}

\n \n\n \n
\n
\n )\n}\n","import * as React from \"react\"\nimport { useEffect } from \"react\"\n\nimport \"./SiteFooter.scss\"\nimport { createIntlLink } from \"../IntlLink\"\nimport { whenEscape } from \"../../lib/whenEscape\"\nimport { Customize } from \"./SiteFooter-Customize\"\n\nexport type Props = {\n lang: string\n suppressCustomization?: true\n suppressDocRecommendations?: true\n}\n\nconst useLinguaFrancaLinks = [\n {\n title: \"Get Started\",\n url: \"/docs/handbook/overview\",\n },\n {\n title: \"Download\",\n url: \"/download\",\n },\n {\n title: \"Why Lingua Franca\",\n url: \"/\",\n },\n {\n title: \"Publications\",\n url: \"/publications-and-presentations\",\n },\n]\n\nconst communityLinks = [\n {\n title: \"Get Help\",\n url: \"/community\",\n },\n {\n title: \"GitHub Repo\",\n url: \"https://github.com/lf-lang/lingua-franca\",\n },\n {\n title: \"@thelflang\",\n url: \"https://twitter.com/thelflang\",\n },\n {\n title: \"Web Repo\",\n url: \"https://github.com/lf-lang/website-lingua-franca\",\n },\n {\n title: \"Zulip\",\n url: \"https://zulip.lf-lang.org\",\n },\n]\n\nconst faviconForURL = (url: string) => {\n switch (url) {\n case \"https://github.com/lf-lang/website-lingua-franca\":\n case \"https://github.com/lf-lang/lingua-franca\":\n return (\n \n \n \n )\n case \"https://twitter.com/thelflang\":\n return (\n \n \n \n )\n case \"https://zulip.lf-lang.org\":\n return (\n \n \n \n \n )\n }\n}\n\nexport const SiteFooter = (props: Props) => {\n const normalLinks = useLinguaFrancaLinks.filter(\n l => !l.url.includes(\"#show-examples\")\n )\n\n const Link = createIntlLink(props.lang)\n\n const hideDocs = props.suppressDocRecommendations\n return (\n
\n {props.suppressCustomization ? null : }\n\n
\n
\n\n \n \n \n

Made with ♥ in Berkeley, Dallas, Dresden, Kiel, and Seoul\n

\n

\n © 2019-{new Date().getFullYear()} The Lingua Franca Team\n
\n

\n
\n\n
\n

Using Lingua Franca

\n
    \n {normalLinks.map(page => (\n
  • \n {page.title}\n
  • \n ))}\n
\n
\n\n
\n

Community

\n
    \n {communityLinks.map(page => {\n const favicon = faviconForURL(page.url)\n const favSpan = favicon ? ({favicon}) : null\n return (\n
  • \n \n {favSpan}\n {page.title}\n \n
  • \n )\n })}\n
\n
\n
\n\n
\n )\n}\n","import * as React from \"react\"\nimport { Helmet } from \"react-helmet\";\n\nexport type SeoProps = {\n title: string\n description: string\n ogTags?: { [key: string]: string }\n}\n\nexport const HeadSEO = (props: SeoProps) => {\n\n const ogTags = {\n ...props.ogTags,\n \"og:title\": props.title,\n \"og:description\": props.description,\n \"twitter:site\": \"thelflang\",\n }\n\n // Skip search engine indexing on the staging site, this is changed by running:\n // yarn workspace lingua-franca setup-staging\n const staging = false;\n\n if (staging) {\n ogTags[\"robots\"] = \"noindex\"\n }\n\n // do we want localized pages to be the english version?\n //{seo.url && }\n\n // TODO: a lot of pages should have this\n // \n\n // TODO: Maybe on prod we can generate an image for each file\n // \n\n return (\n <>\n \n \n {\n Object.keys(ogTags).map(k => )\n }\n \n \n )\n}\n","import * as React from \"react\"\nimport { SiteNav, Props } from \"./layout/TopNav\"\nimport { SiteFooter } from \"./layout/SiteFooter\"\nimport { SeoProps, HeadSEO } from \"./HeadSEO\";\nimport \"./layout/main.scss\"\nimport { Helmet } from \"react-helmet\";\nimport { withPrefix } from \"gatsby\";\n\ntype LayoutProps = SeoProps & Props & {\n lang: string,\n children: any\n suppressCustomization?: true\n suppressDocRecommendations?: true\n}\nexport const Layout = (props: LayoutProps) => {\n return (\n <>\n \n {/* Should be a NOOP for anything but edge, and much older browsers */}\n
Skip to main content

Publications

View our publications and presentations.

GitHub

Found a bug, or want to provide feedback? Tell us on GitHub.

Twitter

Stay up to date. Follow us on Twitter @thelflang!

Zulip

Have questions, or want to chat with other users? Join the conversation on Zulip.

Active Contributors

logo of Peter Donovan
Peter Donovan
Student Assistant at UC Berkeley.
🇺🇸
logo of Clément Fournier
Clément Fournier
Master student at TU Dresden.
🇩🇪
logo of Erling Rennemo Jellum
Erling Rennemo Jellum
PhD Candidate at Norwegian University of Science and Technology.
🇳🇴
Website
logo of Byeonggil Jun
Byeonggil Jun
Undergraduate student at Hanyang University.
🇰🇷
logo of Dongha Kim
Dongha Kim
Ph.D. student at Arizona State University.
🇺🇸
logo of Hokeun Kim
Hokeun Kim
Assistant Professor at Arizona State University.
🇺🇸
Website
logo of Edward A. Lee
Edward A. Lee
Professor in the Graduate School at UC Berkeley.
🇺🇸
Website
logo of Shaokai Lin
Shaokai Lin
Graduate Student at UC Berkeley.
🇺🇸
Website
logo of Marten Lohstroh
Marten Lohstroh
Postdoctoral researcher at UC Berkeley.
🇺🇸
Website Twitter
logo of Johannes Hayeß
Johannes Hayeß
Master's Student at TU Dresden.
🇩🇪
logo of Christian Menard
Christian Menard
Graduate Student at TU Dresden.
🇩🇪
Website
logo of Alexander Schulz-Rosengarten
Alexander Schulz-Rosengarten
Graduate student at Kiel University.
🇩🇪

Past Contributors

logo of Soroush Bateni
Soroush Bateni
Software Engineer at Apple.
🇺🇸
Website
logo of Matt Chorlian
Matt Chorlian
Applied Math and CS student at UC Berkeley.
🇺🇸
logo of Anirudh Rengarajan
Anirudh Rengarajan
Software Engineer at Bloomberg.
🇺🇸
logo of Martin Schoeberl
Martin Schoeberl
Professor at TU Denmark.
🇩🇰
logo of Matt Weber
Matt Weber
Software Engineer at Anyscale.
🇺🇸
logo of Hou Seng (Steven) Wong
Hou Seng (Steven) Wong
Software Development Engineer at Amazon AWS.
🇺🇸
\ No newline at end of file +
Skip to main content

Publications

View our publications and presentations.

GitHub

Found a bug, or want to provide feedback? Tell us on GitHub.

Twitter

Stay up to date. Follow us on Twitter @thelflang!

Zulip

Have questions, or want to chat with other users? Join the conversation on Zulip.

Active Contributors

logo of Peter Donovan
Peter Donovan
Student Assistant at UC Berkeley.
🇺🇸
logo of Clément Fournier
Clément Fournier
Master student at TU Dresden.
🇩🇪
logo of Erling Rennemo Jellum
Erling Rennemo Jellum
PhD Candidate at Norwegian University of Science and Technology.
🇳🇴
Website
logo of Byeonggil Jun
Byeonggil Jun
Undergraduate student at Hanyang University.
🇰🇷
logo of Dongha Kim
Dongha Kim
Ph.D. student at Arizona State University.
🇺🇸
logo of Hokeun Kim
Hokeun Kim
Assistant Professor at Arizona State University.
🇺🇸
Website
logo of Edward A. Lee
Edward A. Lee
Professor in the Graduate School at UC Berkeley.
🇺🇸
Website
logo of Shaokai Lin
Shaokai Lin
Graduate Student at UC Berkeley.
🇺🇸
Website
logo of Marten Lohstroh
Marten Lohstroh
Postdoctoral researcher at UC Berkeley.
🇺🇸
Website Twitter
logo of Johannes Hayeß
Johannes Hayeß
Master's Student at TU Dresden.
🇩🇪
logo of Christian Menard
Christian Menard
Graduate Student at TU Dresden.
🇩🇪
Website
logo of Alexander Schulz-Rosengarten
Alexander Schulz-Rosengarten
Graduate student at Kiel University.
🇩🇪

Past Contributors

logo of Soroush Bateni
Soroush Bateni
Software Engineer at Apple.
🇺🇸
Website
logo of Matt Chorlian
Matt Chorlian
Applied Math and CS student at UC Berkeley.
🇺🇸
logo of Anirudh Rengarajan
Anirudh Rengarajan
Software Engineer at Bloomberg.
🇺🇸
logo of Martin Schoeberl
Martin Schoeberl
Professor at TU Denmark.
🇩🇰
logo of Matt Weber
Matt Weber
Software Engineer at Anyscale.
🇺🇸
logo of Hou Seng (Steven) Wong
Hou Seng (Steven) Wong
Software Development Engineer at Amazon AWS.
🇺🇸
\ No newline at end of file diff --git a/component---src-templates-documentation-tsx-04d26b5abab5434898b4.js b/component---src-templates-documentation-tsx-04d26b5abab5434898b4.js deleted file mode 100644 index 3fe224596..000000000 --- a/component---src-templates-documentation-tsx-04d26b5abab5434898b4.js +++ /dev/null @@ -1,2 +0,0 @@ -(self.webpackChunklingua_franca=self.webpackChunklingua_franca||[]).push([[517],{8652:function(e,a,t){"use strict";function n(e){var a={en:[{title:"Resources",oneline:"Overview of the project.",id:"resources",chronological:!0,items:[{title:"Overview",id:"0-overview",permalink:"/docs/handbook/overview",oneline:"Overview of Lingua Franca."},{title:"Tutorial Video",id:"0-tutorial-video",permalink:"/docs/handbook/tutorial-video",oneline:"Tutorial video presented by the Lingua Franca team."}]},{title:"Writing Reactors",oneline:"Introduction to writing reactors:",id:"writing-reactors",chronological:!0,items:[{title:"A First Reactor",id:"1-a-first-reactor",permalink:"/docs/handbook/a-first-reactor",oneline:"Writing your first Lingua Franca reactor."},{title:"Inputs and Outputs",id:"1-inputs-and-outputs",permalink:"/docs/handbook/inputs-and-outputs",oneline:"Inputs, outputs, and reactions in Lingua Franca."},{title:"Parameters and State Variables",id:"1-parameters-and-state-variables",permalink:"/docs/handbook/parameters-and-state-variables",oneline:"Parameters and state variables in Lingua Franca."},{title:"Time and Timers",id:"1-time-and-timers",permalink:"/docs/handbook/time-and-timers",oneline:"Time and timers in Lingua Franca."},{title:"Composing Reactors",id:"1-composing-reactors",permalink:"/docs/handbook/composing-reactors",oneline:"Composing reactors in Lingua Franca."},{title:"Reactions",id:"1-reactions",permalink:"/docs/handbook/reactions",oneline:"Reactions in Lingua Franca."},{title:"Methods",id:"1-methods",permalink:"/docs/handbook/methods",oneline:"Methods in Lingua Franca."},{title:"Causality Loops",id:"1-causality-loops",permalink:"/docs/handbook/causality-loops",oneline:"Causality loops in Lingua Franca."},{title:"Extending Reactors",id:"1-extending-reactors",permalink:"/docs/handbook/extending-reactors",oneline:"Extending reactors in Lingua Franca."},{title:"Actions",id:"1-actions",permalink:"/docs/handbook/actions",oneline:"Actions in Lingua Franca."},{title:"Superdense Time",id:"1-superdense-time",permalink:"/docs/handbook/superdense-time",oneline:"Superdense time in Lingua Franca."},{title:"Modal Reactors",id:"1-modal-reactors",permalink:"/docs/handbook/modal-models",oneline:"Modal Reactors"},{title:"Deadlines",id:"1-deadlines",permalink:"/docs/handbook/deadlines",oneline:"Deadlines in Lingua Franca."},{title:"Multiports and Banks",id:"1-multiports-and-banks",permalink:"/docs/handbook/multiports-and-banks",oneline:"Multiports and Banks of Reactors."},{title:"Generic Reactors",id:"1-generic-reactors",permalink:"/docs/handbook/generics",oneline:"Defining generic reactors in Lingua Franca."},{title:"Preambles",id:"1-preambles",permalink:"/docs/handbook/preambles",oneline:"Defining preambles in Lingua Franca."},{title:"Distributed Execution",id:"1-distributed-execution",permalink:"/docs/handbook/distributed-execution",oneline:"Distributed Execution (preliminary)"},{title:"Termination",id:"1-termination",permalink:"/docs/handbook/termination",oneline:"Terminating a Lingua Franca execution."}]},{title:"Tools",oneline:"Tools for developing Lingua Franca programs.",id:"tools",chronological:!0,items:[{title:"Code Extension",id:"2-code-extension",permalink:"/docs/handbook/code-extension",oneline:"Visual Studio Code Extension for Lingua Franca."},{title:"Epoch IDE",id:"2-epoch-ide",permalink:"/docs/handbook/epoch-ide",oneline:"Epoch IDE for Lingua Franca."},{title:"Command Line Tools",id:"2-command-line-tools",permalink:"/docs/handbook/command-line-tools",oneline:"Command-line tools for Lingua Franca."},{title:"Troubleshooting",id:"2-troubleshooting",permalink:"/docs/handbook/troubleshooting",oneline:"Troubleshooting page for Lingua Franca tools."}]},{title:"Reference",oneline:"Reference documentation.",id:"reference",chronological:!0,items:[{title:"Expressions",id:"3-expressions",permalink:"/docs/handbook/expressions",oneline:"Expressions in Lingua Franca."},{title:"Target Language Details",id:"3-target-language-details",permalink:"/docs/handbook/target-language-details",oneline:"Detailed reference for each target langauge."},{title:"Target Declaration",id:"3-target-declaration",permalink:"/docs/handbook/target-declaration",oneline:"The target declaration and its parameters in Lingua Franca."},{title:"Tracing",id:"3-tracing",permalink:"/docs/handbook/tracing",oneline:"Tracing (preliminary)"},{title:"Containerized Execution",id:"3-containerized-execution",permalink:"/docs/handbook/containerized-execution",oneline:"Containerized Execution using Docker"},{title:"Security",id:"3-security",permalink:"/docs/handbook/security",oneline:"Secure Federated Execution"}]},{title:"Embedded Platforms",oneline:"Documentation for developing Lingua Franca on Embedded Platforms.",id:"embedded-platforms",chronological:!0,items:[{title:"Arduino",id:"4-arduino",permalink:"/docs/handbook/arduino",oneline:"Developing LF Programs on Arduino."},{title:"Zephyr",id:"4-zephyr",permalink:"/docs/handbook/zephyr",oneline:"Developing LF Programs for Zephyr RTOS."}]},{title:"Developer",oneline:"Information for developers of the Lingua Franca language and tools.",id:"developer",chronological:!0,items:[{title:"Contributing",id:"5-contributing",permalink:"/docs/handbook/contributing",oneline:"Contribute to Lingua Franca."},{title:"Developer Setup",id:"5-developer-setup",permalink:"/docs/handbook/developer-setup",oneline:"Setting up Lingua Franca for developers."},{title:"Developer IntelliJ Setup",id:"5-developer-intellij-setup",permalink:"/docs/handbook/intellij",oneline:"Developer IntelliJ Setup."},{title:"Regression Tests",id:"5-regression-tests",permalink:"/docs/handbook/regression-tests",oneline:"Regression Tests for Lingua Franca."},{title:"Running Benchmarks",id:"5-running-benchmarks",permalink:"/docs/handbook/running-benchmarks",oneline:"Running Benchmarks."},{title:"Website Development",id:"5-website-development",permalink:"/docs/handbook/website-development",oneline:"Development of the Lingua Franca website."}]}]};return a[["en"].includes(e)?e:"en"]}t.d(a,{m8:function(){return n}})},9438:function(e,a,t){"use strict";t.r(a),t.d(a,{default:function(){return V}});var n=t(2784),u=t(3314),r=t(1952);function o(e,a){var t="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(t)return(t=t.call(e)).next.bind(t);if(Array.isArray(e)||(t=function(e,a){if(!e)return;if("string"==typeof e)return c(e,a);var t=Object.prototype.toString.call(e).slice(8,-1);"Object"===t&&e.constructor&&(t=e.constructor.name);if("Map"===t||"Set"===t)return Array.from(e);if("Arguments"===t||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t))return c(e,a)}(e))||a&&e&&"number"==typeof e.length){t&&(e=t);var n=0;return function(){return n>=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function c(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);t=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function D(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);t")}}}),[]),n.createElement("div",{className:"whitespace-tight raised",style:{padding:0}},n.createElement(R,{className:"justify-between small-columns"},n.createElement(I,{sKey:"pr"},n.createElement("p",null,"Lingua Franca is an open source project. Help us improve these pages ",n.createElement("a",{href:u},"by sending a Pull Request")," ❤")),n.createElement("div",{key:"line1",className:"hide-small vertical-line",style:{marginTop:"1.5rem"}}),n.createElement(I,{sKey:"contribs"},"Contributors to this page:",n.createElement("br",null),n.createElement(j,{data:t})),n.createElement("div",{key:"line2",className:"hide-small vertical-line",style:{marginTop:"1.5rem"}}),n.createElement(I,{sKey:"updated"},n.createElement("p",null,"Last updated: "+o,n.createElement("br",null),n.createElement("br",null),n.createElement("span",{id:"page-loaded-time"}," ")))))},j=function(e){var a=e.data&&e.data.total>e.data.top.length;return n.createElement("div",null,e.data&&e.data.top.map((function(e){var a=e.gravatar.startsWith("http")?e.gravatar:"https://gravatar.com/avatar/"+e.gravatar+"?s=32&&d=blank",t=e.name+" ("+e.count+")",u=e.name.split(" ").map((function(e){return e.substr(0,1)})).join("").toUpperCase();return n.createElement("div",{key:e.gravatar,className:"circle-bg"},u,n.createElement("img",{id:e.gravatar,src:a,alt:t}))})),a&&n.createElement("div",{className:"circle-bg"},e.data.total-e.data.top.length,"+"))},N=function(){var e,a=document.querySelectorAll("#handbook-content nav ul li a"),t=window.scrollY;a.forEach((function(a){try{var n=document.querySelector(decodeURIComponent(a.hash));if(!n)return;n.offsetTop-100<=t&&(e=a)}catch(u){return}})),a.forEach((function(a){a===e?a.classList.add("current"):a.classList.remove("current")}))},_=function(){return n.createElement("svg",{width:"20",height:"20",viewBox:"0 0 20 20",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("path",{d:"M10.052 2.29429C10.3913 1.31699 11.6841 0.866721 12.4829 1.70385C12.6455 1.87427 12.8081 2.05843 12.9176 2.22265C13.2379 2.70316 13.3725 3.33595 13.4218 3.95232C13.4721 4.58045 13.438 5.25457 13.3738 5.86484C13.3093 6.47746 13.2129 7.03959 13.1328 7.44777C13.1294 7.46547 13.1259 7.48288 13.1225 7.5H14.006C15.8777 7.5 17.2924 9.19514 16.9576 11.0367L16.2737 14.7984C15.8017 17.3943 13.2078 19.0291 10.6622 18.3348L5.06251 16.8076C4.14894 16.5585 3.45455 15.8145 3.26885 14.886L2.91581 13.1208C2.63809 11.7322 3.69991 10.5624 4.82905 10.1161C5.15163 9.98861 5.44337 9.82679 5.66974 9.62597C7.37583 8.11245 7.99442 6.90287 9.05406 4.77695C9.4084 4.06605 9.77205 3.10054 10.052 2.29429ZM12.0165 7.87862L12.0169 7.87707L12.0187 7.86973L12.0262 7.83863C12.0328 7.81079 12.0426 7.76903 12.0549 7.71494C12.0793 7.60669 12.1135 7.4493 12.1515 7.25536C12.2277 6.86666 12.3188 6.33504 12.3793 5.76016C12.4401 5.18293 12.4685 4.5758 12.425 4.03206C12.3806 3.47655 12.2652 3.04684 12.0855 2.77735C12.0264 2.6887 11.9138 2.55604 11.7594 2.39421C11.5605 2.18576 11.1314 2.23428 10.9967 2.62228C10.7141 3.43609 10.3334 4.45194 9.94904 5.22305C8.88216 7.36349 8.19326 8.72408 6.33336 10.374C5.99304 10.6759 5.58878 10.8911 5.19665 11.0461C4.31631 11.3941 3.75035 12.1945 3.89639 12.9247L4.24943 14.6899C4.36085 15.247 4.77748 15.6934 5.32562 15.8428L10.9254 17.3701C12.9052 17.91 14.9227 16.6385 15.2898 14.6195L15.9738 10.8578C16.197 9.63009 15.2538 8.5 14.006 8.5H12.5015C12.3476 8.5 12.2022 8.42906 12.1074 8.30771C12.0127 8.18638 11.9792 8.02796 12.0165 7.87862C12.0165 7.87858 12.0165 7.87866 12.0165 7.87862Z"}))},O=function(){return n.createElement("svg",{width:"20",height:"20",viewBox:"0 0 20 20",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("path",{d:"M10.052 17.7057C10.3913 18.683 11.6841 19.1333 12.4829 18.2962C12.6455 18.1257 12.8081 17.9416 12.9176 17.7773C13.2379 17.2968 13.3725 16.664 13.4218 16.0477C13.4721 15.4195 13.438 14.7454 13.3738 14.1352C13.3093 13.5225 13.2129 12.9604 13.1328 12.5522C13.1294 12.5345 13.1259 12.5171 13.1225 12.5H14.006C15.8777 12.5 17.2924 10.8049 16.9576 8.96334L16.2737 5.20164C15.8017 2.60569 13.2078 0.970948 10.6622 1.66518L5.06251 3.19238C4.14894 3.44154 3.45455 4.18546 3.26885 5.11401L2.91581 6.87918C2.63809 8.26779 3.69991 9.43756 4.82905 9.88388C5.15163 10.0114 5.44337 10.1732 5.66974 10.374C7.37583 11.8875 7.99442 13.0971 9.05406 15.223C9.4084 15.9339 9.77205 16.8995 10.052 17.7057ZM12.0165 12.1214L12.0169 12.1229L12.0187 12.1303L12.0262 12.1614C12.0328 12.1892 12.0426 12.231 12.0549 12.2851C12.0793 12.3933 12.1135 12.5507 12.1515 12.7446C12.2277 13.1333 12.3188 13.665 12.3793 14.2398C12.4401 14.8171 12.4685 15.4242 12.425 15.9679C12.3806 16.5234 12.2652 16.9532 12.0855 17.2226C12.0264 17.3113 11.9138 17.444 11.7594 17.6058C11.5605 17.8142 11.1314 17.7657 10.9967 17.3777C10.7141 16.5639 10.3334 15.5481 9.94904 14.777C8.88216 12.6365 8.19326 11.2759 6.33336 9.62597C5.99304 9.32406 5.58878 9.1089 5.19665 8.9539C4.31631 8.60592 3.75035 7.80549 3.89639 7.0753L4.24943 5.31012C4.36085 4.753 4.77748 4.30664 5.32562 4.15715L10.9254 2.62995C12.9052 2.08999 14.9227 3.36145 15.2898 5.38052L15.9738 9.14222C16.197 10.3699 15.2538 11.5 14.006 11.5H12.5015C12.3476 11.5 12.2022 11.5709 12.1074 11.6923C12.0127 11.8136 11.9792 11.972 12.0165 12.1214C12.0165 12.1214 12.0165 12.1213 12.0165 12.1214Z"}))},H=t(4795),J=t(7162),K=t.n(J);function W(e,a){var t="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(t)return(t=t.call(e)).next.bind(t);if(Array.isArray(e)||(t=function(e,a){if(!e)return;if("string"==typeof e)return z(e,a);var t=Object.prototype.toString.call(e).slice(8,-1);"Object"===t&&e.constructor&&(t=e.constructor.name);if("Map"===t||"Set"===t)return Array.from(e);if("Arguments"===t||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t))return z(e,a)}(e))||a&&e&&"number"==typeof e.length){t&&(e=t);var n=0;return function(){return n>=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function z(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);tTSConfig Reference: "+c+""+n[c],o((function(e){return Object.assign({},e,{html:i})}));case 12:case"end":return a.stop()}}),a)})))).apply(this,arguments)}r.show&&function(){a.apply(this,arguments)}()}),[r.show,r.url,r.html]),r}(e.pageContext.lang),o=(0,n.useState)(t.frontmatter.deprecated_by),c=o[0],i=o[1],d=(0,x.D)((0,S.Z)()),l=(0,M.i)(e.pageContext.lang);if((0,n.useEffect)((function(){if(document.location.hash){var a,n=(null===(a=t.frontmatter)||void 0===a?void 0:a.deprecation_redirects)||[],u=n.indexOf(document.location.hash.slice(1));-1!==u&&i(n[u+1])}return document.querySelectorAll("#handbook-content nav ul li a").forEach((function(e){e.addEventListener("click",(function(e){e.preventDefault(),document.querySelector(decodeURIComponent(e.target.hash)).scrollIntoView({behavior:"smooth",block:"start"}),document.location.hash=e.target.hash}))})),window.addEventListener("scroll",N,{passive:!0,capture:!0}),N(),function(e,a){var t=document.getElementById("like-button"),n=document.getElementById("dislike-button");if(t&&n){var u=function(t){return function(){window.appInsights&&window.appInsights.trackEvent({name:t,properties:{slug:e,ab:"b"}});var n=a("handb_thanks"),u=document.getElementById("like-dislike-subnav"),r=document.getElementById("page-helpful-popup");u.innerHTML="
"+n+"
",r.innerHTML="

"+n+"

"}};t.onclick=u("Liked Page"),n.onclick=u("Disliked Page");var r=document.getElementById("like-button-popup"),o=document.getElementById("dislike-button-popup");r.onclick=u("Liked Page"),o.onclick=u("Disliked Page"),window.addEventListener("scroll",(function(){var e=document.body,a=document.documentElement,t=Math.max(e.scrollHeight,e.offsetHeight,a.clientHeight,a.scrollHeight,a.offsetHeight),n=Math.max(window.pageYOffset)+window.innerHeight>t-document.getElementById("site-footer").clientHeight+150,u=document.getElementById("page-helpful-popup"),r=document.getElementById("like-dislike-subnav");if(u&&r){var o=n?"1":"0";u.style.opacity!=o&&(u.style.opacity=o);var c=n?"0":"1";r.style.opacity!=c&&(r.style.opacity=c)}}),{passive:!0,capture:!0})}}(e.pageContext.slug,d),function(){window.removeEventListener("scroll",N)}}),[]),!t.frontmatter)throw new Error("No front-matter found for the file with props: "+e);if(!t.html)throw new Error("No html found for the file with props: "+e);var f=e.pageContext.id||"NO-ID",s=(null===(a=t.headings)||void 0===a?void 0:a.filter((function(e){return((null==e?void 0:e.depth)||0)<=2})))||[],D=!t.frontmatter.disable_toc,m=t.frontmatter.experimental,b=t.headings&&s.length<=30&&s.length>0,F=(0,v.m8)(e.pageContext.lang),p=t.frontmatter.handbook?"Handbook":"Documentation",h=B()();return n.createElement(u.A,{title:p+" - "+t.frontmatter.title,description:t.frontmatter.oneline||"",lang:e.pageContext.lang},n.createElement("section",{id:"doc-layout"},n.createElement(g,null),n.createElement("div",{className:"page-popup",id:"page-helpful-popup",style:{opacity:0}},n.createElement("p",null,"Was this page helpful?"),n.createElement("div",null,n.createElement("button",{className:"first",id:"like-button-popup",title:"Like this page"},n.createElement(_,null)),n.createElement("button",{id:"dislike-button-popup",title:"Dislike this page"},n.createElement(O,null)))),n.createElement("noscript",null,n.createElement("style",{dangerouslySetInnerHTML:{__html:"\n nav#sidebar > ul > li.closed ul {\n display: block !important;\n }\n "}})),n.createElement(E,{navItems:F,selectedID:f}),n.createElement("div",{id:"handbook-content",role:"article"},c&&n.createElement(n.Fragment,null,n.createElement($.Z,null,n.createElement("link",{rel:"canonical",href:"https://www.lf-lang.org"+t.frontmatter.deprecated_by})),n.createElement("div",{id:"deprecated-header"},n.createElement("div",{id:"deprecated-content"},n.createElement("div",{id:"deprecated-icon"},n.createElement("svg",{width:"16",height:"16",viewBox:"0 0 16 16",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("circle",{cx:"8",cy:"8",r:"7.5",stroke:"black"}),n.createElement("path",{d:"M8 3V9",stroke:"black"}),n.createElement("path",{d:"M8 11L8 13",stroke:"black"}))),n.createElement("div",null,n.createElement("h3",null,d("handb_deprecated_title")),n.createElement("p",null,d("handb_deprecated_subtitle"),n.createElement(l,{className:"deprecation-redirect-link",to:c},d("handb_deprecated_subtitle_link"))))),n.createElement("div",{id:"deprecated-action"},n.createElement(l,{className:"deprecation-redirect-link",to:c},d("handb_deprecated_subtitle_action"))))),m&&n.createElement("div",{id:"deprecated-header"},n.createElement("div",{id:"deprecated-content"},n.createElement("div",{id:"deprecated-icon"},n.createElement("svg",{width:"16",height:"16",viewBox:"0 0 16 16",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("circle",{cx:"8",cy:"8",r:"7.5",stroke:"black"}),n.createElement("path",{d:"M8 3V9",stroke:"black"}),n.createElement("path",{d:"M8 11L8 13",stroke:"black"}))),n.createElement("div",null,n.createElement("h3",null,d("handb_experimental_title")),n.createElement("p",null,d("handb_experimental_subtitle"))))),n.createElement("h2",null,t.frontmatter.title),t.frontmatter.preamble&&n.createElement("div",{className:"preamble",dangerouslySetInnerHTML:{__html:t.frontmatter.preamble}}),n.createElement("article",null,n.createElement("div",{className:"whitespace raised"},n.createElement("div",{className:"markdown",dangerouslySetInnerHTML:{__html:C.postProcessHTML(t.html)}})),D&&n.createElement("aside",{className:"handbook-toc"},n.createElement("nav",{className:c?"deprecated":""},b&&n.createElement(n.Fragment,null,n.createElement("h5",null,d("handb_on_this_page")),n.createElement("ul",null,s.map((function(e){var a=h.slug(e.value,!1);return n.createElement("li",{key:a},n.createElement("a",{href:"#"+a},e.value))})))),n.createElement("div",{id:"like-dislike-subnav"},n.createElement("h5",null,d("handb_like_dislike_title")),n.createElement("div",null,n.createElement("button",{title:"Like this page",id:"like-button"},n.createElement(_,null)," ",d("handb_like_desc")),n.createElement("button",{title:"Dislike this page",id:"dislike-button"},n.createElement(O,null)," ",d("handb_dislike_desc"))))))),n.createElement(k,{next:e.data.next,prev:e.data.prev,i:d,IntlLink:l}),n.createElement(P,{lang:e.pageContext.lang,i:d,path:e.pageContext.repoPath,lastEdited:e.pageContext.modifiedTime}))),n.createElement(Z,r))},V=function(e){return n.createElement(h.R,{locale:e.pageContext.lang},n.createElement(U,e))}},8711:function(e){function a(e,a){var n="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(n)return(n=n.call(e)).next.bind(n);if(Array.isArray(e)||(n=function(e,a){if(!e)return;if("string"==typeof e)return t(e,a);var n=Object.prototype.toString.call(e).slice(8,-1);"Object"===n&&e.constructor&&(n=e.constructor.name);if("Map"===n||"Set"===n)return Array.from(e);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return t(e,a)}(e))||a&&e&&"number"==typeof e.length){n&&(e=n);var u=0;return function(){return u>=e.length?{done:!0}:{done:!1,value:e[u++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function t(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);tThis page is showing examples in the target language [C]{lf-c}[C++]{lf-cpp}[Python]{lf-py}[TypeScript]{lf-ts}[Rust]{lf-rs}.\nYou can change the target language in the left sidebar.'},o=new RegExp("\\b(?:"+n.join("|")+")\\b"),i=new RegExp("\\$("+n.join("|")+")\\$","gm"),d=new RegExp("\\$("+["start","end"].join("|")+")\\((.*)\\)\\$","gm"),l=new RegExp("\\$("+Object.keys(r).join("|")+")\\$","gm"),f=/\[([^\]]*)\]\{([^\}]*)\}/gm;function s(e,t,n){for(var r,o=[],i=a(n.split(" "));!(r=i()).done;)c=r.value,u[c]?Array.prototype.push.apply(o,u[c]):o.push(c);return''+t+""}function D(e,a){return''+a+""}function m(e,a){return r[a]?r[a]:'ERROR: textSubstitutions key not found.'}e.exports={postProcessHTML:function(e){var a=e.replace(l,m);return a=(a=(a=a.replace(f,s)).replace(i,D)).replace(d,"")},keywordMatcher:o}},6209:function(e,a,t){var n=t(3965);e.exports=r;var u=Object.hasOwnProperty;function r(){if(!(this instanceof r))return new r;this.reset()}function o(e,a){return"string"!=typeof e?"":(a||(e=e.toLowerCase()),e.replace(n,"").replace(/ /g,"-"))}r.prototype.slug=function(e,a){for(var t=this,n=o(e,!0===a),r=n;u.call(t.occurrences,n);)t.occurrences[r]++,n=r+"-"+t.occurrences[r];return t.occurrences[n]=0,n},r.prototype.reset=function(){this.occurrences=Object.create(null)},r.slug=o},3965:function(e){e.exports=/[\0-\x1F!-,\.\/:-@\[-\^`\{-\xA9\xAB-\xB4\xB6-\xB9\xBB-\xBF\xD7\xF7\u02C2-\u02C5\u02D2-\u02DF\u02E5-\u02EB\u02ED\u02EF-\u02FF\u0375\u0378\u0379\u037E\u0380-\u0385\u0387\u038B\u038D\u03A2\u03F6\u0482\u0530\u0557\u0558\u055A-\u055F\u0589-\u0590\u05BE\u05C0\u05C3\u05C6\u05C8-\u05CF\u05EB-\u05EE\u05F3-\u060F\u061B-\u061F\u066A-\u066D\u06D4\u06DD\u06DE\u06E9\u06FD\u06FE\u0700-\u070F\u074B\u074C\u07B2-\u07BF\u07F6-\u07F9\u07FB\u07FC\u07FE\u07FF\u082E-\u083F\u085C-\u085F\u086B-\u089F\u08B5\u08BE-\u08D2\u08E2\u0964\u0965\u0970\u0984\u098D\u098E\u0991\u0992\u09A9\u09B1\u09B3-\u09B5\u09BA\u09BB\u09C5\u09C6\u09C9\u09CA\u09CF-\u09D6\u09D8-\u09DB\u09DE\u09E4\u09E5\u09F2-\u09FB\u09FD\u09FF\u0A00\u0A04\u0A0B-\u0A0E\u0A11\u0A12\u0A29\u0A31\u0A34\u0A37\u0A3A\u0A3B\u0A3D\u0A43-\u0A46\u0A49\u0A4A\u0A4E-\u0A50\u0A52-\u0A58\u0A5D\u0A5F-\u0A65\u0A76-\u0A80\u0A84\u0A8E\u0A92\u0AA9\u0AB1\u0AB4\u0ABA\u0ABB\u0AC6\u0ACA\u0ACE\u0ACF\u0AD1-\u0ADF\u0AE4\u0AE5\u0AF0-\u0AF8\u0B00\u0B04\u0B0D\u0B0E\u0B11\u0B12\u0B29\u0B31\u0B34\u0B3A\u0B3B\u0B45\u0B46\u0B49\u0B4A\u0B4E-\u0B55\u0B58-\u0B5B\u0B5E\u0B64\u0B65\u0B70\u0B72-\u0B81\u0B84\u0B8B-\u0B8D\u0B91\u0B96-\u0B98\u0B9B\u0B9D\u0BA0-\u0BA2\u0BA5-\u0BA7\u0BAB-\u0BAD\u0BBA-\u0BBD\u0BC3-\u0BC5\u0BC9\u0BCE\u0BCF\u0BD1-\u0BD6\u0BD8-\u0BE5\u0BF0-\u0BFF\u0C0D\u0C11\u0C29\u0C3A-\u0C3C\u0C45\u0C49\u0C4E-\u0C54\u0C57\u0C5B-\u0C5F\u0C64\u0C65\u0C70-\u0C7F\u0C84\u0C8D\u0C91\u0CA9\u0CB4\u0CBA\u0CBB\u0CC5\u0CC9\u0CCE-\u0CD4\u0CD7-\u0CDD\u0CDF\u0CE4\u0CE5\u0CF0\u0CF3-\u0CFF\u0D04\u0D0D\u0D11\u0D45\u0D49\u0D4F-\u0D53\u0D58-\u0D5E\u0D64\u0D65\u0D70-\u0D79\u0D80\u0D81\u0D84\u0D97-\u0D99\u0DB2\u0DBC\u0DBE\u0DBF\u0DC7-\u0DC9\u0DCB-\u0DCE\u0DD5\u0DD7\u0DE0-\u0DE5\u0DF0\u0DF1\u0DF4-\u0E00\u0E3B-\u0E3F\u0E4F\u0E5A-\u0E80\u0E83\u0E85\u0E8B\u0EA4\u0EA6\u0EBE\u0EBF\u0EC5\u0EC7\u0ECE\u0ECF\u0EDA\u0EDB\u0EE0-\u0EFF\u0F01-\u0F17\u0F1A-\u0F1F\u0F2A-\u0F34\u0F36\u0F38\u0F3A-\u0F3D\u0F48\u0F6D-\u0F70\u0F85\u0F98\u0FBD-\u0FC5\u0FC7-\u0FFF\u104A-\u104F\u109E\u109F\u10C6\u10C8-\u10CC\u10CE\u10CF\u10FB\u1249\u124E\u124F\u1257\u1259\u125E\u125F\u1289\u128E\u128F\u12B1\u12B6\u12B7\u12BF\u12C1\u12C6\u12C7\u12D7\u1311\u1316\u1317\u135B\u135C\u1360-\u137F\u1390-\u139F\u13F6\u13F7\u13FE-\u1400\u166D\u166E\u1680\u169B-\u169F\u16EB-\u16ED\u16F9-\u16FF\u170D\u1715-\u171F\u1735-\u173F\u1754-\u175F\u176D\u1771\u1774-\u177F\u17D4-\u17D6\u17D8-\u17DB\u17DE\u17DF\u17EA-\u180A\u180E\u180F\u181A-\u181F\u1879-\u187F\u18AB-\u18AF\u18F6-\u18FF\u191F\u192C-\u192F\u193C-\u1945\u196E\u196F\u1975-\u197F\u19AC-\u19AF\u19CA-\u19CF\u19DA-\u19FF\u1A1C-\u1A1F\u1A5F\u1A7D\u1A7E\u1A8A-\u1A8F\u1A9A-\u1AA6\u1AA8-\u1AAF\u1ABF-\u1AFF\u1B4C-\u1B4F\u1B5A-\u1B6A\u1B74-\u1B7F\u1BF4-\u1BFF\u1C38-\u1C3F\u1C4A-\u1C4C\u1C7E\u1C7F\u1C89-\u1C8F\u1CBB\u1CBC\u1CC0-\u1CCF\u1CD3\u1CFB-\u1CFF\u1DFA\u1F16\u1F17\u1F1E\u1F1F\u1F46\u1F47\u1F4E\u1F4F\u1F58\u1F5A\u1F5C\u1F5E\u1F7E\u1F7F\u1FB5\u1FBD\u1FBF-\u1FC1\u1FC5\u1FCD-\u1FCF\u1FD4\u1FD5\u1FDC-\u1FDF\u1FED-\u1FF1\u1FF5\u1FFD-\u203E\u2041-\u2053\u2055-\u2070\u2072-\u207E\u2080-\u208F\u209D-\u20CF\u20F1-\u2101\u2103-\u2106\u2108\u2109\u2114\u2116-\u2118\u211E-\u2123\u2125\u2127\u2129\u212E\u213A\u213B\u2140-\u2144\u214A-\u214D\u214F-\u215F\u2189-\u24B5\u24EA-\u2BFF\u2C2F\u2C5F\u2CE5-\u2CEA\u2CF4-\u2CFF\u2D26\u2D28-\u2D2C\u2D2E\u2D2F\u2D68-\u2D6E\u2D70-\u2D7E\u2D97-\u2D9F\u2DA7\u2DAF\u2DB7\u2DBF\u2DC7\u2DCF\u2DD7\u2DDF\u2E00-\u2E2E\u2E30-\u3004\u3008-\u3020\u3030\u3036\u3037\u303D-\u3040\u3097\u3098\u309B\u309C\u30A0\u30FB\u3100-\u3104\u3130\u318F-\u319F\u31BB-\u31EF\u3200-\u33FF\u4DB6-\u4DFF\u9FF0-\u9FFF\uA48D-\uA4CF\uA4FE\uA4FF\uA60D-\uA60F\uA62C-\uA63F\uA673\uA67E\uA6F2-\uA716\uA720\uA721\uA789\uA78A\uA7C0\uA7C1\uA7C7-\uA7F6\uA828-\uA83F\uA874-\uA87F\uA8C6-\uA8CF\uA8DA-\uA8DF\uA8F8-\uA8FA\uA8FC\uA92E\uA92F\uA954-\uA95F\uA97D-\uA97F\uA9C1-\uA9CE\uA9DA-\uA9DF\uA9FF\uAA37-\uAA3F\uAA4E\uAA4F\uAA5A-\uAA5F\uAA77-\uAA79\uAAC3-\uAADA\uAADE\uAADF\uAAF0\uAAF1\uAAF7-\uAB00\uAB07\uAB08\uAB0F\uAB10\uAB17-\uAB1F\uAB27\uAB2F\uAB5B\uAB68-\uAB6F\uABEB\uABEE\uABEF\uABFA-\uABFF\uD7A4-\uD7AF\uD7C7-\uD7CA\uD7FC-\uD7FF\uE000-\uF8FF\uFA6E\uFA6F\uFADA-\uFAFF\uFB07-\uFB12\uFB18-\uFB1C\uFB29\uFB37\uFB3D\uFB3F\uFB42\uFB45\uFBB2-\uFBD2\uFD3E-\uFD4F\uFD90\uFD91\uFDC8-\uFDEF\uFDFC-\uFDFF\uFE10-\uFE1F\uFE30-\uFE32\uFE35-\uFE4C\uFE50-\uFE6F\uFE75\uFEFD-\uFF0F\uFF1A-\uFF20\uFF3B-\uFF3E\uFF40\uFF5B-\uFF65\uFFBF-\uFFC1\uFFC8\uFFC9\uFFD0\uFFD1\uFFD8\uFFD9\uFFDD-\uFFFF]|\uD800[\uDC0C\uDC27\uDC3B\uDC3E\uDC4E\uDC4F\uDC5E-\uDC7F\uDCFB-\uDD3F\uDD75-\uDDFC\uDDFE-\uDE7F\uDE9D-\uDE9F\uDED1-\uDEDF\uDEE1-\uDEFF\uDF20-\uDF2C\uDF4B-\uDF4F\uDF7B-\uDF7F\uDF9E\uDF9F\uDFC4-\uDFC7\uDFD0\uDFD6-\uDFFF]|\uD801[\uDC9E\uDC9F\uDCAA-\uDCAF\uDCD4-\uDCD7\uDCFC-\uDCFF\uDD28-\uDD2F\uDD64-\uDDFF\uDF37-\uDF3F\uDF56-\uDF5F\uDF68-\uDFFF]|\uD802[\uDC06\uDC07\uDC09\uDC36\uDC39-\uDC3B\uDC3D\uDC3E\uDC56-\uDC5F\uDC77-\uDC7F\uDC9F-\uDCDF\uDCF3\uDCF6-\uDCFF\uDD16-\uDD1F\uDD3A-\uDD7F\uDDB8-\uDDBD\uDDC0-\uDDFF\uDE04\uDE07-\uDE0B\uDE14\uDE18\uDE36\uDE37\uDE3B-\uDE3E\uDE40-\uDE5F\uDE7D-\uDE7F\uDE9D-\uDEBF\uDEC8\uDEE7-\uDEFF\uDF36-\uDF3F\uDF56-\uDF5F\uDF73-\uDF7F\uDF92-\uDFFF]|\uD803[\uDC49-\uDC7F\uDCB3-\uDCBF\uDCF3-\uDCFF\uDD28-\uDD2F\uDD3A-\uDEFF\uDF1D-\uDF26\uDF28-\uDF2F\uDF51-\uDFDF\uDFF7-\uDFFF]|\uD804[\uDC47-\uDC65\uDC70-\uDC7E\uDCBB-\uDCCF\uDCE9-\uDCEF\uDCFA-\uDCFF\uDD35\uDD40-\uDD43\uDD47-\uDD4F\uDD74\uDD75\uDD77-\uDD7F\uDDC5-\uDDC8\uDDCD-\uDDCF\uDDDB\uDDDD-\uDDFF\uDE12\uDE38-\uDE3D\uDE3F-\uDE7F\uDE87\uDE89\uDE8E\uDE9E\uDEA9-\uDEAF\uDEEB-\uDEEF\uDEFA-\uDEFF\uDF04\uDF0D\uDF0E\uDF11\uDF12\uDF29\uDF31\uDF34\uDF3A\uDF45\uDF46\uDF49\uDF4A\uDF4E\uDF4F\uDF51-\uDF56\uDF58-\uDF5C\uDF64\uDF65\uDF6D-\uDF6F\uDF75-\uDFFF]|\uD805[\uDC4B-\uDC4F\uDC5A-\uDC5D\uDC60-\uDC7F\uDCC6\uDCC8-\uDCCF\uDCDA-\uDD7F\uDDB6\uDDB7\uDDC1-\uDDD7\uDDDE-\uDDFF\uDE41-\uDE43\uDE45-\uDE4F\uDE5A-\uDE7F\uDEB9-\uDEBF\uDECA-\uDEFF\uDF1B\uDF1C\uDF2C-\uDF2F\uDF3A-\uDFFF]|\uD806[\uDC3B-\uDC9F\uDCEA-\uDCFE\uDD00-\uDD9F\uDDA8\uDDA9\uDDD8\uDDD9\uDDE2\uDDE5-\uDDFF\uDE3F-\uDE46\uDE48-\uDE4F\uDE9A-\uDE9C\uDE9E-\uDEBF\uDEF9-\uDFFF]|\uD807[\uDC09\uDC37\uDC41-\uDC4F\uDC5A-\uDC71\uDC90\uDC91\uDCA8\uDCB7-\uDCFF\uDD07\uDD0A\uDD37-\uDD39\uDD3B\uDD3E\uDD48-\uDD4F\uDD5A-\uDD5F\uDD66\uDD69\uDD8F\uDD92\uDD99-\uDD9F\uDDAA-\uDEDF\uDEF7-\uDFFF]|\uD808[\uDF9A-\uDFFF]|\uD809[\uDC6F-\uDC7F\uDD44-\uDFFF]|[\uD80A\uD80B\uD80E-\uD810\uD812-\uD819\uD823-\uD82B\uD82D\uD82E\uD830-\uD833\uD837\uD839\uD83D-\uD83F\uD87B-\uD87D\uD87F-\uDB3F\uDB41-\uDBFF][\uDC00-\uDFFF]|\uD80D[\uDC2F-\uDFFF]|\uD811[\uDE47-\uDFFF]|\uD81A[\uDE39-\uDE3F\uDE5F\uDE6A-\uDECF\uDEEE\uDEEF\uDEF5-\uDEFF\uDF37-\uDF3F\uDF44-\uDF4F\uDF5A-\uDF62\uDF78-\uDF7C\uDF90-\uDFFF]|\uD81B[\uDC00-\uDE3F\uDE80-\uDEFF\uDF4B-\uDF4E\uDF88-\uDF8E\uDFA0-\uDFDF\uDFE2\uDFE4-\uDFFF]|\uD821[\uDFF8-\uDFFF]|\uD822[\uDEF3-\uDFFF]|\uD82C[\uDD1F-\uDD4F\uDD53-\uDD63\uDD68-\uDD6F\uDEFC-\uDFFF]|\uD82F[\uDC6B-\uDC6F\uDC7D-\uDC7F\uDC89-\uDC8F\uDC9A-\uDC9C\uDC9F-\uDFFF]|\uD834[\uDC00-\uDD64\uDD6A-\uDD6C\uDD73-\uDD7A\uDD83\uDD84\uDD8C-\uDDA9\uDDAE-\uDE41\uDE45-\uDFFF]|\uD835[\uDC55\uDC9D\uDCA0\uDCA1\uDCA3\uDCA4\uDCA7\uDCA8\uDCAD\uDCBA\uDCBC\uDCC4\uDD06\uDD0B\uDD0C\uDD15\uDD1D\uDD3A\uDD3F\uDD45\uDD47-\uDD49\uDD51\uDEA6\uDEA7\uDEC1\uDEDB\uDEFB\uDF15\uDF35\uDF4F\uDF6F\uDF89\uDFA9\uDFC3\uDFCC\uDFCD]|\uD836[\uDC00-\uDDFF\uDE37-\uDE3A\uDE6D-\uDE74\uDE76-\uDE83\uDE85-\uDE9A\uDEA0\uDEB0-\uDFFF]|\uD838[\uDC07\uDC19\uDC1A\uDC22\uDC25\uDC2B-\uDCFF\uDD2D-\uDD2F\uDD3E\uDD3F\uDD4A-\uDD4D\uDD4F-\uDEBF\uDEFA-\uDFFF]|\uD83A[\uDCC5-\uDCCF\uDCD7-\uDCFF\uDD4C-\uDD4F\uDD5A-\uDFFF]|\uD83B[\uDC00-\uDDFF\uDE04\uDE20\uDE23\uDE25\uDE26\uDE28\uDE33\uDE38\uDE3A\uDE3C-\uDE41\uDE43-\uDE46\uDE48\uDE4A\uDE4C\uDE50\uDE53\uDE55\uDE56\uDE58\uDE5A\uDE5C\uDE5E\uDE60\uDE63\uDE65\uDE66\uDE6B\uDE73\uDE78\uDE7D\uDE7F\uDE8A\uDE9C-\uDEA0\uDEA4\uDEAA\uDEBC-\uDFFF]|\uD83C[\uDC00-\uDD2F\uDD4A-\uDD4F\uDD6A-\uDD6F\uDD8A-\uDFFF]|\uD869[\uDED7-\uDEFF]|\uD86D[\uDF35-\uDF3F]|\uD86E[\uDC1E\uDC1F]|\uD873[\uDEA2-\uDEAF]|\uD87A[\uDFE1-\uDFFF]|\uD87E[\uDE1E-\uDFFF]|\uDB40[\uDC00-\uDCFF\uDDF0-\uDFFF]/g}}]); -//# sourceMappingURL=component---src-templates-documentation-tsx-04d26b5abab5434898b4.js.map \ No newline at end of file diff --git a/component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js b/component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js new file mode 100644 index 000000000..d9de759ca --- /dev/null +++ b/component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js @@ -0,0 +1,2 @@ +(self.webpackChunklingua_franca=self.webpackChunklingua_franca||[]).push([[517],{8652:function(e,a,t){"use strict";function n(e){var a={en:[{title:"Resources",oneline:"Overview of the project.",id:"resources",chronological:!0,items:[{title:"Overview",id:"0-overview",permalink:"/docs/handbook/overview",oneline:"Overview of Lingua Franca."},{title:"Tutorial Video",id:"0-tutorial-video",permalink:"/docs/handbook/tutorial-video",oneline:"Tutorial video presented by the Lingua Franca team."}]},{title:"Writing Reactors",oneline:"Introduction to writing reactors:",id:"writing-reactors",chronological:!0,items:[{title:"A First Reactor",id:"1-a-first-reactor",permalink:"/docs/handbook/a-first-reactor",oneline:"Writing your first Lingua Franca reactor."},{title:"Inputs and Outputs",id:"1-inputs-and-outputs",permalink:"/docs/handbook/inputs-and-outputs",oneline:"Inputs, outputs, and reactions in Lingua Franca."},{title:"Parameters and State Variables",id:"1-parameters-and-state-variables",permalink:"/docs/handbook/parameters-and-state-variables",oneline:"Parameters and state variables in Lingua Franca."},{title:"Time and Timers",id:"1-time-and-timers",permalink:"/docs/handbook/time-and-timers",oneline:"Time and timers in Lingua Franca."},{title:"Composing Reactors",id:"1-composing-reactors",permalink:"/docs/handbook/composing-reactors",oneline:"Composing reactors in Lingua Franca."},{title:"Reactions",id:"1-reactions",permalink:"/docs/handbook/reactions",oneline:"Reactions in Lingua Franca."},{title:"Methods",id:"1-methods",permalink:"/docs/handbook/methods",oneline:"Methods in Lingua Franca."},{title:"Causality Loops",id:"1-causality-loops",permalink:"/docs/handbook/causality-loops",oneline:"Causality loops in Lingua Franca."},{title:"Extending Reactors",id:"1-extending-reactors",permalink:"/docs/handbook/extending-reactors",oneline:"Extending reactors in Lingua Franca."},{title:"Actions",id:"1-actions",permalink:"/docs/handbook/actions",oneline:"Actions in Lingua Franca."},{title:"Superdense Time",id:"1-superdense-time",permalink:"/docs/handbook/superdense-time",oneline:"Superdense time in Lingua Franca."},{title:"Modal Reactors",id:"1-modal-reactors",permalink:"/docs/handbook/modal-models",oneline:"Modal Reactors"},{title:"Deadlines",id:"1-deadlines",permalink:"/docs/handbook/deadlines",oneline:"Deadlines in Lingua Franca."},{title:"Multiports and Banks",id:"1-multiports-and-banks",permalink:"/docs/handbook/multiports-and-banks",oneline:"Multiports and Banks of Reactors."},{title:"Generic Reactors",id:"1-generic-reactors",permalink:"/docs/handbook/generics",oneline:"Defining generic reactors in Lingua Franca."},{title:"Preambles",id:"1-preambles",permalink:"/docs/handbook/preambles",oneline:"Defining preambles in Lingua Franca."},{title:"Distributed Execution",id:"1-distributed-execution",permalink:"/docs/handbook/distributed-execution",oneline:"Distributed Execution (preliminary)"},{title:"Termination",id:"1-termination",permalink:"/docs/handbook/termination",oneline:"Terminating a Lingua Franca execution."}]},{title:"Tools",oneline:"Tools for developing Lingua Franca programs.",id:"tools",chronological:!0,items:[{title:"Code Extension",id:"2-code-extension",permalink:"/docs/handbook/code-extension",oneline:"Visual Studio Code Extension for Lingua Franca."},{title:"Epoch IDE",id:"2-epoch-ide",permalink:"/docs/handbook/epoch-ide",oneline:"Epoch IDE for Lingua Franca."},{title:"Command Line Tools",id:"2-command-line-tools",permalink:"/docs/handbook/command-line-tools",oneline:"Command-line tools for Lingua Franca."},{title:"Troubleshooting",id:"2-troubleshooting",permalink:"/docs/handbook/troubleshooting",oneline:"Troubleshooting page for Lingua Franca tools."}]},{title:"Reference",oneline:"Reference documentation.",id:"reference",chronological:!0,items:[{title:"Expressions",id:"3-expressions",permalink:"/docs/handbook/expressions",oneline:"Expressions in Lingua Franca."},{title:"Target Language Details",id:"3-target-language-details",permalink:"/docs/handbook/target-language-details",oneline:"Detailed reference for each target langauge."},{title:"Target Declaration",id:"3-target-declaration",permalink:"/docs/handbook/target-declaration",oneline:"The target declaration and its parameters in Lingua Franca."},{title:"Tracing",id:"3-tracing",permalink:"/docs/handbook/tracing",oneline:"Tracing (preliminary)"},{title:"Containerized Execution",id:"3-containerized-execution",permalink:"/docs/handbook/containerized-execution",oneline:"Containerized Execution using Docker"},{title:"Security",id:"3-security",permalink:"/docs/handbook/security",oneline:"Secure Federated Execution"}]},{title:"Embedded Platforms",oneline:"Documentation for developing Lingua Franca on Embedded Platforms.",id:"embedded-platforms",chronological:!0,items:[{title:"Arduino",id:"4-arduino",permalink:"/docs/handbook/arduino",oneline:"Developing LF Programs on Arduino."},{title:"Zephyr",id:"4-zephyr",permalink:"/docs/handbook/zephyr",oneline:"Developing LF Programs for Zephyr RTOS."}]},{title:"Developer",oneline:"Information for developers of the Lingua Franca language and tools.",id:"developer",chronological:!0,items:[{title:"Contributing",id:"5-contributing",permalink:"/docs/handbook/contributing",oneline:"Contribute to Lingua Franca."},{title:"Developer Setup",id:"5-developer-setup",permalink:"/docs/handbook/developer-setup",oneline:"Setting up Lingua Franca for developers."},{title:"Developer IntelliJ Setup",id:"5-developer-intellij-setup",permalink:"/docs/handbook/intellij",oneline:"Developer IntelliJ Setup."},{title:"Regression Tests",id:"5-regression-tests",permalink:"/docs/handbook/regression-tests",oneline:"Regression Tests for Lingua Franca."},{title:"Running Benchmarks",id:"5-running-benchmarks",permalink:"/docs/handbook/running-benchmarks",oneline:"Running Benchmarks."},{title:"Website Development",id:"5-website-development",permalink:"/docs/handbook/website-development",oneline:"Development of the Lingua Franca website."}]}]};return a[["en"].includes(e)?e:"en"]}t.d(a,{m8:function(){return n}})},9438:function(e,a,t){"use strict";t.r(a),t.d(a,{default:function(){return V}});var n=t(2784),u=t(3314),r=t(1952);function o(e,a){var t="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(t)return(t=t.call(e)).next.bind(t);if(Array.isArray(e)||(t=function(e,a){if(!e)return;if("string"==typeof e)return c(e,a);var t=Object.prototype.toString.call(e).slice(8,-1);"Object"===t&&e.constructor&&(t=e.constructor.name);if("Map"===t||"Set"===t)return Array.from(e);if("Arguments"===t||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t))return c(e,a)}(e))||a&&e&&"number"==typeof e.length){t&&(e=t);var n=0;return function(){return n>=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function c(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);t=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function D(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);t")}}}),[]),n.createElement("div",{className:"whitespace-tight raised",style:{padding:0}},n.createElement(T,{className:"justify-between small-columns"},n.createElement(I,{sKey:"pr"},n.createElement("p",null,"Lingua Franca is an open source project. Help us improve these pages ",n.createElement("a",{href:u},"by sending a Pull Request")," ❤")),n.createElement("div",{key:"line1",className:"hide-small vertical-line",style:{marginTop:"1.5rem"}}),n.createElement(I,{sKey:"contribs"},"Contributors to this page:",n.createElement("br",null),n.createElement(j,{data:t})),n.createElement("div",{key:"line2",className:"hide-small vertical-line",style:{marginTop:"1.5rem"}}),n.createElement(I,{sKey:"updated"},n.createElement("p",null,"Last updated: "+o,n.createElement("br",null),n.createElement("br",null),n.createElement("span",{id:"page-loaded-time"}," ")))))},j=function(e){var a=e.data&&e.data.total>e.data.top.length;return n.createElement("div",null,e.data&&e.data.top.map((function(e){var a=e.gravatar.startsWith("http")?e.gravatar:"https://gravatar.com/avatar/"+e.gravatar+"?s=32&&d=blank",t=e.name+" ("+e.count+")",u=e.name.split(" ").map((function(e){return e.substr(0,1)})).join("").toUpperCase();return n.createElement("div",{key:e.gravatar,className:"circle-bg"},u,n.createElement("img",{id:e.gravatar,src:a,alt:t}))})),a&&n.createElement("div",{className:"circle-bg"},e.data.total-e.data.top.length,"+"))},N=function(){var e,a=document.querySelectorAll("#handbook-content nav ul li a"),t=window.scrollY;a.forEach((function(a){try{var n=document.querySelector(decodeURIComponent(a.hash));if(!n)return;n.offsetTop-100<=t&&(e=a)}catch(u){return}})),a.forEach((function(a){a===e?a.classList.add("current"):a.classList.remove("current")}))},_=function(){return n.createElement("svg",{width:"20",height:"20",viewBox:"0 0 20 20",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("path",{d:"M10.052 2.29429C10.3913 1.31699 11.6841 0.866721 12.4829 1.70385C12.6455 1.87427 12.8081 2.05843 12.9176 2.22265C13.2379 2.70316 13.3725 3.33595 13.4218 3.95232C13.4721 4.58045 13.438 5.25457 13.3738 5.86484C13.3093 6.47746 13.2129 7.03959 13.1328 7.44777C13.1294 7.46547 13.1259 7.48288 13.1225 7.5H14.006C15.8777 7.5 17.2924 9.19514 16.9576 11.0367L16.2737 14.7984C15.8017 17.3943 13.2078 19.0291 10.6622 18.3348L5.06251 16.8076C4.14894 16.5585 3.45455 15.8145 3.26885 14.886L2.91581 13.1208C2.63809 11.7322 3.69991 10.5624 4.82905 10.1161C5.15163 9.98861 5.44337 9.82679 5.66974 9.62597C7.37583 8.11245 7.99442 6.90287 9.05406 4.77695C9.4084 4.06605 9.77205 3.10054 10.052 2.29429ZM12.0165 7.87862L12.0169 7.87707L12.0187 7.86973L12.0262 7.83863C12.0328 7.81079 12.0426 7.76903 12.0549 7.71494C12.0793 7.60669 12.1135 7.4493 12.1515 7.25536C12.2277 6.86666 12.3188 6.33504 12.3793 5.76016C12.4401 5.18293 12.4685 4.5758 12.425 4.03206C12.3806 3.47655 12.2652 3.04684 12.0855 2.77735C12.0264 2.6887 11.9138 2.55604 11.7594 2.39421C11.5605 2.18576 11.1314 2.23428 10.9967 2.62228C10.7141 3.43609 10.3334 4.45194 9.94904 5.22305C8.88216 7.36349 8.19326 8.72408 6.33336 10.374C5.99304 10.6759 5.58878 10.8911 5.19665 11.0461C4.31631 11.3941 3.75035 12.1945 3.89639 12.9247L4.24943 14.6899C4.36085 15.247 4.77748 15.6934 5.32562 15.8428L10.9254 17.3701C12.9052 17.91 14.9227 16.6385 15.2898 14.6195L15.9738 10.8578C16.197 9.63009 15.2538 8.5 14.006 8.5H12.5015C12.3476 8.5 12.2022 8.42906 12.1074 8.30771C12.0127 8.18638 11.9792 8.02796 12.0165 7.87862C12.0165 7.87858 12.0165 7.87866 12.0165 7.87862Z"}))},O=function(){return n.createElement("svg",{width:"20",height:"20",viewBox:"0 0 20 20",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("path",{d:"M10.052 17.7057C10.3913 18.683 11.6841 19.1333 12.4829 18.2962C12.6455 18.1257 12.8081 17.9416 12.9176 17.7773C13.2379 17.2968 13.3725 16.664 13.4218 16.0477C13.4721 15.4195 13.438 14.7454 13.3738 14.1352C13.3093 13.5225 13.2129 12.9604 13.1328 12.5522C13.1294 12.5345 13.1259 12.5171 13.1225 12.5H14.006C15.8777 12.5 17.2924 10.8049 16.9576 8.96334L16.2737 5.20164C15.8017 2.60569 13.2078 0.970948 10.6622 1.66518L5.06251 3.19238C4.14894 3.44154 3.45455 4.18546 3.26885 5.11401L2.91581 6.87918C2.63809 8.26779 3.69991 9.43756 4.82905 9.88388C5.15163 10.0114 5.44337 10.1732 5.66974 10.374C7.37583 11.8875 7.99442 13.0971 9.05406 15.223C9.4084 15.9339 9.77205 16.8995 10.052 17.7057ZM12.0165 12.1214L12.0169 12.1229L12.0187 12.1303L12.0262 12.1614C12.0328 12.1892 12.0426 12.231 12.0549 12.2851C12.0793 12.3933 12.1135 12.5507 12.1515 12.7446C12.2277 13.1333 12.3188 13.665 12.3793 14.2398C12.4401 14.8171 12.4685 15.4242 12.425 15.9679C12.3806 16.5234 12.2652 16.9532 12.0855 17.2226C12.0264 17.3113 11.9138 17.444 11.7594 17.6058C11.5605 17.8142 11.1314 17.7657 10.9967 17.3777C10.7141 16.5639 10.3334 15.5481 9.94904 14.777C8.88216 12.6365 8.19326 11.2759 6.33336 9.62597C5.99304 9.32406 5.58878 9.1089 5.19665 8.9539C4.31631 8.60592 3.75035 7.80549 3.89639 7.0753L4.24943 5.31012C4.36085 4.753 4.77748 4.30664 5.32562 4.15715L10.9254 2.62995C12.9052 2.08999 14.9227 3.36145 15.2898 5.38052L15.9738 9.14222C16.197 10.3699 15.2538 11.5 14.006 11.5H12.5015C12.3476 11.5 12.2022 11.5709 12.1074 11.6923C12.0127 11.8136 11.9792 11.972 12.0165 12.1214C12.0165 12.1214 12.0165 12.1213 12.0165 12.1214Z"}))},H=t(4795),J=t(7162),K=t.n(J);function W(e,a){var t="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(t)return(t=t.call(e)).next.bind(t);if(Array.isArray(e)||(t=function(e,a){if(!e)return;if("string"==typeof e)return z(e,a);var t=Object.prototype.toString.call(e).slice(8,-1);"Object"===t&&e.constructor&&(t=e.constructor.name);if("Map"===t||"Set"===t)return Array.from(e);if("Arguments"===t||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t))return z(e,a)}(e))||a&&e&&"number"==typeof e.length){t&&(e=t);var n=0;return function(){return n>=e.length?{done:!0}:{done:!1,value:e[n++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function z(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);tTSConfig Reference: "+c+""+n[c],o((function(e){return Object.assign({},e,{html:i})}));case 12:case"end":return a.stop()}}),a)})))).apply(this,arguments)}r.show&&function(){a.apply(this,arguments)}()}),[r.show,r.url,r.html]),r}(e.pageContext.lang),o=(0,n.useState)(t.frontmatter.deprecated_by),c=o[0],i=o[1],d=(0,x.D)((0,S.Z)()),l=(0,M.i)(e.pageContext.lang);if((0,n.useEffect)((function(){if(document.location.hash){var a,n=(null===(a=t.frontmatter)||void 0===a?void 0:a.deprecation_redirects)||[],u=n.indexOf(document.location.hash.slice(1));-1!==u&&i(n[u+1])}return document.querySelectorAll("#handbook-content nav ul li a").forEach((function(e){e.addEventListener("click",(function(e){e.preventDefault(),document.querySelector(decodeURIComponent(e.target.hash)).scrollIntoView({behavior:"smooth",block:"start"}),document.location.hash=e.target.hash}))})),window.addEventListener("scroll",N,{passive:!0,capture:!0}),N(),function(e,a){var t=document.getElementById("like-button"),n=document.getElementById("dislike-button");if(t&&n){var u=function(t){return function(){window.appInsights&&window.appInsights.trackEvent({name:t,properties:{slug:e,ab:"b"}});var n=a("handb_thanks"),u=document.getElementById("like-dislike-subnav"),r=document.getElementById("page-helpful-popup");u.innerHTML="
"+n+"
",r.innerHTML="

"+n+"

"}};t.onclick=u("Liked Page"),n.onclick=u("Disliked Page");var r=document.getElementById("like-button-popup"),o=document.getElementById("dislike-button-popup");r.onclick=u("Liked Page"),o.onclick=u("Disliked Page"),window.addEventListener("scroll",(function(){var e=document.body,a=document.documentElement,t=Math.max(e.scrollHeight,e.offsetHeight,a.clientHeight,a.scrollHeight,a.offsetHeight),n=Math.max(window.pageYOffset)+window.innerHeight>t-document.getElementById("site-footer").clientHeight+150,u=document.getElementById("page-helpful-popup"),r=document.getElementById("like-dislike-subnav");if(u&&r){var o=n?"1":"0";u.style.opacity!=o&&(u.style.opacity=o);var c=n?"0":"1";r.style.opacity!=c&&(r.style.opacity=c)}}),{passive:!0,capture:!0})}}(e.pageContext.slug,d),function(){window.removeEventListener("scroll",N)}}),[]),!t.frontmatter)throw new Error("No front-matter found for the file with props: "+e);if(!t.html)throw new Error("No html found for the file with props: "+e);var f=e.pageContext.id||"NO-ID",s=(null===(a=t.headings)||void 0===a?void 0:a.filter((function(e){return((null==e?void 0:e.depth)||0)<=2})))||[],D=!t.frontmatter.disable_toc,m=t.frontmatter.experimental,b=t.headings&&s.length<=30&&s.length>0,F=(0,v.m8)(e.pageContext.lang),p=t.frontmatter.handbook?"Handbook":"Documentation",h=B()();return n.createElement(u.A,{title:p+" - "+t.frontmatter.title,description:t.frontmatter.oneline||"",lang:e.pageContext.lang},n.createElement("section",{id:"doc-layout"},n.createElement(g,null),n.createElement("div",{className:"page-popup",id:"page-helpful-popup",style:{opacity:0}},n.createElement("p",null,"Was this page helpful?"),n.createElement("div",null,n.createElement("button",{className:"first",id:"like-button-popup",title:"Like this page"},n.createElement(_,null)),n.createElement("button",{id:"dislike-button-popup",title:"Dislike this page"},n.createElement(O,null)))),n.createElement("noscript",null,n.createElement("style",{dangerouslySetInnerHTML:{__html:"\n nav#sidebar > ul > li.closed ul {\n display: block !important;\n }\n "}})),n.createElement(E,{navItems:F,selectedID:f}),n.createElement("div",{id:"handbook-content",role:"article"},c&&n.createElement(n.Fragment,null,n.createElement($.Z,null,n.createElement("link",{rel:"canonical",href:"https://www.lf-lang.org"+t.frontmatter.deprecated_by})),n.createElement("div",{id:"deprecated-header"},n.createElement("div",{id:"deprecated-content"},n.createElement("div",{id:"deprecated-icon"},n.createElement("svg",{width:"16",height:"16",viewBox:"0 0 16 16",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("circle",{cx:"8",cy:"8",r:"7.5",stroke:"black"}),n.createElement("path",{d:"M8 3V9",stroke:"black"}),n.createElement("path",{d:"M8 11L8 13",stroke:"black"}))),n.createElement("div",null,n.createElement("h3",null,d("handb_deprecated_title")),n.createElement("p",null,d("handb_deprecated_subtitle"),n.createElement(l,{className:"deprecation-redirect-link",to:c},d("handb_deprecated_subtitle_link"))))),n.createElement("div",{id:"deprecated-action"},n.createElement(l,{className:"deprecation-redirect-link",to:c},d("handb_deprecated_subtitle_action"))))),m&&n.createElement("div",{id:"deprecated-header"},n.createElement("div",{id:"deprecated-content"},n.createElement("div",{id:"deprecated-icon"},n.createElement("svg",{width:"16",height:"16",viewBox:"0 0 16 16",fill:"none",xmlns:"http://www.w3.org/2000/svg"},n.createElement("circle",{cx:"8",cy:"8",r:"7.5",stroke:"black"}),n.createElement("path",{d:"M8 3V9",stroke:"black"}),n.createElement("path",{d:"M8 11L8 13",stroke:"black"}))),n.createElement("div",null,n.createElement("h3",null,d("handb_experimental_title")),n.createElement("p",null,d("handb_experimental_subtitle"))))),n.createElement("h2",null,t.frontmatter.title),t.frontmatter.preamble&&n.createElement("div",{className:"preamble",dangerouslySetInnerHTML:{__html:t.frontmatter.preamble}}),n.createElement("article",null,n.createElement("div",{className:"whitespace raised"},n.createElement("div",{className:"markdown",dangerouslySetInnerHTML:{__html:C.postProcessHTML(t.html)}})),D&&n.createElement("aside",{className:"handbook-toc"},n.createElement("nav",{className:c?"deprecated":""},b&&n.createElement(n.Fragment,null,n.createElement("h5",null,d("handb_on_this_page")),n.createElement("ul",null,s.map((function(e){var a=h.slug(e.value,!1);return n.createElement("li",{key:a},n.createElement("a",{href:"#"+a},e.value))})))),n.createElement("div",{id:"like-dislike-subnav"},n.createElement("h5",null,d("handb_like_dislike_title")),n.createElement("div",null,n.createElement("button",{title:"Like this page",id:"like-button"},n.createElement(_,null)," ",d("handb_like_desc")),n.createElement("button",{title:"Dislike this page",id:"dislike-button"},n.createElement(O,null)," ",d("handb_dislike_desc"))))))),n.createElement(k,{next:e.data.next,prev:e.data.prev,i:d,IntlLink:l}),n.createElement(P,{lang:e.pageContext.lang,i:d,path:e.pageContext.repoPath,lastEdited:e.pageContext.modifiedTime}))),n.createElement(Z,r))},V=function(e){return n.createElement(h.R,{locale:e.pageContext.lang},n.createElement(U,e))}},8711:function(e){function a(e,a){var n="undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(n)return(n=n.call(e)).next.bind(n);if(Array.isArray(e)||(n=function(e,a){if(!e)return;if("string"==typeof e)return t(e,a);var n=Object.prototype.toString.call(e).slice(8,-1);"Object"===n&&e.constructor&&(n=e.constructor.name);if("Map"===n||"Set"===n)return Array.from(e);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return t(e,a)}(e))||a&&e&&"number"==typeof e.length){n&&(e=n);var u=0;return function(){return u>=e.length?{done:!0}:{done:!1,value:e[u++]}}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function t(e,a){(null==a||a>e.length)&&(a=e.length);for(var t=0,n=new Array(a);tThis page is showing examples in the target language [C]{lf-c}[C++]{lf-cpp}[Python]{lf-py}[TypeScript]{lf-ts}[Rust]{lf-rs}.\nYou can change the target language in the left sidebar.'},o=new RegExp("\\b(?:"+n.join("|")+")\\b"),i=new RegExp("\\$("+n.join("|")+")\\$","gm"),d=new RegExp("\\$("+["start","end"].join("|")+")\\((.*)\\)\\$","gm"),l=new RegExp("\\$("+Object.keys(r).join("|")+")\\$","gm"),f=/\[([^\]]*)\]\{([^\}]*)\}/gm;function s(e,t,n){for(var r,o=[],i=a(n.split(" "));!(r=i()).done;)c=r.value,u[c]?Array.prototype.push.apply(o,u[c]):o.push(c);return''+t+""}function D(e,a){return''+a+""}function m(e,a){return r[a]?r[a]:'ERROR: textSubstitutions key not found.'}e.exports={postProcessHTML:function(e){var a=e.replace(l,m);return a=(a=(a=a.replace(f,s)).replace(i,D)).replace(d,"")},keywordMatcher:o}},6209:function(e,a,t){var n=t(3965);e.exports=r;var u=Object.hasOwnProperty;function r(){if(!(this instanceof r))return new r;this.reset()}function o(e,a){return"string"!=typeof e?"":(a||(e=e.toLowerCase()),e.replace(n,"").replace(/ /g,"-"))}r.prototype.slug=function(e,a){for(var t=this,n=o(e,!0===a),r=n;u.call(t.occurrences,n);)t.occurrences[r]++,n=r+"-"+t.occurrences[r];return t.occurrences[n]=0,n},r.prototype.reset=function(){this.occurrences=Object.create(null)},r.slug=o},3965:function(e){e.exports=/[\0-\x1F!-,\.\/:-@\[-\^`\{-\xA9\xAB-\xB4\xB6-\xB9\xBB-\xBF\xD7\xF7\u02C2-\u02C5\u02D2-\u02DF\u02E5-\u02EB\u02ED\u02EF-\u02FF\u0375\u0378\u0379\u037E\u0380-\u0385\u0387\u038B\u038D\u03A2\u03F6\u0482\u0530\u0557\u0558\u055A-\u055F\u0589-\u0590\u05BE\u05C0\u05C3\u05C6\u05C8-\u05CF\u05EB-\u05EE\u05F3-\u060F\u061B-\u061F\u066A-\u066D\u06D4\u06DD\u06DE\u06E9\u06FD\u06FE\u0700-\u070F\u074B\u074C\u07B2-\u07BF\u07F6-\u07F9\u07FB\u07FC\u07FE\u07FF\u082E-\u083F\u085C-\u085F\u086B-\u089F\u08B5\u08BE-\u08D2\u08E2\u0964\u0965\u0970\u0984\u098D\u098E\u0991\u0992\u09A9\u09B1\u09B3-\u09B5\u09BA\u09BB\u09C5\u09C6\u09C9\u09CA\u09CF-\u09D6\u09D8-\u09DB\u09DE\u09E4\u09E5\u09F2-\u09FB\u09FD\u09FF\u0A00\u0A04\u0A0B-\u0A0E\u0A11\u0A12\u0A29\u0A31\u0A34\u0A37\u0A3A\u0A3B\u0A3D\u0A43-\u0A46\u0A49\u0A4A\u0A4E-\u0A50\u0A52-\u0A58\u0A5D\u0A5F-\u0A65\u0A76-\u0A80\u0A84\u0A8E\u0A92\u0AA9\u0AB1\u0AB4\u0ABA\u0ABB\u0AC6\u0ACA\u0ACE\u0ACF\u0AD1-\u0ADF\u0AE4\u0AE5\u0AF0-\u0AF8\u0B00\u0B04\u0B0D\u0B0E\u0B11\u0B12\u0B29\u0B31\u0B34\u0B3A\u0B3B\u0B45\u0B46\u0B49\u0B4A\u0B4E-\u0B55\u0B58-\u0B5B\u0B5E\u0B64\u0B65\u0B70\u0B72-\u0B81\u0B84\u0B8B-\u0B8D\u0B91\u0B96-\u0B98\u0B9B\u0B9D\u0BA0-\u0BA2\u0BA5-\u0BA7\u0BAB-\u0BAD\u0BBA-\u0BBD\u0BC3-\u0BC5\u0BC9\u0BCE\u0BCF\u0BD1-\u0BD6\u0BD8-\u0BE5\u0BF0-\u0BFF\u0C0D\u0C11\u0C29\u0C3A-\u0C3C\u0C45\u0C49\u0C4E-\u0C54\u0C57\u0C5B-\u0C5F\u0C64\u0C65\u0C70-\u0C7F\u0C84\u0C8D\u0C91\u0CA9\u0CB4\u0CBA\u0CBB\u0CC5\u0CC9\u0CCE-\u0CD4\u0CD7-\u0CDD\u0CDF\u0CE4\u0CE5\u0CF0\u0CF3-\u0CFF\u0D04\u0D0D\u0D11\u0D45\u0D49\u0D4F-\u0D53\u0D58-\u0D5E\u0D64\u0D65\u0D70-\u0D79\u0D80\u0D81\u0D84\u0D97-\u0D99\u0DB2\u0DBC\u0DBE\u0DBF\u0DC7-\u0DC9\u0DCB-\u0DCE\u0DD5\u0DD7\u0DE0-\u0DE5\u0DF0\u0DF1\u0DF4-\u0E00\u0E3B-\u0E3F\u0E4F\u0E5A-\u0E80\u0E83\u0E85\u0E8B\u0EA4\u0EA6\u0EBE\u0EBF\u0EC5\u0EC7\u0ECE\u0ECF\u0EDA\u0EDB\u0EE0-\u0EFF\u0F01-\u0F17\u0F1A-\u0F1F\u0F2A-\u0F34\u0F36\u0F38\u0F3A-\u0F3D\u0F48\u0F6D-\u0F70\u0F85\u0F98\u0FBD-\u0FC5\u0FC7-\u0FFF\u104A-\u104F\u109E\u109F\u10C6\u10C8-\u10CC\u10CE\u10CF\u10FB\u1249\u124E\u124F\u1257\u1259\u125E\u125F\u1289\u128E\u128F\u12B1\u12B6\u12B7\u12BF\u12C1\u12C6\u12C7\u12D7\u1311\u1316\u1317\u135B\u135C\u1360-\u137F\u1390-\u139F\u13F6\u13F7\u13FE-\u1400\u166D\u166E\u1680\u169B-\u169F\u16EB-\u16ED\u16F9-\u16FF\u170D\u1715-\u171F\u1735-\u173F\u1754-\u175F\u176D\u1771\u1774-\u177F\u17D4-\u17D6\u17D8-\u17DB\u17DE\u17DF\u17EA-\u180A\u180E\u180F\u181A-\u181F\u1879-\u187F\u18AB-\u18AF\u18F6-\u18FF\u191F\u192C-\u192F\u193C-\u1945\u196E\u196F\u1975-\u197F\u19AC-\u19AF\u19CA-\u19CF\u19DA-\u19FF\u1A1C-\u1A1F\u1A5F\u1A7D\u1A7E\u1A8A-\u1A8F\u1A9A-\u1AA6\u1AA8-\u1AAF\u1ABF-\u1AFF\u1B4C-\u1B4F\u1B5A-\u1B6A\u1B74-\u1B7F\u1BF4-\u1BFF\u1C38-\u1C3F\u1C4A-\u1C4C\u1C7E\u1C7F\u1C89-\u1C8F\u1CBB\u1CBC\u1CC0-\u1CCF\u1CD3\u1CFB-\u1CFF\u1DFA\u1F16\u1F17\u1F1E\u1F1F\u1F46\u1F47\u1F4E\u1F4F\u1F58\u1F5A\u1F5C\u1F5E\u1F7E\u1F7F\u1FB5\u1FBD\u1FBF-\u1FC1\u1FC5\u1FCD-\u1FCF\u1FD4\u1FD5\u1FDC-\u1FDF\u1FED-\u1FF1\u1FF5\u1FFD-\u203E\u2041-\u2053\u2055-\u2070\u2072-\u207E\u2080-\u208F\u209D-\u20CF\u20F1-\u2101\u2103-\u2106\u2108\u2109\u2114\u2116-\u2118\u211E-\u2123\u2125\u2127\u2129\u212E\u213A\u213B\u2140-\u2144\u214A-\u214D\u214F-\u215F\u2189-\u24B5\u24EA-\u2BFF\u2C2F\u2C5F\u2CE5-\u2CEA\u2CF4-\u2CFF\u2D26\u2D28-\u2D2C\u2D2E\u2D2F\u2D68-\u2D6E\u2D70-\u2D7E\u2D97-\u2D9F\u2DA7\u2DAF\u2DB7\u2DBF\u2DC7\u2DCF\u2DD7\u2DDF\u2E00-\u2E2E\u2E30-\u3004\u3008-\u3020\u3030\u3036\u3037\u303D-\u3040\u3097\u3098\u309B\u309C\u30A0\u30FB\u3100-\u3104\u3130\u318F-\u319F\u31BB-\u31EF\u3200-\u33FF\u4DB6-\u4DFF\u9FF0-\u9FFF\uA48D-\uA4CF\uA4FE\uA4FF\uA60D-\uA60F\uA62C-\uA63F\uA673\uA67E\uA6F2-\uA716\uA720\uA721\uA789\uA78A\uA7C0\uA7C1\uA7C7-\uA7F6\uA828-\uA83F\uA874-\uA87F\uA8C6-\uA8CF\uA8DA-\uA8DF\uA8F8-\uA8FA\uA8FC\uA92E\uA92F\uA954-\uA95F\uA97D-\uA97F\uA9C1-\uA9CE\uA9DA-\uA9DF\uA9FF\uAA37-\uAA3F\uAA4E\uAA4F\uAA5A-\uAA5F\uAA77-\uAA79\uAAC3-\uAADA\uAADE\uAADF\uAAF0\uAAF1\uAAF7-\uAB00\uAB07\uAB08\uAB0F\uAB10\uAB17-\uAB1F\uAB27\uAB2F\uAB5B\uAB68-\uAB6F\uABEB\uABEE\uABEF\uABFA-\uABFF\uD7A4-\uD7AF\uD7C7-\uD7CA\uD7FC-\uD7FF\uE000-\uF8FF\uFA6E\uFA6F\uFADA-\uFAFF\uFB07-\uFB12\uFB18-\uFB1C\uFB29\uFB37\uFB3D\uFB3F\uFB42\uFB45\uFBB2-\uFBD2\uFD3E-\uFD4F\uFD90\uFD91\uFDC8-\uFDEF\uFDFC-\uFDFF\uFE10-\uFE1F\uFE30-\uFE32\uFE35-\uFE4C\uFE50-\uFE6F\uFE75\uFEFD-\uFF0F\uFF1A-\uFF20\uFF3B-\uFF3E\uFF40\uFF5B-\uFF65\uFFBF-\uFFC1\uFFC8\uFFC9\uFFD0\uFFD1\uFFD8\uFFD9\uFFDD-\uFFFF]|\uD800[\uDC0C\uDC27\uDC3B\uDC3E\uDC4E\uDC4F\uDC5E-\uDC7F\uDCFB-\uDD3F\uDD75-\uDDFC\uDDFE-\uDE7F\uDE9D-\uDE9F\uDED1-\uDEDF\uDEE1-\uDEFF\uDF20-\uDF2C\uDF4B-\uDF4F\uDF7B-\uDF7F\uDF9E\uDF9F\uDFC4-\uDFC7\uDFD0\uDFD6-\uDFFF]|\uD801[\uDC9E\uDC9F\uDCAA-\uDCAF\uDCD4-\uDCD7\uDCFC-\uDCFF\uDD28-\uDD2F\uDD64-\uDDFF\uDF37-\uDF3F\uDF56-\uDF5F\uDF68-\uDFFF]|\uD802[\uDC06\uDC07\uDC09\uDC36\uDC39-\uDC3B\uDC3D\uDC3E\uDC56-\uDC5F\uDC77-\uDC7F\uDC9F-\uDCDF\uDCF3\uDCF6-\uDCFF\uDD16-\uDD1F\uDD3A-\uDD7F\uDDB8-\uDDBD\uDDC0-\uDDFF\uDE04\uDE07-\uDE0B\uDE14\uDE18\uDE36\uDE37\uDE3B-\uDE3E\uDE40-\uDE5F\uDE7D-\uDE7F\uDE9D-\uDEBF\uDEC8\uDEE7-\uDEFF\uDF36-\uDF3F\uDF56-\uDF5F\uDF73-\uDF7F\uDF92-\uDFFF]|\uD803[\uDC49-\uDC7F\uDCB3-\uDCBF\uDCF3-\uDCFF\uDD28-\uDD2F\uDD3A-\uDEFF\uDF1D-\uDF26\uDF28-\uDF2F\uDF51-\uDFDF\uDFF7-\uDFFF]|\uD804[\uDC47-\uDC65\uDC70-\uDC7E\uDCBB-\uDCCF\uDCE9-\uDCEF\uDCFA-\uDCFF\uDD35\uDD40-\uDD43\uDD47-\uDD4F\uDD74\uDD75\uDD77-\uDD7F\uDDC5-\uDDC8\uDDCD-\uDDCF\uDDDB\uDDDD-\uDDFF\uDE12\uDE38-\uDE3D\uDE3F-\uDE7F\uDE87\uDE89\uDE8E\uDE9E\uDEA9-\uDEAF\uDEEB-\uDEEF\uDEFA-\uDEFF\uDF04\uDF0D\uDF0E\uDF11\uDF12\uDF29\uDF31\uDF34\uDF3A\uDF45\uDF46\uDF49\uDF4A\uDF4E\uDF4F\uDF51-\uDF56\uDF58-\uDF5C\uDF64\uDF65\uDF6D-\uDF6F\uDF75-\uDFFF]|\uD805[\uDC4B-\uDC4F\uDC5A-\uDC5D\uDC60-\uDC7F\uDCC6\uDCC8-\uDCCF\uDCDA-\uDD7F\uDDB6\uDDB7\uDDC1-\uDDD7\uDDDE-\uDDFF\uDE41-\uDE43\uDE45-\uDE4F\uDE5A-\uDE7F\uDEB9-\uDEBF\uDECA-\uDEFF\uDF1B\uDF1C\uDF2C-\uDF2F\uDF3A-\uDFFF]|\uD806[\uDC3B-\uDC9F\uDCEA-\uDCFE\uDD00-\uDD9F\uDDA8\uDDA9\uDDD8\uDDD9\uDDE2\uDDE5-\uDDFF\uDE3F-\uDE46\uDE48-\uDE4F\uDE9A-\uDE9C\uDE9E-\uDEBF\uDEF9-\uDFFF]|\uD807[\uDC09\uDC37\uDC41-\uDC4F\uDC5A-\uDC71\uDC90\uDC91\uDCA8\uDCB7-\uDCFF\uDD07\uDD0A\uDD37-\uDD39\uDD3B\uDD3E\uDD48-\uDD4F\uDD5A-\uDD5F\uDD66\uDD69\uDD8F\uDD92\uDD99-\uDD9F\uDDAA-\uDEDF\uDEF7-\uDFFF]|\uD808[\uDF9A-\uDFFF]|\uD809[\uDC6F-\uDC7F\uDD44-\uDFFF]|[\uD80A\uD80B\uD80E-\uD810\uD812-\uD819\uD823-\uD82B\uD82D\uD82E\uD830-\uD833\uD837\uD839\uD83D-\uD83F\uD87B-\uD87D\uD87F-\uDB3F\uDB41-\uDBFF][\uDC00-\uDFFF]|\uD80D[\uDC2F-\uDFFF]|\uD811[\uDE47-\uDFFF]|\uD81A[\uDE39-\uDE3F\uDE5F\uDE6A-\uDECF\uDEEE\uDEEF\uDEF5-\uDEFF\uDF37-\uDF3F\uDF44-\uDF4F\uDF5A-\uDF62\uDF78-\uDF7C\uDF90-\uDFFF]|\uD81B[\uDC00-\uDE3F\uDE80-\uDEFF\uDF4B-\uDF4E\uDF88-\uDF8E\uDFA0-\uDFDF\uDFE2\uDFE4-\uDFFF]|\uD821[\uDFF8-\uDFFF]|\uD822[\uDEF3-\uDFFF]|\uD82C[\uDD1F-\uDD4F\uDD53-\uDD63\uDD68-\uDD6F\uDEFC-\uDFFF]|\uD82F[\uDC6B-\uDC6F\uDC7D-\uDC7F\uDC89-\uDC8F\uDC9A-\uDC9C\uDC9F-\uDFFF]|\uD834[\uDC00-\uDD64\uDD6A-\uDD6C\uDD73-\uDD7A\uDD83\uDD84\uDD8C-\uDDA9\uDDAE-\uDE41\uDE45-\uDFFF]|\uD835[\uDC55\uDC9D\uDCA0\uDCA1\uDCA3\uDCA4\uDCA7\uDCA8\uDCAD\uDCBA\uDCBC\uDCC4\uDD06\uDD0B\uDD0C\uDD15\uDD1D\uDD3A\uDD3F\uDD45\uDD47-\uDD49\uDD51\uDEA6\uDEA7\uDEC1\uDEDB\uDEFB\uDF15\uDF35\uDF4F\uDF6F\uDF89\uDFA9\uDFC3\uDFCC\uDFCD]|\uD836[\uDC00-\uDDFF\uDE37-\uDE3A\uDE6D-\uDE74\uDE76-\uDE83\uDE85-\uDE9A\uDEA0\uDEB0-\uDFFF]|\uD838[\uDC07\uDC19\uDC1A\uDC22\uDC25\uDC2B-\uDCFF\uDD2D-\uDD2F\uDD3E\uDD3F\uDD4A-\uDD4D\uDD4F-\uDEBF\uDEFA-\uDFFF]|\uD83A[\uDCC5-\uDCCF\uDCD7-\uDCFF\uDD4C-\uDD4F\uDD5A-\uDFFF]|\uD83B[\uDC00-\uDDFF\uDE04\uDE20\uDE23\uDE25\uDE26\uDE28\uDE33\uDE38\uDE3A\uDE3C-\uDE41\uDE43-\uDE46\uDE48\uDE4A\uDE4C\uDE50\uDE53\uDE55\uDE56\uDE58\uDE5A\uDE5C\uDE5E\uDE60\uDE63\uDE65\uDE66\uDE6B\uDE73\uDE78\uDE7D\uDE7F\uDE8A\uDE9C-\uDEA0\uDEA4\uDEAA\uDEBC-\uDFFF]|\uD83C[\uDC00-\uDD2F\uDD4A-\uDD4F\uDD6A-\uDD6F\uDD8A-\uDFFF]|\uD869[\uDED7-\uDEFF]|\uD86D[\uDF35-\uDF3F]|\uD86E[\uDC1E\uDC1F]|\uD873[\uDEA2-\uDEAF]|\uD87A[\uDFE1-\uDFFF]|\uD87E[\uDE1E-\uDFFF]|\uDB40[\uDC00-\uDCFF\uDDF0-\uDFFF]/g}}]); +//# sourceMappingURL=component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js.map \ No newline at end of file diff --git a/component---src-templates-documentation-tsx-04d26b5abab5434898b4.js.map b/component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js.map similarity index 99% rename from component---src-templates-documentation-tsx-04d26b5abab5434898b4.js.map rename to component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js.map index 952e9a25d..4e53c3713 100644 --- a/component---src-templates-documentation-tsx-04d26b5abab5434898b4.js.map +++ b/component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js.map @@ -1 +1 @@ -{"version":3,"file":"component---src-templates-documentation-tsx-04d26b5abab5434898b4.js","mappings":"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","sources":["webpack://lingua-franca/./src/lib/documentationNavigation.ts","webpack://lingua-franca/./src/components/layout/Sidebar-keyboard.tsx","webpack://lingua-franca/./src/components/layout/Sidebar.tsx","webpack://lingua-franca/./src/components/handbook/NextPrev.tsx","webpack://lingua-franca/./src/components/handbook/Contributors.tsx","webpack://lingua-franca/./src/templates/scripts/setupSubNavigationSidebar.ts","webpack://lingua-franca/./src/components/svgs/documentation.tsx","webpack://lingua-franca/./src/components/Popup.tsx","webpack://lingua-franca/./src/templates/documentation.tsx","webpack://lingua-franca/./src/templates/scripts/setupLikeDislikeButtons.ts","webpack://lingua-franca/../documentation/scripts/linguaFrancaUtils.js","webpack://lingua-franca/../../node_modules/github-slugger/index.js","webpack://lingua-franca/../../node_modules/github-slugger/regex.js"],"sourcesContent":["/* This function is completely auto-generated via the `yarn bootstrap` phase of\n the app. You can re-run it when adding new localized handbook pages by running:\n\n yarn workspace documentation create-handbook-nav\n\n Find the source of truth at packages/documentation/scripts/generateDocsNavigationPerLanguage.js\n*/\n\nexport interface SidebarNavItem {\n title: string;\n id: string;\n permalink?: string;\n chronological?: boolean;\n oneline?: string;\n items?: SidebarNavItem[];\n}\n\n/** ---INSERT--- */\n\nexport function getDocumentationNavForLanguage(\n langRequest: string\n): SidebarNavItem[] {\n const langs = [\"en\"];\n const lang = langs.includes(langRequest) ? langRequest : \"en\";\n const navigations: Record = {};\n\n navigations.en = [\n {\n title: \"Resources\",\n oneline: \"Overview of the project.\",\n id: \"resources\",\n chronological: true,\n\n items: [\n {\n title: \"Overview\",\n id: \"0-overview\",\n permalink: \"/docs/handbook/overview\",\n oneline: \"Overview of Lingua Franca.\",\n },\n {\n title: \"Tutorial Video\",\n id: \"0-tutorial-video\",\n permalink: \"/docs/handbook/tutorial-video\",\n oneline: \"Tutorial video presented by the Lingua Franca team.\",\n },\n ],\n },\n {\n title: \"Writing Reactors\",\n oneline: \"Introduction to writing reactors:\",\n id: \"writing-reactors\",\n chronological: true,\n\n items: [\n {\n title: \"A First Reactor\",\n id: \"1-a-first-reactor\",\n permalink: \"/docs/handbook/a-first-reactor\",\n oneline: \"Writing your first Lingua Franca reactor.\",\n },\n {\n title: \"Inputs and Outputs\",\n id: \"1-inputs-and-outputs\",\n permalink: \"/docs/handbook/inputs-and-outputs\",\n oneline: \"Inputs, outputs, and reactions in Lingua Franca.\",\n },\n {\n title: \"Parameters and State Variables\",\n id: \"1-parameters-and-state-variables\",\n permalink: \"/docs/handbook/parameters-and-state-variables\",\n oneline: \"Parameters and state variables in Lingua Franca.\",\n },\n {\n title: \"Time and Timers\",\n id: \"1-time-and-timers\",\n permalink: \"/docs/handbook/time-and-timers\",\n oneline: \"Time and timers in Lingua Franca.\",\n },\n {\n title: \"Composing Reactors\",\n id: \"1-composing-reactors\",\n permalink: \"/docs/handbook/composing-reactors\",\n oneline: \"Composing reactors in Lingua Franca.\",\n },\n {\n title: \"Reactions\",\n id: \"1-reactions\",\n permalink: \"/docs/handbook/reactions\",\n oneline: \"Reactions in Lingua Franca.\",\n },\n {\n title: \"Methods\",\n id: \"1-methods\",\n permalink: \"/docs/handbook/methods\",\n oneline: \"Methods in Lingua Franca.\",\n },\n {\n title: \"Causality Loops\",\n id: \"1-causality-loops\",\n permalink: \"/docs/handbook/causality-loops\",\n oneline: \"Causality loops in Lingua Franca.\",\n },\n {\n title: \"Extending Reactors\",\n id: \"1-extending-reactors\",\n permalink: \"/docs/handbook/extending-reactors\",\n oneline: \"Extending reactors in Lingua Franca.\",\n },\n {\n title: \"Actions\",\n id: \"1-actions\",\n permalink: \"/docs/handbook/actions\",\n oneline: \"Actions in Lingua Franca.\",\n },\n {\n title: \"Superdense Time\",\n id: \"1-superdense-time\",\n permalink: \"/docs/handbook/superdense-time\",\n oneline: \"Superdense time in Lingua Franca.\",\n },\n {\n title: \"Modal Reactors\",\n id: \"1-modal-reactors\",\n permalink: \"/docs/handbook/modal-models\",\n oneline: \"Modal Reactors\",\n },\n {\n title: \"Deadlines\",\n id: \"1-deadlines\",\n permalink: \"/docs/handbook/deadlines\",\n oneline: \"Deadlines in Lingua Franca.\",\n },\n {\n title: \"Multiports and Banks\",\n id: \"1-multiports-and-banks\",\n permalink: \"/docs/handbook/multiports-and-banks\",\n oneline: \"Multiports and Banks of Reactors.\",\n },\n {\n title: \"Generic Reactors\",\n id: \"1-generic-reactors\",\n permalink: \"/docs/handbook/generics\",\n oneline: \"Defining generic reactors in Lingua Franca.\",\n },\n {\n title: \"Preambles\",\n id: \"1-preambles\",\n permalink: \"/docs/handbook/preambles\",\n oneline: \"Defining preambles in Lingua Franca.\",\n },\n {\n title: \"Distributed Execution\",\n id: \"1-distributed-execution\",\n permalink: \"/docs/handbook/distributed-execution\",\n oneline: \"Distributed Execution (preliminary)\",\n },\n {\n title: \"Termination\",\n id: \"1-termination\",\n permalink: \"/docs/handbook/termination\",\n oneline: \"Terminating a Lingua Franca execution.\",\n },\n ],\n },\n {\n title: \"Tools\",\n oneline: \"Tools for developing Lingua Franca programs.\",\n id: \"tools\",\n chronological: true,\n\n items: [\n {\n title: \"Code Extension\",\n id: \"2-code-extension\",\n permalink: \"/docs/handbook/code-extension\",\n oneline: \"Visual Studio Code Extension for Lingua Franca.\",\n },\n {\n title: \"Epoch IDE\",\n id: \"2-epoch-ide\",\n permalink: \"/docs/handbook/epoch-ide\",\n oneline: \"Epoch IDE for Lingua Franca.\",\n },\n {\n title: \"Command Line Tools\",\n id: \"2-command-line-tools\",\n permalink: \"/docs/handbook/command-line-tools\",\n oneline: \"Command-line tools for Lingua Franca.\",\n },\n {\n title: \"Troubleshooting\",\n id: \"2-troubleshooting\",\n permalink: \"/docs/handbook/troubleshooting\",\n oneline: \"Troubleshooting page for Lingua Franca tools.\",\n },\n ],\n },\n {\n title: \"Reference\",\n oneline: \"Reference documentation.\",\n id: \"reference\",\n chronological: true,\n\n items: [\n {\n title: \"Expressions\",\n id: \"3-expressions\",\n permalink: \"/docs/handbook/expressions\",\n oneline: \"Expressions in Lingua Franca.\",\n },\n {\n title: \"Target Language Details\",\n id: \"3-target-language-details\",\n permalink: \"/docs/handbook/target-language-details\",\n oneline: \"Detailed reference for each target langauge.\",\n },\n {\n title: \"Target Declaration\",\n id: \"3-target-declaration\",\n permalink: \"/docs/handbook/target-declaration\",\n oneline:\n \"The target declaration and its parameters in Lingua Franca.\",\n },\n {\n title: \"Tracing\",\n id: \"3-tracing\",\n permalink: \"/docs/handbook/tracing\",\n oneline: \"Tracing (preliminary)\",\n },\n {\n title: \"Containerized Execution\",\n id: \"3-containerized-execution\",\n permalink: \"/docs/handbook/containerized-execution\",\n oneline: \"Containerized Execution using Docker\",\n },\n {\n title: \"Security\",\n id: \"3-security\",\n permalink: \"/docs/handbook/security\",\n oneline: \"Secure Federated Execution\",\n },\n ],\n },\n {\n title: \"Embedded Platforms\",\n oneline:\n \"Documentation for developing Lingua Franca on Embedded Platforms.\",\n id: \"embedded-platforms\",\n chronological: true,\n\n items: [\n {\n title: \"Arduino\",\n id: \"4-arduino\",\n permalink: \"/docs/handbook/arduino\",\n oneline: \"Developing LF Programs on Arduino.\",\n },\n {\n title: \"Zephyr\",\n id: \"4-zephyr\",\n permalink: \"/docs/handbook/zephyr\",\n oneline: \"Developing LF Programs for Zephyr RTOS.\",\n },\n ],\n },\n {\n title: \"Developer\",\n oneline:\n \"Information for developers of the Lingua Franca language and tools.\",\n id: \"developer\",\n chronological: true,\n\n items: [\n {\n title: \"Contributing\",\n id: \"5-contributing\",\n permalink: \"/docs/handbook/contributing\",\n oneline: \"Contribute to Lingua Franca.\",\n },\n {\n title: \"Developer Setup\",\n id: \"5-developer-setup\",\n permalink: \"/docs/handbook/developer-setup\",\n oneline: \"Setting up Lingua Franca for developers.\",\n },\n {\n title: \"Developer IntelliJ Setup\",\n id: \"5-developer-intellij-setup\",\n permalink: \"/docs/handbook/intellij\",\n oneline: \"Developer IntelliJ Setup.\",\n },\n {\n title: \"Regression Tests\",\n id: \"5-regression-tests\",\n permalink: \"/docs/handbook/regression-tests\",\n oneline: \"Regression Tests for Lingua Franca.\",\n },\n {\n title: \"Running Benchmarks\",\n id: \"5-running-benchmarks\",\n permalink: \"/docs/handbook/running-benchmarks\",\n oneline: \"Running Benchmarks.\",\n },\n {\n title: \"Website Development\",\n id: \"5-website-development\",\n permalink: \"/docs/handbook/website-development\",\n oneline: \"Development of the Lingua Franca website.\",\n },\n ],\n },\n ];\n\n return navigations[lang];\n}\n\n/** ---INSERT-END--- */\n\nconst findInNav = (\n item: SidebarNavItem | SidebarNavItem[],\n fun: (item: SidebarNavItem) => boolean\n): SidebarNavItem | undefined => {\n if (Array.isArray(item)) {\n for (const subItem of item) {\n const sub = findInNav(subItem, fun);\n if (sub) return sub;\n }\n } else {\n if (fun(item)) return item;\n if (!item.items) return undefined;\n for (const subItem of item.items) {\n const sub = findInNav(subItem, fun);\n if (sub) return sub;\n }\n return undefined;\n }\n};\n\nexport function getNextPageID(navs: SidebarNavItem[], currentID: string) {\n // prettier-ignore\n const section = findInNav(navs, (i) => i && !!i.items && !!i.items.find(i => i.id === currentID)) || false\n if (!section) return undefined;\n if (!section.chronological) return undefined;\n if (!section.items) return;\n\n const currentIndex = section.items.findIndex((i) => i.id === currentID);\n const next = section.items[currentIndex + 1];\n if (next) {\n if (next.items) {\n return {\n path: next.items[0].permalink,\n ...section.items[currentIndex + 1],\n };\n } else {\n return {\n path: next.permalink,\n ...section.items[currentIndex + 1],\n };\n }\n }\n}\n\nexport function getPreviousPageID(navs: SidebarNavItem[], currentID: string) {\n // prettier-ignore\n const section = findInNav(navs, (i) => i && !!i.items && !!i.items.find(i => i.id === currentID)) || false\n\n if (!section) return undefined;\n if (!section.chronological) return undefined;\n if (!section.items) return;\n\n const currentIndex = section.items.findIndex((i) => i.id === currentID);\n const prev = section.items[currentIndex - 1];\n\n if (prev) {\n return {\n path: prev.permalink,\n ...section.items[currentIndex - 1],\n };\n }\n}\n","import { KeyboardEventHandler } from \"react\"\nimport { getTagFromParents } from \"./Sidebar\"\n\nconst UpArrow = 38\nconst DownArrow = 40\n\nconst childOfType = (tag: string, element: any) => {\n let found: HTMLElement | undefined\n for (const e of element.children) {\n if (e.nodeName === tag.toUpperCase()) found = e\n }\n return found\n}\n\n/**\n * Handles moving up and down through the navigation hierarchy\n * selecting leaf nodes and jumping up into section categories\n */\nexport const onAnchorKeyDown: KeyboardEventHandler = event => {\n const li = getTagFromParents(\"li\", event.target as any)\n\n // Up, and jump into section headers\n if (event.keyCode == UpArrow) {\n const aboveLI = li.previousElementSibling\n const a = aboveLI && childOfType(\"a\", aboveLI)\n const button = aboveLI && childOfType(\"button\", aboveLI)\n\n if (a) {\n // next link\n a.focus()\n } else if (aboveLI && button) {\n // Jump to the subnav above, either at the bottom item if open or\n // the main button otherwise\n const open = aboveLI.classList.contains(\"open\")\n if (open) {\n const listOfLinks = childOfType(\"ul\", aboveLI)!\n const lastLI = listOfLinks.lastElementChild\n childOfType(\"a\", lastLI)!.focus()\n } else {\n button.focus()\n }\n } else {\n // at the top\n const sectionHostingLI = getTagFromParents(\"li\", li)\n childOfType(\"button\", sectionHostingLI)!.focus()\n }\n\n event.preventDefault()\n }\n\n // Down, and jump into section header belows\n if (event.keyCode === DownArrow) {\n const belowLI = li.nextElementSibling\n const a = belowLI && childOfType(\"a\", belowLI)\n const button = belowLI && childOfType(\"button\", belowLI)\n\n if (a) {\n // next link\n a.focus()\n } else if (button) {\n // potential subnav above\n button.focus()\n } else {\n // at the bottom\n const sectionHostingLI = getTagFromParents(\"li\", li)\n const nextLI = sectionHostingLI.nextElementSibling\n const a = nextLI && childOfType(\"a\", nextLI)\n const button = nextLI && childOfType(\"button\", nextLI)\n\n if (a) {\n // next link\n a.focus()\n } else if (button) {\n // potential subnav above\n button.focus()\n }\n }\n\n event.preventDefault()\n }\n}\n\n/**\n * Handles moving up and down through the navigation hierarchy\n * when it's at a section category, which has different semantics\n * from the a's above\n */\nexport const onButtonKeydown: KeyboardEventHandler = event => {\n const li = getTagFromParents(\"li\", event.target as any)\n // Up, either go to the bottom of the a's in the section above\n // if it's open or jump to the previous sibling button\n if (event.keyCode == UpArrow) {\n const aboveLI = li.previousElementSibling\n if (!aboveLI) return // Hit the top\n\n const a = aboveLI && childOfType(\"a\", aboveLI)\n const button = aboveLI && childOfType(\"button\", aboveLI)\n\n if (a) {\n // next link\n a.focus()\n } else if (button) {\n // potential subnav above\n const open = aboveLI.classList.contains(\"open\")\n if (open) {\n const listOfLinks = childOfType(\"ul\", aboveLI)!\n const lastLI = listOfLinks.lastElementChild\n childOfType(\"a\", lastLI)!.focus()\n } else {\n button.focus()\n }\n } else {\n // at the top\n const sectionHostingLI = getTagFromParents(\"li\", li)\n childOfType(\"button\", sectionHostingLI)!.focus()\n }\n\n event.preventDefault()\n }\n\n // Down, and jump into section header belows\n if (event.keyCode == DownArrow) {\n const open = li.classList.contains(\"open\")\n if (open) {\n // Need to jump to first in the section\n const listOfLinks = childOfType(\"ul\", li)!\n const lastLI = listOfLinks.firstElementChild\n childOfType(\"a\", lastLI)!.focus()\n } else {\n const belowLI = li.nextElementSibling\n if (belowLI) {\n const a = belowLI && childOfType(\"a\", belowLI)\n const button = belowLI && childOfType(\"button\", belowLI)\n\n if (a) {\n // next link\n a.focus()\n } else if (button) {\n // potential subnav above\n button.focus()\n }\n }\n }\n event.preventDefault()\n }\n\n // Right, open\n if (event.key === \"ArrowRight\") {\n li.classList.remove(\"closed\")\n li.classList.add(\"open\")\n\n event.preventDefault()\n }\n\n // Right, close\n if (event.key === \"ArrowLeft\") {\n li.classList.remove(\"open\")\n li.classList.add(\"closed\")\n\n event.preventDefault()\n }\n}\n","import React, { MouseEventHandler, useEffect } from \"react\"\nimport { Link } from \"gatsby\"\n\nimport \"./Sidebar.scss\"\nimport { onAnchorKeyDown, onButtonKeydown } from \"./Sidebar-keyboard\"\nimport { SidebarNavItem } from \"../../lib/documentationNavigation\"\nimport { setInitialTargetLanguage } from \"../../lib/setInitialTargetLanguage\"\nimport { getTargetLanguage, setTargetLanguage } from \"../../lib/setTargetLanguage\"\nimport { globalHistory } from '@reach/router'\n\nexport type Props = {\n navItems: SidebarNavItem[]\n selectedID: string\n openAllSectionsExceptWhatsNew?: true\n}\nconst closedChevron = \nconst openChevron = \n\nexport const getTagFromParents = (tag: string, root: { nodeName: string, parentElement: any }) => {\n let parent = root.parentElement\n while (parent.nodeName !== tag.toUpperCase()) {\n parent = parent.parentElement\n if (parent.nodeName === \"BODY\") throw new Error(\"Could not find parent LI for toggle \")\n }\n return parent as HTMLElement\n}\n\nconst toggleNavigationSection: MouseEventHandler = (event) => {\n const li = getTagFromParents(\"li\", event.target as any)\n const isOpen = li.classList.contains(\"open\")\n if (isOpen) {\n li.classList.remove(\"open\")\n li.classList.add(\"closed\")\n\n } else {\n li.classList.remove(\"closed\")\n li.classList.add(\"open\")\n }\n}\n\nexport const SidebarToggleButton = () => {\n const toggleClick = () => {\n const navSidebar = document.getElementById(\"sidebar\")\n const isOpen = navSidebar?.classList.contains(\"show\")\n if (isOpen) {\n navSidebar?.classList.remove(\"show\")\n } else {\n navSidebar?.classList.add(\"show\")\n }\n }\n\n\n return (\n \n )\n}\n\nexport const Sidebar = (props: Props) => {\n useEffect(() => {\n // Keep all of the sidebar open at launch, then use JS to close the ones after\n // because otherwise you can't jump between sections\n document.querySelectorAll(\".closed-at-launch\").forEach(f => {\n f.classList.remove(\"closed-at-launch\")\n f.classList.remove(\"open\")\n f.classList.add(\"closed\")\n })\n }, [])\n\n const RenderItem = (props: { item: SidebarNavItem, selectedID: string, openAllSectionsExceptWhatsNew?: boolean }) => {\n const item = props.item\n if (!item.items) {\n // Is it the leaf in the nav?\n const isSelected = item.id === props.selectedID\n const aria: any = {}\n if (isSelected) {\n aria[\"aria-current\"] = \"page\"\n aria.className = \"highlight\"\n }\n\n const href = item.permalink!\n return
  • \n {item.title}\n
  • \n } else {\n // Has children\n const findSelected = (item: SidebarNavItem) => {\n if (item.id === props.selectedID) return true\n if (!item.items) return false\n for (const subItem of item.items) {\n if (findSelected(subItem)) return true\n }\n return false\n }\n\n const hostsSelected = findSelected(item)\n const classes = [] as string[]\n\n const forceOpen = props.openAllSectionsExceptWhatsNew && item.id !== \"whats-new\"\n if (hostsSelected || forceOpen) {\n classes.push(\"open\")\n classes.push(\"highlighted\")\n } else {\n classes.push(\"closed\")\n }\n\n const opened = { \"aria-expanded\": \"true\", \"aria-label\": item.title + \" close\" }\n const closed = { \"aria-label\": item.title + \" expand\" }\n const aria = hostsSelected ? opened : closed\n\n return (\n
  • \n \n
      \n {item.items.map(item => )}\n
    \n
  • \n )\n }\n }\n\n const TargetLanguageLink = (props: {target: string, children: string}) => {\n return \n setTargetLanguage(props.target)}>\n {props.children}\n \n \n }\n\n const CurrentTarget = (props: {target: string, children: string}) => {\n const id = `lf-current-target-${props.target}`\n const ret = \n {props.children}\n \n return ret;\n }\n\n function toggleOpen() {\n const selector = document.getElementById(\"targetChooser\");\n if (selector === null) return;\n selector.className = selector.className === \"open\" ? \"closed\" : \"open\";\n }\n\n /* Target language chooser */\n const RenderTargetChooser = () => {\n return (\n
  • \n \n
      \n C\n C++\n Python\n TypeScript\n Rust\n
    \n
  • \n )\n }\n\n return (\n \n )\n}\n","import * as React from \"react\"\nimport { Link } from \"gatsby\"\n\ninterface NextPrevProps {\n prev: { childMarkdownRemark: { frontmatter: { title: string, oneline: string, permalink: string } } } | undefined\n next: { childMarkdownRemark: { frontmatter: { title: string, oneline: string, permalink: string } } } | undefined,\n i: (string) => string,\n IntlLink: typeof Link\n}\n\nconst Row = (props: { children: any, className?: string }) =>
    {props.children}
    \n\n\nexport const NextPrev = (props: NextPrevProps) => {\n if (!props.prev && !props.next) return null\n const prev = props.prev && props.prev.childMarkdownRemark?.frontmatter\n const next = props.next && props.next.childMarkdownRemark?.frontmatter\n\n return (\n
    \n \n {!prev ? : }\n
    \n {!next ? : }\n \n
    \n )\n}\n\nconst EmptyLink = () =>
    \n\ninterface Section {\n data: { title: string, oneline: string, permalink: string }\n i: (string) => string,\n InltLink: typeof Link\n type: string\n}\n\nconst LinkSection = (props: Section) =>\n \n \n
    \n \n \n \n
    \n\n
    \n
    {props.i(\"handb_\" + props.type)}
    \n

    {props.data.title}

    \n

    {props.data.oneline}

    \n
    \n
    \n \n","import React, { useEffect } from \"react\";\nimport attribution from \"../../../../documentation/output/attribution.json\";\n\ninterface ContributorsProps {\n i: (string) => string;\n path: string;\n lastEdited: string;\n lang: string;\n}\n\nconst Row = (props: { children: any; className?: string }) =>\n
    {props.children}
    ;\n\nconst Section = (props: { children: any; className?: string; sKey: string }) =>\n
    \n {props.children}\n
    ;\n\nexport const Contributors = (props: ContributorsProps) => {\n const attrPath = props.path.replace(\"/packages/documentation/\", \"\");\n const page = attribution[attrPath];\n\n const reposRootURL =\n \"https://github.com/lf-lang/website-lingua-franca\";\n const repoPageURL = reposRootURL + \"/tree/main\" + props.path;\n\n const d = new Date(props.lastEdited);\n const dtf = new Intl.DateTimeFormat(\n props.lang,\n { year: \"numeric\", month: \"short\", day: \"2-digit\" },\n );\n const lastEdited = dtf.format(d);\n\n useEffect(() => {\n // @ts-ignore\n const perf = window.performance || window.mozPerformance ||\n // @ts-ignore\n window.msPerformance || window.webkitPerformance || {};\n const t = perf.timing;\n if (!t) return;\n\n const pageLoadIndicator = document.querySelector(\"#page-loaded-time\");\n if (pageLoadIndicator?.innerHTML.includes(\"This page\")) return;\n\n const start = t.navigationStart;\n const end = t.domInteractive;\n const loadTime = (end - start) / 1000;\n\n // No idea how this is happening, likely from React re-rendering\n if (loadTime < 0) return;\n\n if (pageLoadIndicator) {\n pageLoadIndicator.innerHTML = \"This page loaded in \" + loadTime +\n \" seconds.

    \";\n }\n }, []);\n\n return (\n
    \n \n
    \n

    \n Lingua Franca is an open source project. Help us improve\n these pages by sending a Pull Request ❤\n

    \n
    \n \n
    \n Contributors to this page:
    \n \n
    \n \n
    \n

    \n {`Last updated: ${lastEdited}`}\n
    \n
    \n  \n

    \n
    \n
    \n
    \n );\n};\n\nconst Avatars = (\n props: {\n data: typeof attribution[\"copy/en/topics/Contributing.md\"];\n },\n) => {\n const showRest = props.data && props.data.total > props.data.top.length;\n return
    \n {props.data && props.data.top.map((t) => {\n const grav = t.gravatar.startsWith(\"http\") ? t.gravatar : `https://gravatar.com/avatar/${t.gravatar}?s=32&&d=blank`;\n const alt = `${t.name} (${t.count})`;\n const chars = t.name.split(\" \").map((dp) => dp.substr(0, 1)).join(\"\")\n .toUpperCase();\n return
    \n {chars}\n {alt}\n
    ;\n })}\n {showRest &&\n
    \n {props.data.total - props.data.top.length}+\n
    }\n
    ;\n};\n","export const overrideSubNavLinksWithSmoothScroll = () => {\n // Overrides the anchor behavior to smooth scroll instead\n // Came from https://css-tricks.com/sticky-smooth-active-nav/\n const subnavLinks = document.querySelectorAll(\n \"#handbook-content nav ul li a\"\n )\n subnavLinks.forEach(link => {\n link.addEventListener(\"click\", event => {\n event.preventDefault()\n\n let target = document.querySelector(\n decodeURIComponent(event.target![\"hash\"])\n )\n target!.scrollIntoView({ behavior: \"smooth\", block: \"start\" })\n document.location.hash = event.target![\"hash\"]\n })\n })\n}\n\n// Sets the current selection\nexport const updateSidebarOnScroll = () => {\n const subnavLinks = document.querySelectorAll(\n \"#handbook-content nav ul li a\"\n )\n\n const fromTop = window.scrollY\n let currentPossibleAnchor: HTMLAnchorElement | undefined\n const offset = 100\n\n // Scroll down to find the highest anchor on the screen\n subnavLinks.forEach(link => {\n try {\n const section = document.querySelector(\n decodeURIComponent(link.hash)\n )\n if (!section) {\n return\n }\n const isBelow = section.offsetTop - offset <= fromTop\n if (isBelow) currentPossibleAnchor = link\n } catch (error) {\n return\n }\n })\n\n // Then set the active tag\n subnavLinks.forEach(link => {\n if (link === currentPossibleAnchor) {\n link.classList.add(\"current\")\n } else {\n link.classList.remove(\"current\")\n }\n })\n}\n","import React from \"react\"\n\n// Taken from https://github.com/microsoft/fluentui-system-icons\n// They don't have a web pipeline, so I just C&P'd directly\n// It's licensed MIT\n// https://github.com/microsoft/fluentui-system-icons/blob/master/LICENSE\n\nexport const LikeFilledSVG = () => \n \n\n\nexport const LikeUnfilledSVG = () => \n \n\n\nexport const DislikeFilledSVG = () => \n \n\n\nexport const DislikeUnfilledSVG = () => \n \n\n\n","import React, { useEffect, useState } from \"react\"\n\nexport interface PopupProps {\n\tshow: boolean\n\thtml?: string\n\turl?: string\n\t// These are absolute to the page\n\tposition?: {left: number, top: number} | null\n}\n\nexport const Popup = (props: PopupProps) => (\n\t
    \n\t\t
    \n\t\t\t\n\t\t\t\t
    \t\t\n\t\t\t\t\n\t\t
    \n\t
    \n)\n\n\nexport const useQuickInfoPopup = (lang: string) => {\n\tconst [showPopup, setShowPopup] = useState({ show: false });\n\n // Add event listeners for individual links and the popup itself on pageload\n useEffect(() => {\n const aTags = document.getElementsByTagName(\"a\")\n const links: HTMLAnchorElement[] = []\n\n for (let i = 0; i < aTags.length; i++) {\n const href = aTags[i].getAttribute(\"href\") || \"\";\n if (/\\/tsconfig\\/?#\\w+$/.test(href)) {\n aTags[i].addEventListener(\"mouseenter\", handleLinkMouseEnter)\n aTags[i].addEventListener(\"mouseleave\", handleLinkMouseLeave)\n links.push(aTags[i])\n }\n }\n\n const popupEl = document.getElementById(\"quickTipPopup\")\n popupEl?.addEventListener(\"mouseenter\", handlePopupMouseEnter)\n popupEl?.addEventListener(\"mouseleave\", handlePopupMouseLeave)\n\n // don't forget to clear them on leave\n return () => {\n for (const el of links) {\n el.removeEventListener(\"mouseenter\", handleLinkMouseEnter)\n el.removeEventListener(\"mouseleave\", handleLinkMouseLeave)\n }\n\n popupEl?.removeEventListener(\"mouseenter\", handlePopupMouseEnter)\n popupEl?.removeEventListener(\"mouseleave\", handlePopupMouseLeave)\n }\n }, [])\n\n // Keep track of how long user is hovering\n // or how long they have left the link.\n var enterTimeoutId, leaveTimeoutId\n function handleLinkMouseEnter(e) {\n clearTimeout(leaveTimeoutId); \n const target = e.target as HTMLElement\n const url = target.getAttribute(\"href\") || \"\";\n\tconst rect = target.getBoundingClientRect()\n\n\tenterTimeoutId = setTimeout((args) => {\n setShowPopup(prevProps => {\n \treturn { ...prevProps, show: true, url: args[0], position: args[1] } })\n }, 500, [url, { left: rect.x, top: rect.bottom + window.scrollY }])\n }\n\n function handleLinkMouseLeave(e) {\n clearTimeout(enterTimeoutId)\n leaveTimeoutId = setTimeout(() => {\n setShowPopup({\n show: false,\n html: \"\",\n url: \"\",\n position: null,\n })\n }, 300);\n }\n\n // Fetch content from the JSON based on url and set inner HTML\n useEffect(() => {\n async function fetchHTML() {\n\t if (!showPopup.url) return\n\n const response = await fetch(`/js/json/${lang}-tsconfig-popup.json`);\n const json = await response.json();\n const url = showPopup.url\n const configType = url.substr(url.indexOf(\"#\") + 1)\n\t const html = `
    TSConfig Reference: ${configType}
    ${json[configType]}`\n\n setShowPopup(prevProps => ({ ...prevProps, html }))\n }\n if (showPopup.show) fetchHTML();\n\n }, [showPopup.show, showPopup.url, showPopup.html])\n\n // In order to keep the popups when user leaves link\n // but still hovers over the popup itself\n function handlePopupMouseEnter(e) {\n clearTimeout(leaveTimeoutId)\n }\n\n function handlePopupMouseLeave(e) {\n clearTimeout(enterTimeoutId)\n leaveTimeoutId = setTimeout(() => {\n setShowPopup({\n show: false,\n html: \"\",\n url: \"\",\n position: null,\n })\n }, 300);\n }\n\n return showPopup\n}","import React, { useEffect, useState } from \"react\"\nimport { graphql } from \"gatsby\"\nimport { Layout } from \"../components/layout\"\nimport { Sidebar, SidebarToggleButton } from \"../components/layout/Sidebar\"\nimport { getDocumentationNavForLanguage } from \"../lib/documentationNavigation\"\nimport { Intl } from \"../components/Intl\"\nimport * as lf from \"../../../documentation/scripts/linguaFrancaUtils\";\n\n// This dependency is used in gatsby-remark-autolink-headers to generate the slugs\nimport slugger from \"github-slugger\"\n\nimport \"./documentation.scss\"\nimport \"./markdown.scss\"\n\nimport { NextPrev } from \"../components/handbook/NextPrev\"\nimport { createInternational } from \"../lib/createInternational\"\nimport { useIntl } from \"react-intl\"\nimport { createIntlLink } from \"../components/IntlLink\"\nimport { handbookCopy } from \"../copy/en/handbook\"\nimport { Contributors } from \"../components/handbook/Contributors\"\nimport { overrideSubNavLinksWithSmoothScroll, updateSidebarOnScroll } from \"./scripts/setupSubNavigationSidebar\"\nimport { setupLikeDislikeButtons } from \"./scripts/setupLikeDislikeButtons\"\nimport { DislikeUnfilledSVG, LikeUnfilledSVG } from \"../components/svgs/documentation\"\nimport { Popup, useQuickInfoPopup } from \"../components/Popup\"\nimport Helmet from \"react-helmet\"\nimport { setInitialTargetLanguage } from \"../lib/setInitialTargetLanguage\"\n\ntype Props = {\n pageContext: {\n // This is only set up if it's in the handbook nav\n id: string | undefined\n nextID: string\n previousID: string\n repoPath: string\n slug: string\n lang: string\n modifiedTime: string\n }\n data: GatsbyTypes.GetDocumentBySlugQuery\n path: string\n}\n\nconst HandbookTemplate: React.FC = (props) => {\n const post = props.data.markdownRemark\n if (!post) {\n console.log(\"Could not render:\", JSON.stringify(props))\n return
    \n }\n\n // Note: This can, and does, change triggering re-renders\n const showPopup = useQuickInfoPopup(props.pageContext.lang)\n\n const [deprecationURL, setDeprecationURL] = useState(post.frontmatter!.deprecated_by)\n\n const i = createInternational(useIntl())\n const IntlLink = createIntlLink(props.pageContext.lang)\n\n useEffect(() => {\n if (document.location.hash) {\n const redirects = post.frontmatter?.deprecation_redirects || []\n const indexOfHash = redirects.indexOf(document.location.hash.slice(1))\n if (indexOfHash !== -1) {\n setDeprecationURL(redirects[indexOfHash + 1])\n }\n }\n\n overrideSubNavLinksWithSmoothScroll()\n\n // Handles setting the scroll \n window.addEventListener(\"scroll\", updateSidebarOnScroll, { passive: true, capture: true });\n // Sets current selection\n updateSidebarOnScroll()\n\n setupLikeDislikeButtons(props.pageContext.slug, i)\n\n return () => {\n window.removeEventListener(\"scroll\", updateSidebarOnScroll)\n }\n }, [])\n\n\n if (!post.frontmatter) throw new Error(`No front-matter found for the file with props: ${props}`)\n if (!post.html) throw new Error(`No html found for the file with props: ${props}`)\n\n const selectedID = props.pageContext.id || \"NO-ID\"\n const sidebarHeaders = post.headings?.filter(h => (h?.depth || 0) <= 2) || []\n const showSidebar = !post.frontmatter.disable_toc\n const showExperimental = post.frontmatter.experimental\n const showSidebarHeadings = post.headings && sidebarHeaders.length <= 30 && sidebarHeaders.length > 0\n const navigation = getDocumentationNavForLanguage(props.pageContext.lang)\n const isHandbook = post.frontmatter.handbook\n const prefix = isHandbook ? \"Handbook\" : \"Documentation\"\n\n const slug = slugger()\n return (\n \n
    \n \n\n
    \n

    Was this page helpful?

    \n
    \n \n \n
    \n
    \n\n

    Contributing

    The preferred way to contribute to Lingua Franca is to issue pull requests through GitHub. -See the Contributing document for more details.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (3)
    CMChristian Menard  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +See the Contributing document for more details.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (3)
    CMChristian Menard  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/deadlines/index.html b/docs/handbook/deadlines/index.html index b8160ddde..d674138c8 100644 --- a/docs/handbook/deadlines/index.html +++ b/docs/handbook/deadlines/index.html @@ -279,4 +279,4 @@

    This is a (rather trivial) example of an anytime computation. Such computations proceed to improve results until time runs out and then produce the most improved result.

    The arguments to the lf_check_deadline are the self struct and a boolean that indicates whether the deadline violation handler should be invoked upon detecting a deadline violation. Because the argument is true above, the handler is invoked and Stopped counting is printed.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    SBSoroush Bateni  (5)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    SBSoroush Bateni  (5)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/developer-setup/index.html b/docs/handbook/developer-setup/index.html index 9fe111026..a4da4d7bc 100644 --- a/docs/handbook/developer-setup/index.html +++ b/docs/handbook/developer-setup/index.html @@ -115,4 +115,4 @@

    Currently, we provide two IDEs that support Lingua Franca programs. Their source code is located in external repositories. We have a Lingua Franca extension for VS code and an Eclipse based IDE called Epoch. -Please refer to the READMEs for build instructions.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (10)
    PDPeter Donovan  (4)
    EALEdward A. Lee  (3)
    MLMarten Lohstroh  (3)
    Eeal  (2)
    HKHokeun Kim  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +Please refer to the READMEs for build instructions.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (10)
    PDPeter Donovan  (4)
    EALEdward A. Lee  (3)
    MLMarten Lohstroh  (3)
    Eeal  (2)
    HKHokeun Kim  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/distributed-execution/index.html b/docs/handbook/distributed-execution/index.html index 3a02c01fe..8b70c62ff 100644 --- a/docs/handbook/distributed-execution/index.html +++ b/docs/handbook/distributed-execution/index.html @@ -543,4 +543,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    HKHokeun Kim  (13)
    Eeal  (12)
    PDPeter Donovan  (3)
    EALEdward A. Lee  (2)
    SBSoroush Bateni  (2)
    JJakio815  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    HKHokeun Kim  (13)
    Eeal  (12)
    PDPeter Donovan  (3)
    EALEdward A. Lee  (2)
    SBSoroush Bateni  (2)
    JJakio815  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/eclipse-oomph/index.html b/docs/handbook/eclipse-oomph/index.html index 557bb04e6..17b12689c 100644 --- a/docs/handbook/eclipse-oomph/index.html +++ b/docs/handbook/eclipse-oomph/index.html @@ -308,4 +308,4 @@

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (4)
    YYiweiIvy  (3)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (2)
    AHAnees Hlaleh  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (4)
    YYiweiIvy  (3)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (2)
    AHAnees Hlaleh  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/epoch-ide/index.html b/docs/handbook/epoch-ide/index.html index 21b02ed88..0673c6dde 100644 --- a/docs/handbook/epoch-ide/index.html +++ b/docs/handbook/epoch-ide/index.html @@ -79,4 +79,4 @@ }

    Epoch IDE

    Epoch is a standalone application based on Eclipse that provides a syntax-directed editor, compiler, and diagram synthesis tool for Lingua Franca programs.

    Usage

    -

    By default, Epoch is set to “Build Automatically” in the Project menu. This means that the LF code generator and compiler will be invoked every time you change a file and whenever you open a new project (on all files in the project). Many people prefer to turn this feature off and invoke the code generator by hand by clicking on the gear icon at the upper left.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (6)
    Eeal  (5)
    PDPeter Donovan  (2)
    CMChristian Menard  (1)
    EALEdward A. Lee  (1)
    SLShaokai Lin  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    By default, Epoch is set to “Build Automatically” in the Project menu. This means that the LF code generator and compiler will be invoked every time you change a file and whenever you open a new project (on all files in the project). Many people prefer to turn this feature off and invoke the code generator by hand by clicking on the gear icon at the upper left.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (6)
    Eeal  (5)
    PDPeter Donovan  (2)
    CMChristian Menard  (1)
    EALEdward A. Lee  (1)
    SLShaokai Lin  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/expressions/index.html b/docs/handbook/expressions/index.html index 18dcdfef9..5b8689a5e 100644 --- a/docs/handbook/expressions/index.html +++ b/docs/handbook/expressions/index.html @@ -183,4 +183,4 @@

    FIXME: Rust

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    EALEdward A. Lee  (4)
    PDPeter Donovan  (1)
    MLMarten Lohstroh  (1)
    HKHokeun Kim  (1)
    SBSoroush Bateni  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    EALEdward A. Lee  (4)
    PDPeter Donovan  (1)
    MLMarten Lohstroh  (1)
    HKHokeun Kim  (1)
    SBSoroush Bateni  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/extending-reactors/index.html b/docs/handbook/extending-reactors/index.html index 6d3721a87..c85cfe912 100644 --- a/docs/handbook/extending-reactors/index.html +++ b/docs/handbook/extending-reactors/index.html @@ -155,4 +155,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    SBSoroush Bateni  (3)
    EALEdward A. Lee  (1)
    PDPeter Donovan  (1)
    Rrevol-xut  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (6)
    SBSoroush Bateni  (3)
    EALEdward A. Lee  (1)
    PDPeter Donovan  (1)
    Rrevol-xut  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/features/index.html b/docs/handbook/features/index.html index 7790e4810..f76fd3a58 100644 --- a/docs/handbook/features/index.html +++ b/docs/handbook/features/index.html @@ -112,4 +112,4 @@ N -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    Hhousengw  (2)
    ARAnirudh Rengarajan  (1)
    SBSoroush Bateni  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    Hhousengw  (2)
    ARAnirudh Rengarajan  (1)
    SBSoroush Bateni  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/generic-types-interfaces-inheritance/index.html b/docs/handbook/generic-types-interfaces-inheritance/index.html index b648f4467..5d2852ca9 100644 --- a/docs/handbook/generic-types-interfaces-inheritance/index.html +++ b/docs/handbook/generic-types-interfaces-inheritance/index.html @@ -98,4 +98,4 @@

    Reactions are inherited in the order of declaration; and
  • Equally-named ports and actions between subclass and superclass must also be equally typed.
  • -

    Example

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    PDPeter Donovan  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Example

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    PDPeter Donovan  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/generics/index.html b/docs/handbook/generics/index.html index 42b1fb78b..873c017f1 100644 --- a/docs/handbook/generics/index.html +++ b/docs/handbook/generics/index.html @@ -173,4 +173,4 @@

    This reactor could be instantiated for example like this:

    g = new Generic<float, int, bool>()
     
    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (2)
    EALEdward A. Lee  (1)
    CMChristian Menard  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (2)
    EALEdward A. Lee  (1)
    CMChristian Menard  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/import-system/index.html b/docs/handbook/import-system/index.html index 1913f1af7..f918409cc 100644 --- a/docs/handbook/import-system/index.html +++ b/docs/handbook/import-system/index.html @@ -146,4 +146,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (10)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +}

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (10)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/index.html b/docs/handbook/index.html index 2c9a40736..0d1d63c54 100644 --- a/docs/handbook/index.html +++ b/docs/handbook/index.html @@ -73,4 +73,4 @@ const customFontOverride = hasLocalStorage && localStorage.getItem("force-font") || "cascadia" document.documentElement.classList.add('font-' + customFontOverride) })() -
    Skip to main content

    Lingua Franca Documentation

    \ No newline at end of file +
    Skip to main content

    Lingua Franca Documentation

    \ No newline at end of file diff --git a/docs/handbook/inputs-and-outputs/index.html b/docs/handbook/inputs-and-outputs/index.html index 7779f781a..d5b1e956f 100644 --- a/docs/handbook/inputs-and-outputs/index.html +++ b/docs/handbook/inputs-and-outputs/index.html @@ -249,4 +249,4 @@

    mutable input <name> -

    This is a directive to the code generator indicating that reactions that read this input may also modify the value of the input. The code generator will attempt to optimize the scheduling to avoid copying the input value, but this may not be possible, in which case it will automatically insert a copy operation, making it safe to modify the input. The target-specific reference documentation has more details about how this works.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (15)
    SBSoroush Bateni  (6)
    CMChristian Menard  (2)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    Rrevol-xut  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    This is a directive to the code generator indicating that reactions that read this input may also modify the value of the input. The code generator will attempt to optimize the scheduling to avoid copying the input value, but this may not be possible, in which case it will automatically insert a copy operation, making it safe to modify the input. The target-specific reference documentation has more details about how this works.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (15)
    SBSoroush Bateni  (6)
    CMChristian Menard  (2)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    Rrevol-xut  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/intellij/index.html b/docs/handbook/intellij/index.html index 3f60e0a63..da76b23d3 100644 --- a/docs/handbook/intellij/index.html +++ b/docs/handbook/intellij/index.html @@ -265,4 +265,4 @@

    Integration Tests

    -

    You can also run the integration test from IntelliJ. You will find the targetTest and singleTest tasks in the Gradle tab under org.lflang > Tasks > other. Make sure to add a run configuration as shown above and append -Ptarget=...' to the targetTest command or -DsingleTest=... to your singleTest command to specify the target (e.g., C) or the specific test that you would like to run.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    PDPeter Donovan  (5)
    MLMarten Lohstroh  (4)
    Eeal  (3)
    GGuangyu-Joshua-Feng  (2)
    CMChristian Menard  (2)
    EALEdward A. Lee  (2)
    ARAnirudh Rengarajan  (2)
    1+

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    You can also run the integration test from IntelliJ. You will find the targetTest and singleTest tasks in the Gradle tab under org.lflang > Tasks > other. Make sure to add a run configuration as shown above and append -Ptarget=...' to the targetTest command or -DsingleTest=... to your singleTest command to specify the target (e.g., C) or the specific test that you would like to run.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    PDPeter Donovan  (5)
    MLMarten Lohstroh  (4)
    Eeal  (3)
    GGuangyu-Joshua-Feng  (2)
    CMChristian Menard  (2)
    EALEdward A. Lee  (2)
    ARAnirudh Rengarajan  (2)
    1+

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/language-specification/index.html b/docs/handbook/language-specification/index.html index 1b6ed7699..55d6158b3 100644 --- a/docs/handbook/language-specification/index.html +++ b/docs/handbook/language-specification/index.html @@ -611,4 +611,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (3)
    PDPeter Donovan  (2)
    Eeal  (2)
    ARAnirudh Rengarajan  (2)
    HKHokeun Kim  (1)
    MLMarten Lohstroh  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (3)
    PDPeter Donovan  (2)
    Eeal  (2)
    ARAnirudh Rengarajan  (2)
    HKHokeun Kim  (1)
    MLMarten Lohstroh  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/logical-execution-time/index.html b/docs/handbook/logical-execution-time/index.html index 4c612267a..fb64bb35c 100644 --- a/docs/handbook/logical-execution-time/index.html +++ b/docs/handbook/logical-execution-time/index.html @@ -77,4 +77,4 @@ nav#sidebar > ul > li.closed ul { display: block !important; } -

    Logical Execution Time

    FIXME

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Logical Execution Time

    FIXME

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/methods/index.html b/docs/handbook/methods/index.html index b4ae6a66f..f0974d228 100644 --- a/docs/handbook/methods/index.html +++ b/docs/handbook/methods/index.html @@ -169,4 +169,4 @@ and receives one integer argument, which it uses to increment foo. Both methods are visible in all reactions of the reactor. In this example, the reaction to startup calls both methods in order to read and modify its state.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (2)
    Eeal  (2)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (2)
    Eeal  (2)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/modal-models/index.html b/docs/handbook/modal-models/index.html index 492117988..acca5dca9 100644 --- a/docs/handbook/modal-models/index.html +++ b/docs/handbook/modal-models/index.html @@ -209,4 +209,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    ASAlexander Schulz-Rosengarten  (5)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (2)
    ASAlexander SR  (2)
    MLMarten Lohstroh  (1)
    BBen  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    ASAlexander Schulz-Rosengarten  (5)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (2)
    ASAlexander SR  (2)
    MLMarten Lohstroh  (1)
    BBen  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/multiports-and-banks/index.html b/docs/handbook/multiports-and-banks/index.html index dbd19ba59..91034e5b3 100644 --- a/docs/handbook/multiports-and-banks/index.html +++ b/docs/handbook/multiports-and-banks/index.html @@ -994,4 +994,4 @@

    The interleaved keyword is not supported by CCppPythonTypeScriptRust.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (10)
    SBSoroush Bateni  (6)
    PDPeter Donovan  (3)
    EALEdward A. Lee  (1)
    JJakio815  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (10)
    SBSoroush Bateni  (6)
    PDPeter Donovan  (3)
    EALEdward A. Lee  (1)
    JJakio815  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/overview/index.html b/docs/handbook/overview/index.html index 52dd9ba4b..1f510280a 100644 --- a/docs/handbook/overview/index.html +++ b/docs/handbook/overview/index.html @@ -132,4 +132,4 @@

    We could also use a big hammer: model the LF program as timed automata and do model checking (e.g., UupAal)

    To Do List

    -

    Lingua Franca is a work in progress. See our project page for an overview of ongoing and future work.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (33)
    OTOrta Therox  (13)
    MLMarten Lohstroh  (6)
    ARAnirudh Rengarajan  (3)
    PDPeter Donovan  (2)
    MSMartin Schoeberl  (2)
    Eeal  (1)
    13+

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is a work in progress. See our project page for an overview of ongoing and future work.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (33)
    OTOrta Therox  (13)
    MLMarten Lohstroh  (6)
    ARAnirudh Rengarajan  (3)
    PDPeter Donovan  (2)
    MSMartin Schoeberl  (2)
    Eeal  (1)
    13+

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/parameters-and-state-variables/index.html b/docs/handbook/parameters-and-state-variables/index.html index 4f8d72190..bc8af6e90 100644 --- a/docs/handbook/parameters-and-state-variables/index.html +++ b/docs/handbook/parameters-and-state-variables/index.html @@ -245,4 +245,4 @@

    reset state <name> = <value>

    When this is done, if the state variable or the reactor is within a mode of a modal reactor, then when the mode is entered via a reset transition, the state variable will be reset to its initial value. For details, see the Modal Reactors section.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (25)
    SBSoroush Bateni  (3)
    ARAnirudh Rengarajan  (3)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    MLMarten Lohstroh  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (25)
    SBSoroush Bateni  (3)
    ARAnirudh Rengarajan  (3)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    MLMarten Lohstroh  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/preambles/index.html b/docs/handbook/preambles/index.html index 8861b4f31..778c811a7 100644 --- a/docs/handbook/preambles/index.html +++ b/docs/handbook/preambles/index.html @@ -411,4 +411,4 @@

    FIXME: Add preamble example.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (8)
    SBSoroush Bateni  (5)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (3)
    CMChristian Menard  (1)
    Rrevol-xut  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (8)
    SBSoroush Bateni  (5)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (3)
    CMChristian Menard  (1)
    Rrevol-xut  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/proof-import/index.html b/docs/handbook/proof-import/index.html index 81775700b..cc663efa4 100644 --- a/docs/handbook/proof-import/index.html +++ b/docs/handbook/proof-import/index.html @@ -469,4 +469,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (16)
    CMChristian Menard  (15)
    PDPeter Donovan  (1)
    Eeal  (1)
    ARAnirudh Rengarajan  (1)
    CFClément Fournier  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (16)
    CMChristian Menard  (15)
    PDPeter Donovan  (1)
    Eeal  (1)
    ARAnirudh Rengarajan  (1)
    CFClément Fournier  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/reaction-declarations/index.html b/docs/handbook/reaction-declarations/index.html index d51c04bc7..ce855d09d 100644 --- a/docs/handbook/reaction-declarations/index.html +++ b/docs/handbook/reaction-declarations/index.html @@ -189,4 +189,4 @@

    The CCppPythonTypeScriptRust target does not currently support reaction declarations.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (3)
    MLMarten Lohstroh  (3)
    PDPeter Donovan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (3)
    MLMarten Lohstroh  (3)
    PDPeter Donovan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/reactions/index.html b/docs/handbook/reactions/index.html index e5745397c..dd03eb656 100644 --- a/docs/handbook/reactions/index.html +++ b/docs/handbook/reactions/index.html @@ -549,4 +549,4 @@

    The CCppPythonTypeScriptRust target does not currently support reaction declarations.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (6)
    SBSoroush Bateni  (6)
    Eeal  (6)
    EALEdward A. Lee  (4)
    PDPeter Donovan  (2)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (6)
    SBSoroush Bateni  (6)
    Eeal  (6)
    EALEdward A. Lee  (4)
    PDPeter Donovan  (2)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/reactors-on-patmos/index.html b/docs/handbook/reactors-on-patmos/index.html index f44c38da3..f8d638497 100644 --- a/docs/handbook/reactors-on-patmos/index.html +++ b/docs/handbook/reactors-on-patmos/index.html @@ -152,4 +152,4 @@

    Patmos Reference Handbook.

    Note, that the WCET analysis of a reaction does only include the code of the reaction function, not the cache miss cost of calling the function from -the scheduler or the cache miss cost when returning to the scheduler.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (1)
    Eeal  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +the scheduler or the cache miss cost when returning to the scheduler.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (1)
    Eeal  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/regression-tests/index.html b/docs/handbook/regression-tests/index.html index 8ba582876..ea7fb4d6b 100644 --- a/docs/handbook/regression-tests/index.html +++ b/docs/handbook/regression-tests/index.html @@ -146,4 +146,4 @@

    core subproject, the html report will be located in build/reports/html/index.html. Note that this report will only reflect the coverage of the test that have actually executed.

    Continuous Integration

    -

    Each push or pull request will trigger all tests to be run on GitHub Actions. It’s configuration can be found here.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (2)
    PDPeter Donovan  (2)
    Eeal  (2)
    D(KDongha (Jake) Kim  (1)
    MLMarten Lohstroh  (1)
    EALEdward A. Lee  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Each push or pull request will trigger all tests to be run on GitHub Actions. It’s configuration can be found here.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    CMChristian Menard  (2)
    PDPeter Donovan  (2)
    Eeal  (2)
    D(KDongha (Jake) Kim  (1)
    MLMarten Lohstroh  (1)
    EALEdward A. Lee  (1)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/related-work/index.html b/docs/handbook/related-work/index.html index ed215bfd5..e4e065791 100644 --- a/docs/handbook/related-work/index.html +++ b/docs/handbook/related-work/index.html @@ -142,4 +142,4 @@

    RADLER framework from SRI, which is based on a publish-and-subscribe architecture similar to ROS. It introduces some timing constructs such as periodic execution and scheduling constraints, but it makes no effort to be deterministic.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (19)
    MLMarten Lohstroh  (3)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (19)
    MLMarten Lohstroh  (3)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/running-benchmarks/index.html b/docs/handbook/running-benchmarks/index.html index 8277aa32e..2874cd586 100644 --- a/docs/handbook/running-benchmarks/index.html +++ b/docs/handbook/running-benchmarks/index.html @@ -237,4 +237,4 @@

    pings: ["-D", "count=<value>"]

    This is very similar to the C++ configuration. However, the C target of LF currently does not support overriding of parameter values at runtime. Therefore, all parameters need to be provided as arguments to the code generator and the benchmark needs to provide corresponding cog directives.

    -

    New benchmarks can be simply added by replicating this example and adjusting the precise configuration values and parameters to the specific benchmark.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (3)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    New benchmarks can be simply added by replicating this example and adjusting the precise configuration values and parameters to the specific benchmark.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (3)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/security/index.html b/docs/handbook/security/index.html index 166799884..bf9da1a15 100644 --- a/docs/handbook/security/index.html +++ b/docs/handbook/security/index.html @@ -96,4 +96,4 @@

    sudo make install

    If you would like to go back to non-AUTH mode, you would have to remove all contents of the build folder.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    JJakio815  (7)
    D(KDongha (Jake) Kim  (2)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    JJakio815  (7)
    D(KDongha (Jake) Kim  (2)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/superdense-time/index.html b/docs/handbook/superdense-time/index.html index a094e233b..c2e22c1d2 100644 --- a/docs/handbook/superdense-time/index.html +++ b/docs/handbook/superdense-time/index.html @@ -305,4 +305,4 @@

    Alignment of Logical and Physical Times

    -

    Recall that in Lingua Franca, logical time “chases” physical time, invoking reactions at a physical time close to their logical time. For that purpose, the microstep is ignored.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (9)
    Eeal  (4)
    EALEdward A. Lee  (1)
    PDPeter Donovan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Recall that in Lingua Franca, logical time “chases” physical time, invoking reactions at a physical time close to their logical time. For that purpose, the microstep is ignored.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (9)
    Eeal  (4)
    EALEdward A. Lee  (1)
    PDPeter Donovan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/target-declaration/index.html b/docs/handbook/target-declaration/index.html index e3224c539..efc5bc66f 100644 --- a/docs/handbook/target-declaration/index.html +++ b/docs/handbook/target-declaration/index.html @@ -641,4 +641,4 @@

    =} } -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (7)
    EALEdward A. Lee  (6)
    JJakio815  (4)
    CMChristian Menard  (4)
    HKHokeun Kim  (3)
    PDPeter Donovan  (2)
    JHJohannes Hayeß  (2)
    3+

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (7)
    EALEdward A. Lee  (6)
    JJakio815  (4)
    CMChristian Menard  (4)
    HKHokeun Kim  (3)
    PDPeter Donovan  (2)
    JHJohannes Hayeß  (2)
    3+

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/target-language-details/index.html b/docs/handbook/target-language-details/index.html index b175763f1..0e2b3aed6 100644 --- a/docs/handbook/target-language-details/index.html +++ b/docs/handbook/target-language-details/index.html @@ -2699,4 +2699,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (21)
    Eeal  (16)
    PDPeter Donovan  (12)
    SBSoroush Bateni  (10)
    SSteven  (4)
    ERJErling Rennemo Jellum  (1)
    MLMarten Lohstroh  (1)
    1+

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +````

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (21)
    Eeal  (16)
    PDPeter Donovan  (12)
    SBSoroush Bateni  (10)
    SSteven  (4)
    ERJErling Rennemo Jellum  (1)
    MLMarten Lohstroh  (1)
    1+

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/termination/index.html b/docs/handbook/termination/index.html index 779be7cb7..eb214ae2c 100644 --- a/docs/handbook/termination/index.html +++ b/docs/handbook/termination/index.html @@ -112,4 +112,4 @@

    For federated programs, each federate and the RTI catches external signals to shut down in an orderly way.

    When a federate gets such an external signal (e.g. control-C), it sends a RESIGN message to the RTI and an EOF (end of file) on each socket connection to another federate. It then closes all sockets and shuts down. The RTI and all other federates should continue running until some other termination condition occurs.

    When the RTI gets such an external signal (e.g. control-C), it broadcasts a STOP_REQUEST message to all federates, waits for their replies (with a timeout in case the federate or the network has failed), chooses the maximum timestamp s on the replies, broadcasts a STOP_GRANTED message to all federates with payload s, and waits for LOGICAL_TIME_COMPLETE messages as above.

    -

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (5)
    PDPeter Donovan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (5)
    PDPeter Donovan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/time-and-timers/index.html b/docs/handbook/time-and-timers/index.html index 7a2eab8d5..1b4e99ef8 100644 --- a/docs/handbook/time-and-timers/index.html +++ b/docs/handbook/time-and-timers/index.html @@ -557,4 +557,4 @@

    shutdown reaction.

    -

    The shutdown trigger typically occurs at microstep 0, but may occur at a larger microstep. See Superdense Time and Termination.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (15)
    SBSoroush Bateni  (8)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (2)
    HKHokeun Kim  (1)
    Rrevol-xut  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    The shutdown trigger typically occurs at microstep 0, but may occur at a larger microstep. See Superdense Time and Termination.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (15)
    SBSoroush Bateni  (8)
    EALEdward A. Lee  (3)
    PDPeter Donovan  (2)
    HKHokeun Kim  (1)
    Rrevol-xut  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/timing-analysis/index.html b/docs/handbook/timing-analysis/index.html index 8e4987abe..18affe9e1 100644 --- a/docs/handbook/timing-analysis/index.html +++ b/docs/handbook/timing-analysis/index.html @@ -132,4 +132,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MSMartin Schoeberl  (7)
    EALEdward A. Lee  (7)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Preemption can be avoided when there are enough cores (or hardware threads in PRET) available to execute r1 and r3 concurrently.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MSMartin Schoeberl  (7)
    EALEdward A. Lee  (7)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/tools/index.html b/docs/handbook/tools/index.html index ff14a04be..9f6c2b8b2 100644 --- a/docs/handbook/tools/index.html +++ b/docs/handbook/tools/index.html @@ -87,4 +87,4 @@ +--------+ err | +----------+ gen err +----------+ | | Language Server | +-------------------------------------+ -

    If the LF compiler encounters any syntax errors, it will report them to the editor (the language client). If the LF code compiles, the output will be sent to the target compiler. If the target compiler reports any errors, these, too, will be reported to the editor via the language server. The tricky part is to match target language errors to LF source locations; the language server will have to do some bookkeeping.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    If the LF compiler encounters any syntax errors, it will report them to the editor (the language client). If the LF code compiles, the output will be sent to the target compiler. If the target compiler reports any errors, these, too, will be reported to the editor via the language server. The tricky part is to match target language errors to LF source locations; the language server will have to do some bookkeeping.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    MLMarten Lohstroh  (7)
    ARAnirudh Rengarajan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/tracing/index.html b/docs/handbook/tracing/index.html index 706133ed3..1dddf7d0f 100644 --- a/docs/handbook/tracing/index.html +++ b/docs/handbook/tracing/index.html @@ -347,4 +347,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (4)
    Eeal  (3)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (2)
    CJChadlia Jerad  (1)
    ERJErling Rennemo Jellum  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    SBSoroush Bateni  (4)
    Eeal  (3)
    EALEdward A. Lee  (2)
    PDPeter Donovan  (2)
    ARAnirudh Rengarajan  (2)
    CJChadlia Jerad  (1)
    ERJErling Rennemo Jellum  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/troubleshooting/index.html b/docs/handbook/troubleshooting/index.html index aa2551fe3..eaf1d9663 100644 --- a/docs/handbook/troubleshooting/index.html +++ b/docs/handbook/troubleshooting/index.html @@ -91,4 +91,4 @@

    Epoch and the Visual Studio Code extension use the same environment as the command-line tools is to invoke them on the command line. For example, on a Mac, you can invoke Epoch and Visual Studio Code as follows:

    $ open -a epoch
     $ code .
    -

    This way, the tools inherit the environment from the shell from which you invoke them. Often, that environment is quite different from what you get if, for example, you invoke the tools by double clicking on their icons.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (4)
    CMChristian Menard  (1)
    PDPeter Donovan  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    This way, the tools inherit the environment from the shell from which you invoke them. Often, that environment is quite different from what you get if, for example, you invoke the tools by double clicking on their icons.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eeal  (4)
    CMChristian Menard  (1)
    PDPeter Donovan  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/tutorial-video/index.html b/docs/handbook/tutorial-video/index.html index 4f62d4e7d..0b12be2a9 100644 --- a/docs/handbook/tutorial-video/index.html +++ b/docs/handbook/tutorial-video/index.html @@ -342,4 +342,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    PDPeter Donovan  (1)
    Eeal  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    PDPeter Donovan  (1)
    Eeal  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/website-development/index.html b/docs/handbook/website-development/index.html index 3c3e1db9e..7d531a4dd 100644 --- a/docs/handbook/website-development/index.html +++ b/docs/handbook/website-development/index.html @@ -81,4 +81,4 @@

    First, for simple changes to the website, such as fixing typos, the easiest way is to scroll to the bottom of the page, follow the link at the bottom to send a pull request, edit the resulting page, and issue a pull request.

    For more elaborate changes, including adding new pages, you will need to clone the GitHub repository. You can then set up your local clone to provide a local copy of the website at http://localhost:8000 by following the instructions in the README file. This way, you can test your changes before issuing a pull request.

    Editing the Handbook

    -

    The handbook is the most updated part of the website and it includes quite a bit of infrastructure to support writing pages that describe features in any or all of the target languages. The root of the handbook pages in the packages/documentation part of the repo. That directory has a useful README file that describes the structure and provides instructions for inserting code examples in any target language and target-specific text within a body of target-independent text.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +

    The handbook is the most updated part of the website and it includes quite a bit of infrastructure to support writing pages that describe features in any or all of the target languages. The root of the handbook pages in the packages/documentation part of the repo. That directory has a useful README file that describes the structure and provides instructions for inserting code examples in any target language and target-specific text within a body of target-independent text.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    EALEdward A. Lee  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/handbook/zephyr/index.html b/docs/handbook/zephyr/index.html index f406de9f5..cda55ac26 100644 --- a/docs/handbook/zephyr/index.html +++ b/docs/handbook/zephyr/index.html @@ -278,4 +278,4 @@

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eerlingrj  (11)
    ERJErling Rennemo Jellum  (8)
    Eerling  (4)
    MLMarten Lohstroh  (1)

    Last updated: Nov 03, 2023

     

    \ No newline at end of file +system of old Zephyr installations.

    Lingua Franca is an open source project. Help us improve these pages by sending a Pull Request

    Contributors to this page:
    Eerlingrj  (11)
    ERJErling Rennemo Jellum  (8)
    Eerling  (4)
    MLMarten Lohstroh  (1)

    Last updated: Nov 08, 2023

     

    \ No newline at end of file diff --git a/docs/index.html b/docs/index.html index 6ee257dd6..a4ebf76ed 100644 --- a/docs/index.html +++ b/docs/index.html @@ -73,4 +73,4 @@ const customFontOverride = hasLocalStorage && localStorage.getItem("force-font") || "cascadia" document.documentElement.classList.add('font-' + customFontOverride) })() -
    Skip to main content

    Lingua Franca Documentation

    Resources

    Overview of the project.

    We also have pdf versions for the following languages:
    c

    cpp

    py

    ts

    rs

    Tools

    Tools for developing Lingua Franca programs.

    Embedded Platforms

    Documentation for developing Lingua Franca on Embedded Platforms.

    Developer

    Information for developers of the Lingua Franca language and tools.

    Learning Resources

    \ No newline at end of file +
    Skip to main content

    Lingua Franca Documentation

    Resources

    Overview of the project.

    We also have pdf versions for the following languages:
    c

    cpp

    py

    ts

    rs

    Tools

    Tools for developing Lingua Franca programs.

    Embedded Platforms

    Documentation for developing Lingua Franca on Embedded Platforms.

    Developer

    Information for developers of the Lingua Franca language and tools.

    Learning Resources

    \ No newline at end of file diff --git a/download/index.html b/download/index.html index 9843316a7..f69e04266 100644 --- a/download/index.html +++ b/download/index.html @@ -73,4 +73,4 @@ const customFontOverride = hasLocalStorage && localStorage.getItem("force-font") || "cascadia" document.documentElement.classList.add('font-' + customFontOverride) })() -
    Skip to main content

    Download and Install Lingua Franca

    The Lingua Franca toolchain requires Java 17 (download from Oracle). Each target language may have additional requirements (see the Target Language Details page and select your target language).

    You can use Lingua Franca:

    You can also spin up one of our pre-configured Cloud-based dev environments:
    Open in GitPod  Open in GitHub Codespaces
    Have a look at the Lingua Franca playground for more details.

    Visual Studio Code

    Our Visual Studio Code extension can be installed via the Marketplace or built from source, as detailed below. See the handbook for usage instructions.

    Marketplace

    The Lingua Franca extension is available on the Visual Studio Marketplace and the VSX Registry. To install the extension, open VS Code, launch Quick Open (Ctrl + P) and enter:

    ext install lf-lang.vscode-lingua-franca

    Alternatively, you can run the following command in your terminal:

    code --install-extension lf-lang.vscode-lingua-franca

    To use the nightly pre-release of the extension instead of the latest release, find the Lingua Franca extension in the Extensions tab and click on the "Switch to Pre-Release Version" button.

    From Source

    Please refer to the Lingua Franca VS Code GitHub repository for build instructions.

    Epoch IDE

    There are multiple options available for installing Epoch as listed below. See the handbook for usage instructions.

    Install Script

    Run the following command in your terminal to install the latest release (on Windows, use WSL):

    curl -Ls https://install.lf-lang.org | bash -s epoch

    You can also install the nightly pre-release:

    curl -Ls https://install.lf-lang.org | bash -s epoch nightly

    You can use the --prefix=<path> argument to change the default install location.

    The default prefix is /usr/local/bin on a Mac and ~/.local/bin on Linux and WSL. You may not have write access to this directory by default, in which case, if you still want to use the default prefix, you can replace sh with sudo sh in the above commands.

    AUR

    There are binary packages available in the Arch user repository, which you can install using your favorite AUR helper. For instance, with yay:

    yay -S epoch-bin

    or for the nightly pre-release:

    yay -S epoch-nightly-bin

    Manual Download

    Regular and nightly release builds of Epoch can be downloaded from the Epoch release page. Download the archive that matches your OS and architecture, and extract the contents.

    MacOS requires extra steps before being able to execute the app:

    xattr -cr Epoch.app

    To install, drag the Epoch.app file to your Applications folder. You can then invoke the app as follows:

    open -a Epoch.app

    From Source

    Please refer to the Epoch GitHub repository for build instructions.

    CLI Tools

    There are multiple options available for installing the Lingua Franca compiler and other command line tools, as listed below. See the handbook for usage instructions.

    Install Script

    Run the following command in your terminal to install the latest release (on Windows, use WSL):

    curl -Ls https://install.lf-lang.org | bash -s cli

    You can also install the nightly pre-release:

    curl -Ls https://install.lf-lang.org | bash -s cli nightly

    You can use the --prefix=<path> argument to change the default install location.

    AUR

    There are binary packages available in the Arch user repository, which you can install using your favorite AUR helper. For instance, with yay:

    yay -S lf-cli-bin

    or for the nightly pre-release:

    yay -S lf-cli-nightly-bin

    Manual Download

    Regular and nightly release builds of the command line tools can be downloaded from the Lingua Franca release page. Download the archive that matches your OS and architecture, and extract the contents.

    From Source

    Please refer to the Lingua Franca GitHub repository for build instructions.

    If you'd like to contribute to Lingua Franca, you can find details about the recommended developer setup here.

    \ No newline at end of file +
    Skip to main content

    Download and Install Lingua Franca

    The Lingua Franca toolchain requires Java 17 (download from Oracle). Each target language may have additional requirements (see the Target Language Details page and select your target language).

    You can use Lingua Franca:

    You can also spin up one of our pre-configured Cloud-based dev environments:
    Open in GitPod  Open in GitHub Codespaces
    Have a look at the Lingua Franca playground for more details.

    Visual Studio Code

    Our Visual Studio Code extension can be installed via the Marketplace or built from source, as detailed below. See the handbook for usage instructions.

    Marketplace

    The Lingua Franca extension is available on the Visual Studio Marketplace and the VSX Registry. To install the extension, open VS Code, launch Quick Open (Ctrl + P) and enter:

    ext install lf-lang.vscode-lingua-franca

    Alternatively, you can run the following command in your terminal:

    code --install-extension lf-lang.vscode-lingua-franca

    To use the nightly pre-release of the extension instead of the latest release, find the Lingua Franca extension in the Extensions tab and click on the "Switch to Pre-Release Version" button.

    From Source

    Please refer to the Lingua Franca VS Code GitHub repository for build instructions.

    Epoch IDE

    There are multiple options available for installing Epoch as listed below. See the handbook for usage instructions.

    Install Script

    Run the following command in your terminal to install the latest release (on Windows, use WSL):

    curl -Ls https://install.lf-lang.org | bash -s epoch

    You can also install the nightly pre-release:

    curl -Ls https://install.lf-lang.org | bash -s epoch nightly

    You can use the --prefix=<path> argument to change the default install location.

    The default prefix is /usr/local/bin on a Mac and ~/.local/bin on Linux and WSL. You may not have write access to this directory by default, in which case, if you still want to use the default prefix, you can replace sh with sudo sh in the above commands.

    AUR

    There are binary packages available in the Arch user repository, which you can install using your favorite AUR helper. For instance, with yay:

    yay -S epoch-bin

    or for the nightly pre-release:

    yay -S epoch-nightly-bin

    Manual Download

    Regular and nightly release builds of Epoch can be downloaded from the Epoch release page. Download the archive that matches your OS and architecture, and extract the contents.

    MacOS requires extra steps before being able to execute the app:

    xattr -cr Epoch.app

    To install, drag the Epoch.app file to your Applications folder. You can then invoke the app as follows:

    open -a Epoch.app

    From Source

    Please refer to the Epoch GitHub repository for build instructions.

    CLI Tools

    There are multiple options available for installing the Lingua Franca compiler and other command line tools, as listed below. See the handbook for usage instructions.

    Install Script

    Run the following command in your terminal to install the latest release (on Windows, use WSL):

    curl -Ls https://install.lf-lang.org | bash -s cli

    You can also install the nightly pre-release:

    curl -Ls https://install.lf-lang.org | bash -s cli nightly

    You can use the --prefix=<path> argument to change the default install location.

    AUR

    There are binary packages available in the Arch user repository, which you can install using your favorite AUR helper. For instance, with yay:

    yay -S lf-cli-bin

    or for the nightly pre-release:

    yay -S lf-cli-nightly-bin

    Manual Download

    Regular and nightly release builds of the command line tools can be downloaded from the Lingua Franca release page. Download the archive that matches your OS and architecture, and extract the contents.

    From Source

    Please refer to the Lingua Franca GitHub repository for build instructions.

    If you'd like to contribute to Lingua Franca, you can find details about the recommended developer setup here.

    \ No newline at end of file diff --git a/empty/index.html b/empty/index.html index 52a65c6d0..e8370fa32 100644 --- a/empty/index.html +++ b/empty/index.html @@ -73,4 +73,4 @@ const customFontOverride = hasLocalStorage && localStorage.getItem("force-font") || "cascadia" document.documentElement.classList.add('font-' + customFontOverride) })() -
    Skip to main content

    This page is intentionally left blank

    \ No newline at end of file +
    Skip to main content

    This page is intentionally left blank

    \ No newline at end of file diff --git a/index.html b/index.html index cab559941..dd70bf149 100644 --- a/index.html +++ b/index.html @@ -73,4 +73,4 @@ const customFontOverride = hasLocalStorage && localStorage.getItem("force-font") || "cascadia" document.documentElement.classList.add('font-' + customFontOverride) })() -
    Skip to main content

    Lingua Franca is a polyglot coordination language for reactive, concurrent, and time-sensitive applications.

    Lingua Franca (LF) is a polyglot coordination language built to bring deterministic reactive concurrency and time to mainstream target programming languages (currently C, C++, Python, TypeScript, and Rust). LF is supported by a runtime system that is capable of concurrent and distributed execution of reactive programs that are deployable on the Cloud, the Edge, and even on bare-iron embedded platforms.

    A Lingua Franca program specifies the interactions between components called reactors. The logic of each reactor is written in plain target code. A code generator synthesizes one or more programs in the target language, which are then compiled using standard tool chains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.

    What is Lingua Franca?

    Reactor-oriented

    Reactors are reactive and composable concurrent software components with inputs, outputs, and local state.

    Concurrent

    Reactions to events are concurrent unless there is an explicit dependency between them.

    Deterministic

    Lingua Franca programs are deterministic by default and therefore easy to test.

    \ No newline at end of file +
    Skip to main content

    Lingua Franca is a polyglot coordination language for reactive, concurrent, and time-sensitive applications.

    Lingua Franca (LF) is a polyglot coordination language built to bring deterministic reactive concurrency and time to mainstream target programming languages (currently C, C++, Python, TypeScript, and Rust). LF is supported by a runtime system that is capable of concurrent and distributed execution of reactive programs that are deployable on the Cloud, the Edge, and even on bare-iron embedded platforms.

    A Lingua Franca program specifies the interactions between components called reactors. The logic of each reactor is written in plain target code. A code generator synthesizes one or more programs in the target language, which are then compiled using standard tool chains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.

    What is Lingua Franca?

    Reactor-oriented

    Reactors are reactive and composable concurrent software components with inputs, outputs, and local state.

    Concurrent

    Reactions to events are concurrent unless there is an explicit dependency between them.

    Deterministic

    Lingua Franca programs are deterministic by default and therefore easy to test.

    \ No newline at end of file diff --git a/page-data/app-data.json b/page-data/app-data.json index 3acb77696..e63a02c7f 100644 --- a/page-data/app-data.json +++ b/page-data/app-data.json @@ -1 +1 @@ -{"webpackCompilationHash":"0b8833d0329c8b3a3d30"} +{"webpackCompilationHash":"39c14b80d15174df9b51"} diff --git a/page-data/docs/handbook/a-first-reactor/page-data.json b/page-data/docs/handbook/a-first-reactor/page-data.json index a42f7600f..e1f0fb6fa 100644 --- a/page-data/docs/handbook/a-first-reactor/page-data.json +++ b/page-data/docs/handbook/a-first-reactor/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/a-first-reactor","result":{"data":{"markdownRemark":{"id":"49ea27af-cb5f-5558-96bd-ecfcb618bbd1","excerpt":"$page-showing-target$\nSee the requirements for using this target. Minimal Example A minimal but complete Lingua Franca file with one reactor is this: $start…","html":"

    $page-showing-target$\nSee the requirements for using this target.

    \n

    Minimal Example

    \n

    A minimal but complete Lingua Franca file with one reactor is this:

    \n

    $start(HelloWorld)$

    \n
    target C\nmain reactor {\n  reaction(startup) {=\n    printf("Hello World.\\n");\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  reaction(startup) {=\n    std::cout << "Hello World." << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  reaction(startup) {=\n    print("Hello World.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  reaction(startup) {=\n    console.log("Hello World.")\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  reaction(startup) {=\n    println!("Hello World.");\n  =}\n}\n
    \n

    $end(HelloWorld)$

    \n

    Every Lingua Franca program begins with a target declaration that specifies the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca code generator.

    \n

    Every LF program also has a $main$ [ or $federated$]{federated} reactor, which is the top level of a hierarchy of contained and interconnected reactors. The above simple example has no contained reactors.

    \n

    The $main$ reactor above has a single $reaction$, which is triggered by the $startup$ trigger. This trigger causes the reaction to execute at the start of the program. The body of the reaction, delimited by {= ... =}, is ordinary $target-language$ code which, as we will see, has access to a number of functions and variables specific to Lingua Franca.

    \n

    Examples

    \n

    Examples of Lingua Franca programs can be found in the Lingua Franca Playground.

    \n

    The regression tests have a rich set of examples that illustrate every feature of the language.

    \n

    Structure of an LF Project

    \n

    The Lingua Franca tools assume that LF programs are put into a file with a .lf extension that is stored somewhere within a directory called src. To compile and run the above example, choose a project root directory, create a src directory within that, and put the above code into a file called, say, src/HelloWorld.lf. You can compile the code on the command line, within Visual Studio Code, or within the Epoch IDE. On the command line this will look like this:

    \n
        > lfc src/HelloWorld.lf\n    ... output from the code generator and compiler ...
    \n
    \n

    After this completes, two additional directories will have been created within\nthe project root, bin and src-gen. The bin directory has an\nexecutable file called HelloWorld. Executing that file will result, not\nsurprisingly, in printing “Hello World”. The generated source files will be\nin a subdirectory called HelloWorld within src-gen.

    \n
    \n
    \n

    After this completes, an additional src-gen directory will have been created within the project root. The generated code will be in subdirectory called HelloWorld within src-gen. The output from the code generator will include instructions for executing the generated code:

    \n
    #####################################\nTo run the generated program, use:\n\n    node ...path-to-project.../src-gen/Minimal/dist/Minimal.js\n\n#####################################\n
    \n
    #####################################\nTo run the generated program, use:\n\n    python3 ...path-to-project.../src-gen/Minimal/Minimal.py\n\n#####################################\n
    \n
    \n

    Reactor Block

    \n

    A $reactor$ is a software component that reacts to input events, timer events, and internal events. It has private state variables that are not visible to any other reactor. Its reactions can consist of altering its own state, sending messages to other reactors, or affecting the environment through some kind of actuation or side effect (e.g., printing a message, as in the above HelloWorld example).

    \n

    The general structure of a reactor definition is as follows:

    \n
    \n
    [main or federated] reactor <class-name> [(<parameters>)] {\n    input <name>: <type>\n    output <name>: <type>\n    state <name>: <type> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>[: <type>]\n    physical action <name>[: <type>]\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n
    \n
    [main] reactor <class-name> [(<parameters>)] {\n    input <name>: <type>\n    output <name>: <type>\n    state <name>: <type> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>[: <type>]\n    physical action <name>[: <type>]\n    [const] method <name>(<parameters>):<type> {= ... body ...=}\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n
    \n
    [main or federated] reactor <class-name> [(<parameters>)] {\n    input <name>\n    output <name>\n    state <name> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>\n    physical action <name>\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n

    Contents within square brackets are optional, contents within <...> are user-defined, and each line may appear zero or more times, as explained in the next pages. Parameters, inputs, outputs, timers, actions, and contained reactors all have names, and the names are required to be distinct from one another.

    \n

    If the $reactor$ keyword is preceded by $main$[ or $federated$]{federated}, then this reactor will be instantiated and run by the generated code.

    \n

    Any number of reactors may be defined in one file, and a $main$[ or $federated$]{federated} reactor need not be given a name, but if it is given a name, then that name must match the file name.

    \n

    Reactors may extend other reactors, inheriting their properties, and a file may import reactors from other files. If an imported LF file contains a $main$[ or $federated$]{federated} reactor, that reactor is ignored (it will not be imported). This makes it easy to create a library of reusable reactors that each come with a test case or demonstration in the form of a main reactor.

    \n

    Comments

    \n

    Lingua Franca files can have C/C++/Java-style comments and/or Python-style comments. All of the following are valid comments:

    \n
        // Single-line C-style comment.\n    /*\n     * Multi-line C-style comment.\n     */\n    # Single-line Python-style comment.
    ","headings":[{"value":"Minimal Example","depth":2},{"value":"Examples","depth":2},{"value":"Structure of an LF Project","depth":2},{"value":"Reactor Block","depth":2},{"value":"Comments","depth":2}],"frontmatter":{"permalink":"/docs/handbook/a-first-reactor","title":"A First Reactor","oneline":"Writing your first Lingua Franca reactor.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","permalink":"/docs/handbook/inputs-and-outputs"}}}},"pageContext":{"id":"1-a-first-reactor","slug":"/docs/handbook/a-first-reactor","repoPath":"/packages/documentation/copy/en/topics/A First Reactor.md","nextID":"dcdc6b32-76b0-570a-a6f8-23bb570863c7","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/a-first-reactor","result":{"data":{"markdownRemark":{"id":"49ea27af-cb5f-5558-96bd-ecfcb618bbd1","excerpt":"$page-showing-target$\nSee the requirements for using this target. Minimal Example A minimal but complete Lingua Franca file with one reactor is this: $start…","html":"

    $page-showing-target$\nSee the requirements for using this target.

    \n

    Minimal Example

    \n

    A minimal but complete Lingua Franca file with one reactor is this:

    \n

    $start(HelloWorld)$

    \n
    target C\nmain reactor {\n  reaction(startup) {=\n    printf("Hello World.\\n");\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  reaction(startup) {=\n    std::cout << "Hello World." << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  reaction(startup) {=\n    print("Hello World.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  reaction(startup) {=\n    console.log("Hello World.")\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  reaction(startup) {=\n    println!("Hello World.");\n  =}\n}\n
    \n

    $end(HelloWorld)$

    \n

    Every Lingua Franca program begins with a target declaration that specifies the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca code generator.

    \n

    Every LF program also has a $main$ [ or $federated$]{federated} reactor, which is the top level of a hierarchy of contained and interconnected reactors. The above simple example has no contained reactors.

    \n

    The $main$ reactor above has a single $reaction$, which is triggered by the $startup$ trigger. This trigger causes the reaction to execute at the start of the program. The body of the reaction, delimited by {= ... =}, is ordinary $target-language$ code which, as we will see, has access to a number of functions and variables specific to Lingua Franca.

    \n

    Examples

    \n

    Examples of Lingua Franca programs can be found in the Lingua Franca Playground.

    \n

    The regression tests have a rich set of examples that illustrate every feature of the language.

    \n

    Structure of an LF Project

    \n

    The Lingua Franca tools assume that LF programs are put into a file with a .lf extension that is stored somewhere within a directory called src. To compile and run the above example, choose a project root directory, create a src directory within that, and put the above code into a file called, say, src/HelloWorld.lf. You can compile the code on the command line, within Visual Studio Code, or within the Epoch IDE. On the command line this will look like this:

    \n
        > lfc src/HelloWorld.lf\n    ... output from the code generator and compiler ...
    \n
    \n

    After this completes, two additional directories will have been created within\nthe project root, bin and src-gen. The bin directory has an\nexecutable file called HelloWorld. Executing that file will result, not\nsurprisingly, in printing “Hello World”. The generated source files will be\nin a subdirectory called HelloWorld within src-gen.

    \n
    \n
    \n

    After this completes, an additional src-gen directory will have been created within the project root. The generated code will be in subdirectory called HelloWorld within src-gen. The output from the code generator will include instructions for executing the generated code:

    \n
    #####################################\nTo run the generated program, use:\n\n    node ...path-to-project.../src-gen/Minimal/dist/Minimal.js\n\n#####################################\n
    \n
    #####################################\nTo run the generated program, use:\n\n    python3 ...path-to-project.../src-gen/Minimal/Minimal.py\n\n#####################################\n
    \n
    \n

    Reactor Block

    \n

    A $reactor$ is a software component that reacts to input events, timer events, and internal events. It has private state variables that are not visible to any other reactor. Its reactions can consist of altering its own state, sending messages to other reactors, or affecting the environment through some kind of actuation or side effect (e.g., printing a message, as in the above HelloWorld example).

    \n

    The general structure of a reactor definition is as follows:

    \n
    \n
    [main or federated] reactor <class-name> [(<parameters>)] {\n    input <name>: <type>\n    output <name>: <type>\n    state <name>: <type> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>[: <type>]\n    physical action <name>[: <type>]\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n
    \n
    [main] reactor <class-name> [(<parameters>)] {\n    input <name>: <type>\n    output <name>: <type>\n    state <name>: <type> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>[: <type>]\n    physical action <name>[: <type>]\n    [const] method <name>(<parameters>):<type> {= ... body ...=}\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n
    \n
    [main or federated] reactor <class-name> [(<parameters>)] {\n    input <name>\n    output <name>\n    state <name> [= <value>]\n    timer <name>([<offset>[, <period>]])\n    logical action <name>\n    physical action <name>\n    reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n    <instance-name> = new <class-name>([<parameter-assignments>])\n    <port-name> [, ...] -> <port-name> [, ...] [after <delay>]\n}\n
    \n
    \n

    Contents within square brackets are optional, contents within <...> are user-defined, and each line may appear zero or more times, as explained in the next pages. Parameters, inputs, outputs, timers, actions, and contained reactors all have names, and the names are required to be distinct from one another.

    \n

    If the $reactor$ keyword is preceded by $main$[ or $federated$]{federated}, then this reactor will be instantiated and run by the generated code.

    \n

    Any number of reactors may be defined in one file, and a $main$[ or $federated$]{federated} reactor need not be given a name, but if it is given a name, then that name must match the file name.

    \n

    Reactors may extend other reactors, inheriting their properties, and a file may import reactors from other files. If an imported LF file contains a $main$[ or $federated$]{federated} reactor, that reactor is ignored (it will not be imported). This makes it easy to create a library of reusable reactors that each come with a test case or demonstration in the form of a main reactor.

    \n

    Comments

    \n

    Lingua Franca files can have C/C++/Java-style comments and/or Python-style comments. All of the following are valid comments:

    \n
        // Single-line C-style comment.\n    /*\n     * Multi-line C-style comment.\n     */\n    # Single-line Python-style comment.
    ","headings":[{"value":"Minimal Example","depth":2},{"value":"Examples","depth":2},{"value":"Structure of an LF Project","depth":2},{"value":"Reactor Block","depth":2},{"value":"Comments","depth":2}],"frontmatter":{"permalink":"/docs/handbook/a-first-reactor","title":"A First Reactor","oneline":"Writing your first Lingua Franca reactor.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","permalink":"/docs/handbook/inputs-and-outputs"}}}},"pageContext":{"id":"1-a-first-reactor","slug":"/docs/handbook/a-first-reactor","repoPath":"/packages/documentation/copy/en/topics/A First Reactor.md","nextID":"dcdc6b32-76b0-570a-a6f8-23bb570863c7","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/actions/page-data.json b/page-data/docs/handbook/actions/page-data.json index 75eef7121..01800942d 100644 --- a/page-data/docs/handbook/actions/page-data.json +++ b/page-data/docs/handbook/actions/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/actions","result":{"data":{"markdownRemark":{"id":"dd46179b-1189-5c1f-a60a-37e302c85efc","excerpt":"$page-showing-target$ Action Declaration An action declaration has one of the following forms: The min_delay, min_spacing, and policy are all optional. If only…","html":"

    $page-showing-target$

    \n

    Action Declaration

    \n

    An action declaration has one of the following forms:

    \n
      logical action <name>(<min_delay>, <min_spacing>, <policy>)\n  physical action <name>(<min_delay>, <min_spacing>, <policy>)\n
    \n

    The min_delay, min_spacing, and policy are all optional. If only one argument is given in parentheses, then it is interpreted as an min_delay, if two are given, then they are interpreted as min_delay and min_spacing. The min_delay and min_spacing are time values. The policy argument is a string that can be one of the following: \"defer\" (the default), \"drop\", or \"replace\". Note that the quotation marks are needed.

    \n
    \n

    If the action is to carry a payload, then a type must be given as well:

    \n
      logical action <name>(<min_delay>, <min_spacing>, <policy>):<type>\n  physical action <name>(<min_delay>, <min_spacing>, <policy>):<type>\n
    \n
    \n

    Logical Actions

    \n

    Timers are useful to trigger reactions once or periodically. Actions are used to trigger reactions more irregularly. An action, like an output or input port, can carry data, but unlike a port, an action is visible only within the reactor that defines it.

    \n

    There are two kinds of actions, logical and physical. A $logical$ $action$ is used by a reactor to schedule a trigger at a fixed logical time interval d into the future. The time interval d, which is called a delay, is relative to the logical time t at which the scheduling occurs. If a reaction executes at logical time t and schedules an action a with delay d, then any reaction that is triggered by a will be invoked at logical time t + d. For example, the following reaction schedules something (printing the current elapsed logical time) 200 msec after an input x arrives:

    \n

    $start(Schedule)$

    \n
    target C;\nreactor Schedule {\n  input x:int;\n  logical action a;\n  reaction(x) -> a {=\n    lf_schedule(a, MSEC(200));\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp;\nreactor Schedule {\n  input x:int;\n  logical action a;\n  reaction(x) -> a {=\n    a.schedule(200ms);\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time << "  nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python;\nreactor Schedule {\n  input x;\n  logical action a;\n  reaction(x) -> a {=\n    a.schedule(MSEC(200))\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nreactor Schedule {\n  input x:number\n  logical action a\n  reaction(x) -> a {=\n    actions.a.schedule(TimeValue.nsecs(200), null)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} after start.`)\n  =}\n}\n
    \n
    target Rust;\nreactor Schedule {\n  input x:u32;\n  logical action a;\n  reaction(x) -> a {=\n    ctx.schedule(a, after!(200 ms));\n  =}\n  reaction(a) {=\n    printf("\n      Action triggered at logical time {} nsec after start.",\n      ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Schedule)$

    \n\"Lingua\n

    Here, the delay is specified in the call to schedule within the target language code. Notice that in the diagram, a logical action is shown as a triangle with an L. Logical actions are always scheduled within a reaction of the reactor that declares the action.

    \n

    The time argument is required to be non-negative. If it is zero, then the action will be scheduled one microstep later. See Superdense Time.

    \n
    \n

    The arguments to the lf_schedule() function are the action named a and a time. The action a has to be declared as an effect of the reaction in order to reference it in the call to lf_schedule(). If you fail to declare it as an effect (after the -> in the reaction signature), then you will get an error message.

    \n

    The time argument to the lf_schedule() function has data type interval_t, which, with the exception of some embedded platforms, is a C int64_t. A collection of convenience macros is provided like the MSEC macro above to specify time values in a more readable way. The provided macros are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these, e.g. WEEKS(2).

    \n

    An action may have a data type, in which case, a variant of the lf_schedule() function can be used to specify a payload, a data value that is carried from where the lf_schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n
    \n

    An action may have a data type, in which case, a variant of the schedule() function can be used to specify a payload, a data value that is carried from where the schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n
    \n

    The arguments to the a.schedule() method is a time. The action a has to be\ndeclared as an effect of the reaction in order to reference it in the body of\nthe reaction. If you fail to declare it as an effect (after the -> in the\nreaction signature), then you will get a runtime error message.

    \n

    The time argument to the a.schedule() method expects an integer. A collection\nof convenience functions is provided like the MSEC function above to specify\ntime values in a more readable way. The provided functions are NSEC, USEC\n(for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may\nalso use the plural of any of these, e.g. WEEKS(2).

    \n

    An action may carry data, in which case, the payload data value is just given as a second argument to the .schedule() method. See the Target Language Details.

    \n
    \n
    \n

    The schedule() method of an action takes two arguments, a TimeValue and an (optional) payload. If a payload is given and a type is given for the action, then the type of the payload must match the type of the action. See the Target Language Details for details.

    \n
    \n
    \n

    FIXME

    \n

    An action may have a data type, in which case, a variant of the schedule() function can be used to specify a payload, a data value that is carried from where the schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n

    Physical Actions

    \n

    A $physical$ $action$ is used to schedule reactions at logical times determined by the local physical clock. If a physical action with delay d is scheduled at physical time T, then the logical time assigned to the event is T + d. For example, the following reactor schedules the physical action p to trigger at a logical time equal to the physical time at which the input x arrives:

    \n

    $start(Physical)$

    \n
    target C;\nreactor Physical {\n  input x:int;\n  physical action a;\n  reaction(x) -> a {=\n    lf_schedule(a, 0);\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp;\nreactor Physical {\n  input x:int;\n  physical action a;\n  reaction(x) -> a {=\n    a.schedule(0ms);\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time << " nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python;\nreactor Physical {\n  input x;\n  physical action a;\n  reaction(x) -> a {=\n    a.schedule(0)\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nreactor Physical {\n  input x:int\n  physical action a\n  reaction(x) -> a {=\n    actions.a.schedule(TimeValue.zero(), null)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} nsec after start.`)\n  =}\n}\n
    \n
    target Rust;\nreactor Physical {\n  input x:u32;\n  physical action a;\n  reaction(x) -> a {=\n    let phys_action = a.clone();\n    ctx.spawn_physical_thread(move |link| {\n      link.schedule(&phys_action, Asap).unwrap();\n    });\n  =}\n  reaction(a) {=\n    println!(\n      "Action triggered at logical time {} nsec after start.",\n      ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Physical)$

    \n\"Lingua\n

    If you drive this with a timer, using for example the following structure:

    \n\"Lingua\n

    then running the program will yield an output something like this:

    \n
    Action triggered at logical time 201491000 nsec after start.\nAction triggered at logical time 403685000 nsec after start.\nAction triggered at logical time 603669000 nsec after start.\n...
    \n

    Here, logical time is lagging physical time by a few milliseconds. Note that, unless the fast option is given, logical time t chases physical time T, so t < T. Hence, the event being scheduled in the reaction to input x is assured of being in the future in logical time.

    \n

    Whereas logical actions are required to be scheduled within a reaction of the reactor that declares the action, physical actions can be scheduled by code that is outside the Lingua Franca system. For example, some other thread or a callback function may call schedule(), passing it a physical action. For example:

    \n

    $start(Asynchronous)$

    \n
    target C {\n  keepalive: true  // Do not exit when event queue is empty.\n}\npreamble {=\n  #include "platform.h" // Defines lf_sleep() and thread functions.\n=}\nmain reactor {\n  preamble {=\n    // Schedule an event roughly every 200 msec.\n    void* external(void* a) {\n      while (true) {\n        lf_sleep(MSEC(200));\n        lf_schedule(a, 0);\n      }\n    }\n  =}\n  state thread_id: lf_thread_t = 0\n  physical action a(100 msec): int\n  reaction(startup) -> a {=\n    // Start a thread to schedule physical actions.\n    lf_thread_create(&self->thread_id, &external, a);\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  private preamble {=\n    #include <thread>\n  =}\n  state thread: std::thread\n  physical action a: int\n  reaction(startup) -> a {=\n    // Start a thread to schedule physical actions.\n    thread = std::thread([&]{\n      while (true) {\n        std::this_thread::sleep_for(200ms);\n        // the value that we give it really doesn't matter\n        // but we the action should is scheduled for 100ms into the future\n        a.schedule(0, 100ms);\n      }\n    });\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_physical_time();\n    std::cout << "Action triggered at logical time" << elapsed_time <<"nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  preamble {=\n    import time\n    import threading\n    # Schedule an event roughly every 200 msec.\n    def external(self, a):\n      while (True):\n        self.time.sleep(0.2)\n        a.schedule(0)\n  =}\n  state thread\n  physical action a(100 msec)\n  reaction(startup) -> a {=\n    # Start a thread to schedule physical actions.\n    self.thread = self.threading.Thread(target=self.external, args=(a,))\n    self.thread.start()\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  physical action a(100 msec): number\n  reaction(startup) -> a {=\n    // Have asynchronous callback schedule physical action.\n    setTimeout(() => {\n      actions.a.schedule(TimeValue.zero(), 0)\n    }, 200)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} nsec after start.`)\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  state start_time: Instant = {= Instant::now() =}\n  physical action a(100 msec): u32\n  reaction(startup) -> a {=\n    let phys_action = a.clone(); // clone to move it into other thread\n    // Start a thread to schedule physical actions.\n    ctx.spawn_physical_thread(move |link| {\n      loop {\n        std::thread::sleep(Duration::from_millis(200));\n        link.schedule_physical(&phys_action, Asap).unwrap();\n      }\n    });\n  =}\n  reaction(a) {=\n    let elapsed_time = self.start_time.elapsed();\n    println!("Action triggered at logical time {} nsecs after start.", elapsed_time.as_nanos());\n  =}\n}\n
    \n

    $end(Asynchronous)$

    \n\"Lingua\n

    Physical actions are the mechanism for obtaining input from the outside world. Because they are assigned a logical time derived from the physical clock, their logical time can be interpreted as a measure of the time at which some external event occurred.

    \n
    \n

    In the above example, at $startup$, the main reactor creates an external thread that schedules a physical action roughly every 200 msec.

    \n

    First, the file-level $preamble$ has #include \"platform.h\", which includes the declarations for functions it uses, lf_sleep and lf_thread_create (see Libraries Available to Programmers).

    \n

    Second, the thread uses a function lf_sleep(), which abstracts platform-specific mechanisms for stalling the thread for a specified amount of time, and lf_thread_create(), which abstracts platform-specific mechanisms for creating threads.

    \n

    The external function executed by the thread is defined in a reactor-level $preamble$ section. See Preambles.

    \n
    \n

    Triggering Time for Actions

    \n

    An action will trigger at a logical time that depends on the arguments given to the schedule function, the <min_delay>, <min_spacing>, and <policy> arguments in the action declaration, and whether the action is physical or logical.

    \n

    For a $logical$ action a, the tag assigned to the event resulting from a call to schedule() is computed as follows. First, let t be the current logical time. For a logical action, t is just the logical time at which the reaction calling schedule() is called. The preliminary time of the action is then just t + <min_delay> + <offset>. This preliminary time may be further modified, as explained below.

    \n

    For a physical action, the preliminary time is similar, except that t is replaced by the current physical time T when schedule() is called.

    \n

    If a <min_spacing> has been declared, then it gives a minimum logical time\ninterval between the tags of two subsequently scheduled events. If the\npreliminary time is closer than <min_spacing> to the time of the previously\nscheduled event (if there is one), then <policy> (if supported by the target)\ndetermines how the minimum spacing constraint is enforced.

    \n
    \n

    The <policy> is one of the following:

    \n
      \n
    • \"defer\": (the default) The event is added to the event queue with a tag that is equal to earliest time that satisfies the minimal spacing requirement. Assuming the time of the preceding event is t_prev, then the tag of the new event simply becomes t_prev + <min_spacing>.
    • \n
    • \"drop\": The new event is dropped and schedule() returns without having modified the event queue.
    • \n
    • \"replace\": The payload (if any) of the new event is assigned to the preceding event if it is still pending in the event queue; no new event is added to the event queue in this case. If the preceding event has already been pulled from the event queue, the default \"defer\" policy is applied.
    • \n
    \n

    Note that while the \"defer\" policy is conservative in the sense that it does not discard events, it could potentially cause an unbounded growth of the event queue.

    \n
    \n
    \n
    \n

    The <policy> argument is currently not supported.

    \n
    \n
    \n

    Testing an Action for Presence

    \n

    When a reaction is triggered by more than one action or by an action and an input, it may be necessary to test within the reaction whether the action is present.\nJust like for inputs, this can be done in the C target with a->is_present, where a is the name of the action.\nJust like for inputs, this can be done in the Python target with a.is_present, where a is the name of the action.\nJust like for inputs, this can be done in the C++ target with a.is_present(), where a is the name of the action.\nJust like for inputs, this can be done in the TypeScript target with a != undefined, where a is the name of the action.\nJust like for inputs, this can be done in the Rust target with ctx.is_present(a), where a is the name of the action.

    ","headings":[{"value":"Action Declaration","depth":2},{"value":"Logical Actions","depth":2},{"value":"Physical Actions","depth":2},{"value":"Triggering Time for Actions","depth":2},{"value":"Testing an Action for Presence","depth":2}],"frontmatter":{"permalink":"/docs/handbook/actions","title":"Actions","oneline":"Actions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","permalink":"/docs/handbook/extending-reactors"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","permalink":"/docs/handbook/superdense-time"}}}},"pageContext":{"id":"1-actions","slug":"/docs/handbook/actions","repoPath":"/packages/documentation/copy/en/topics/Actions.md","previousID":"d4ade317-2d91-582b-8cca-37743461f687","nextID":"1986cb1b-8feb-57bc-8088-50548df2f061","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/actions","result":{"data":{"markdownRemark":{"id":"dd46179b-1189-5c1f-a60a-37e302c85efc","excerpt":"$page-showing-target$ Action Declaration An action declaration has one of the following forms: The min_delay, min_spacing, and policy are all optional. If only…","html":"

    $page-showing-target$

    \n

    Action Declaration

    \n

    An action declaration has one of the following forms:

    \n
      logical action <name>(<min_delay>, <min_spacing>, <policy>)\n  physical action <name>(<min_delay>, <min_spacing>, <policy>)\n
    \n

    The min_delay, min_spacing, and policy are all optional. If only one argument is given in parentheses, then it is interpreted as an min_delay, if two are given, then they are interpreted as min_delay and min_spacing. The min_delay and min_spacing are time values. The policy argument is a string that can be one of the following: \"defer\" (the default), \"drop\", or \"replace\". Note that the quotation marks are needed.

    \n
    \n

    If the action is to carry a payload, then a type must be given as well:

    \n
      logical action <name>(<min_delay>, <min_spacing>, <policy>):<type>\n  physical action <name>(<min_delay>, <min_spacing>, <policy>):<type>\n
    \n
    \n

    Logical Actions

    \n

    Timers are useful to trigger reactions once or periodically. Actions are used to trigger reactions more irregularly. An action, like an output or input port, can carry data, but unlike a port, an action is visible only within the reactor that defines it.

    \n

    There are two kinds of actions, logical and physical. A $logical$ $action$ is used by a reactor to schedule a trigger at a fixed logical time interval d into the future. The time interval d, which is called a delay, is relative to the logical time t at which the scheduling occurs. If a reaction executes at logical time t and schedules an action a with delay d, then any reaction that is triggered by a will be invoked at logical time t + d. For example, the following reaction schedules something (printing the current elapsed logical time) 200 msec after an input x arrives:

    \n

    $start(Schedule)$

    \n
    target C;\nreactor Schedule {\n  input x:int;\n  logical action a;\n  reaction(x) -> a {=\n    lf_schedule(a, MSEC(200));\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp;\nreactor Schedule {\n  input x:int;\n  logical action a;\n  reaction(x) -> a {=\n    a.schedule(200ms);\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time << "  nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python;\nreactor Schedule {\n  input x;\n  logical action a;\n  reaction(x) -> a {=\n    a.schedule(MSEC(200))\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nreactor Schedule {\n  input x:number\n  logical action a\n  reaction(x) -> a {=\n    actions.a.schedule(TimeValue.nsecs(200), null)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} after start.`)\n  =}\n}\n
    \n
    target Rust;\nreactor Schedule {\n  input x:u32;\n  logical action a;\n  reaction(x) -> a {=\n    ctx.schedule(a, after!(200 ms));\n  =}\n  reaction(a) {=\n    printf("\n      Action triggered at logical time {} nsec after start.",\n      ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Schedule)$

    \n\"Lingua\n

    Here, the delay is specified in the call to schedule within the target language code. Notice that in the diagram, a logical action is shown as a triangle with an L. Logical actions are always scheduled within a reaction of the reactor that declares the action.

    \n

    The time argument is required to be non-negative. If it is zero, then the action will be scheduled one microstep later. See Superdense Time.

    \n
    \n

    The arguments to the lf_schedule() function are the action named a and a time. The action a has to be declared as an effect of the reaction in order to reference it in the call to lf_schedule(). If you fail to declare it as an effect (after the -> in the reaction signature), then you will get an error message.

    \n

    The time argument to the lf_schedule() function has data type interval_t, which, with the exception of some embedded platforms, is a C int64_t. A collection of convenience macros is provided like the MSEC macro above to specify time values in a more readable way. The provided macros are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these, e.g. WEEKS(2).

    \n

    An action may have a data type, in which case, a variant of the lf_schedule() function can be used to specify a payload, a data value that is carried from where the lf_schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n
    \n

    An action may have a data type, in which case, a variant of the schedule() function can be used to specify a payload, a data value that is carried from where the schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n
    \n

    The arguments to the a.schedule() method is a time. The action a has to be\ndeclared as an effect of the reaction in order to reference it in the body of\nthe reaction. If you fail to declare it as an effect (after the -> in the\nreaction signature), then you will get a runtime error message.

    \n

    The time argument to the a.schedule() method expects an integer. A collection\nof convenience functions is provided like the MSEC function above to specify\ntime values in a more readable way. The provided functions are NSEC, USEC\n(for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may\nalso use the plural of any of these, e.g. WEEKS(2).

    \n

    An action may carry data, in which case, the payload data value is just given as a second argument to the .schedule() method. See the Target Language Details.

    \n
    \n
    \n

    The schedule() method of an action takes two arguments, a TimeValue and an (optional) payload. If a payload is given and a type is given for the action, then the type of the payload must match the type of the action. See the Target Language Details for details.

    \n
    \n
    \n

    FIXME

    \n

    An action may have a data type, in which case, a variant of the schedule() function can be used to specify a payload, a data value that is carried from where the schedule() function is called to the reaction that is triggered by the action. See the Target Language Details.

    \n
    \n

    Physical Actions

    \n

    A $physical$ $action$ is used to schedule reactions at logical times determined by the local physical clock. If a physical action with delay d is scheduled at physical time T, then the logical time assigned to the event is T + d. For example, the following reactor schedules the physical action p to trigger at a logical time equal to the physical time at which the input x arrives:

    \n

    $start(Physical)$

    \n
    target C;\nreactor Physical {\n  input x:int;\n  physical action a;\n  reaction(x) -> a {=\n    lf_schedule(a, 0);\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp;\nreactor Physical {\n  input x:int;\n  physical action a;\n  reaction(x) -> a {=\n    a.schedule(0ms);\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time << " nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python;\nreactor Physical {\n  input x;\n  physical action a;\n  reaction(x) -> a {=\n    a.schedule(0)\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nreactor Physical {\n  input x:int\n  physical action a\n  reaction(x) -> a {=\n    actions.a.schedule(TimeValue.zero(), null)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} nsec after start.`)\n  =}\n}\n
    \n
    target Rust;\nreactor Physical {\n  input x:u32;\n  physical action a;\n  reaction(x) -> a {=\n    let phys_action = a.clone();\n    ctx.spawn_physical_thread(move |link| {\n      link.schedule(&phys_action, Asap).unwrap();\n    });\n  =}\n  reaction(a) {=\n    println!(\n      "Action triggered at logical time {} nsec after start.",\n      ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Physical)$

    \n\"Lingua\n

    If you drive this with a timer, using for example the following structure:

    \n\"Lingua\n

    then running the program will yield an output something like this:

    \n
    Action triggered at logical time 201491000 nsec after start.\nAction triggered at logical time 403685000 nsec after start.\nAction triggered at logical time 603669000 nsec after start.\n...
    \n

    Here, logical time is lagging physical time by a few milliseconds. Note that, unless the fast option is given, logical time t chases physical time T, so t < T. Hence, the event being scheduled in the reaction to input x is assured of being in the future in logical time.

    \n

    Whereas logical actions are required to be scheduled within a reaction of the reactor that declares the action, physical actions can be scheduled by code that is outside the Lingua Franca system. For example, some other thread or a callback function may call schedule(), passing it a physical action. For example:

    \n

    $start(Asynchronous)$

    \n
    target C {\n  keepalive: true  // Do not exit when event queue is empty.\n}\npreamble {=\n  #include "platform.h" // Defines lf_sleep() and thread functions.\n=}\nmain reactor {\n  preamble {=\n    // Schedule an event roughly every 200 msec.\n    void* external(void* a) {\n      while (true) {\n        lf_sleep(MSEC(200));\n        lf_schedule(a, 0);\n      }\n    }\n  =}\n  state thread_id: lf_thread_t = 0\n  physical action a(100 msec): int\n  reaction(startup) -> a {=\n    // Start a thread to schedule physical actions.\n    lf_thread_create(&self->thread_id, &external, a);\n  =}\n  reaction(a) {=\n    interval_t elapsed_time = lf_time_logical_elapsed();\n    printf("Action triggered at logical time %lld nsec after start.\\n", elapsed_time);\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  private preamble {=\n    #include <thread>\n  =}\n  state thread: std::thread\n  physical action a: int\n  reaction(startup) -> a {=\n    // Start a thread to schedule physical actions.\n    thread = std::thread([&]{\n      while (true) {\n        std::this_thread::sleep_for(200ms);\n        // the value that we give it really doesn't matter\n        // but we the action should is scheduled for 100ms into the future\n        a.schedule(0, 100ms);\n      }\n    });\n  =}\n  reaction(a) {=\n    auto elapsed_time = get_physical_time();\n    std::cout << "Action triggered at logical time" << elapsed_time <<"nsec after start." << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  preamble {=\n    import time\n    import threading\n    # Schedule an event roughly every 200 msec.\n    def external(self, a):\n      while (True):\n        self.time.sleep(0.2)\n        a.schedule(0)\n  =}\n  state thread\n  physical action a(100 msec)\n  reaction(startup) -> a {=\n    # Start a thread to schedule physical actions.\n    self.thread = self.threading.Thread(target=self.external, args=(a,))\n    self.thread.start()\n  =}\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"Action triggered at logical time {elapsed_time} nsec after start.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  physical action a(100 msec): number\n  reaction(startup) -> a {=\n    // Have asynchronous callback schedule physical action.\n    setTimeout(() => {\n      actions.a.schedule(TimeValue.zero(), 0)\n    }, 200)\n  =}\n  reaction(a) {=\n    console.log(`Action triggered at logical time ${util.getElapsedLogicalTime()} nsec after start.`)\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  state start_time: Instant = {= Instant::now() =}\n  physical action a(100 msec): u32\n  reaction(startup) -> a {=\n    let phys_action = a.clone(); // clone to move it into other thread\n    // Start a thread to schedule physical actions.\n    ctx.spawn_physical_thread(move |link| {\n      loop {\n        std::thread::sleep(Duration::from_millis(200));\n        link.schedule_physical(&phys_action, Asap).unwrap();\n      }\n    });\n  =}\n  reaction(a) {=\n    let elapsed_time = self.start_time.elapsed();\n    println!("Action triggered at logical time {} nsecs after start.", elapsed_time.as_nanos());\n  =}\n}\n
    \n

    $end(Asynchronous)$

    \n\"Lingua\n

    Physical actions are the mechanism for obtaining input from the outside world. Because they are assigned a logical time derived from the physical clock, their logical time can be interpreted as a measure of the time at which some external event occurred.

    \n
    \n

    In the above example, at $startup$, the main reactor creates an external thread that schedules a physical action roughly every 200 msec.

    \n

    First, the file-level $preamble$ has #include \"platform.h\", which includes the declarations for functions it uses, lf_sleep and lf_thread_create (see Libraries Available to Programmers).

    \n

    Second, the thread uses a function lf_sleep(), which abstracts platform-specific mechanisms for stalling the thread for a specified amount of time, and lf_thread_create(), which abstracts platform-specific mechanisms for creating threads.

    \n

    The external function executed by the thread is defined in a reactor-level $preamble$ section. See Preambles.

    \n
    \n

    Triggering Time for Actions

    \n

    An action will trigger at a logical time that depends on the arguments given to the schedule function, the <min_delay>, <min_spacing>, and <policy> arguments in the action declaration, and whether the action is physical or logical.

    \n

    For a $logical$ action a, the tag assigned to the event resulting from a call to schedule() is computed as follows. First, let t be the current logical time. For a logical action, t is just the logical time at which the reaction calling schedule() is called. The preliminary time of the action is then just t + <min_delay> + <offset>. This preliminary time may be further modified, as explained below.

    \n

    For a physical action, the preliminary time is similar, except that t is replaced by the current physical time T when schedule() is called.

    \n

    If a <min_spacing> has been declared, then it gives a minimum logical time\ninterval between the tags of two subsequently scheduled events. If the\npreliminary time is closer than <min_spacing> to the time of the previously\nscheduled event (if there is one), then <policy> (if supported by the target)\ndetermines how the minimum spacing constraint is enforced.

    \n
    \n

    The <policy> is one of the following:

    \n
      \n
    • \"defer\": (the default) The event is added to the event queue with a tag that is equal to earliest time that satisfies the minimal spacing requirement. Assuming the time of the preceding event is t_prev, then the tag of the new event simply becomes t_prev + <min_spacing>.
    • \n
    • \"drop\": The new event is dropped and schedule() returns without having modified the event queue.
    • \n
    • \"replace\": The payload (if any) of the new event is assigned to the preceding event if it is still pending in the event queue; no new event is added to the event queue in this case. If the preceding event has already been pulled from the event queue, the default \"defer\" policy is applied.
    • \n
    \n

    Note that while the \"defer\" policy is conservative in the sense that it does not discard events, it could potentially cause an unbounded growth of the event queue.

    \n
    \n
    \n
    \n

    The <policy> argument is currently not supported.

    \n
    \n
    \n

    Testing an Action for Presence

    \n

    When a reaction is triggered by more than one action or by an action and an input, it may be necessary to test within the reaction whether the action is present.\nJust like for inputs, this can be done in the C target with a->is_present, where a is the name of the action.\nJust like for inputs, this can be done in the Python target with a.is_present, where a is the name of the action.\nJust like for inputs, this can be done in the C++ target with a.is_present(), where a is the name of the action.\nJust like for inputs, this can be done in the TypeScript target with a != undefined, where a is the name of the action.\nJust like for inputs, this can be done in the Rust target with ctx.is_present(a), where a is the name of the action.

    ","headings":[{"value":"Action Declaration","depth":2},{"value":"Logical Actions","depth":2},{"value":"Physical Actions","depth":2},{"value":"Triggering Time for Actions","depth":2},{"value":"Testing an Action for Presence","depth":2}],"frontmatter":{"permalink":"/docs/handbook/actions","title":"Actions","oneline":"Actions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","permalink":"/docs/handbook/extending-reactors"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","permalink":"/docs/handbook/superdense-time"}}}},"pageContext":{"id":"1-actions","slug":"/docs/handbook/actions","repoPath":"/packages/documentation/copy/en/topics/Actions.md","previousID":"d4ade317-2d91-582b-8cca-37743461f687","nextID":"1986cb1b-8feb-57bc-8088-50548df2f061","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/arduino/page-data.json b/page-data/docs/handbook/arduino/page-data.json index ff1bae43d..c7a174c13 100644 --- a/page-data/docs/handbook/arduino/page-data.json +++ b/page-data/docs/handbook/arduino/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/arduino","result":{"data":{"markdownRemark":{"id":"77295cb2-a818-5f3d-9d3c-9ad07392edf2","excerpt":"Overview To run Lingua Franca on an Arduino-compatible microcontroller, you can use the\nC target with the platform target property set to arduino. The Lingua…","html":"

    Overview

    \n

    To run Lingua Franca on an Arduino-compatible microcontroller, you can use the\nC target with the platform target property set to arduino. The Lingua\nFranca compiler will then not output ordinary C code in a CMake project, but\ngenerate a .ino sketch that can be compiled and deployed using arduino-cli.\nTo flash the compiled sketch onto a board, you need specify a Fully Qualified\nBoard Name (FQBN) as well as the port to which your board is connected. On this\npage, we explain exactly how to do this.

    \n

    Prerequisites

    \n
      \n
    • You need a development system that runs on macOS or Linux (there is currently no Windows support).
    • \n
    • Arduino CLI must be installed on your development system. Confirm that your installation works by running:
    • \n
    \n
    arduino-cli version
    \n

    Writing a simple Arduino Program

    \n

    The most basic Arduino program (Blink) can be defined in LF like so:

    \n
    target C {\n  platform: "Arduino"\n}\n\nmain reactor Blink {\n  timer t1(0, 1 sec)\n  timer t2(500 msec, 1 sec)\n  reaction(startup) {=\n    pinMode(LED_BUILTIN, OUTPUT);\n  =}\n\n  reaction (t1) {=\n    digitalWrite(LED_BUILTIN, HIGH);\n  =}\n\n  reaction (t2) {=\n    digitalWrite(LED_BUILTIN, LOW);\n  =}\n}\n
    \n

    The key difference between writing ordinary Arduino code and writing LF code is\nthat there is no setup() and loop() function. In LF, execution is\nevent-driven, with reactive code being triggered by events, such as the\nexpiration of a timer. Those pieces of reactive code, specified as reactions,\ncan invoke Arduino library functions, just like one would do in the definition\nof an Arduino loop(). For any setup that might be needed at the beginning of\nexecution, a reaction triggered by the built-in startup trigger can be used.

    \n

    Platform Options

    \n

    The platform property can also be used so specify more details. Along with name: \"arduino\",\nyou can specify which board, port, and baud-rate you are using. If you want the program\nto be flashed onto the board automatically after compilation, specify flash: true.\nHere a target declaration that specifies all of these options:

    \n
    target C {\n  platform: {\n    name: "arduino",\n    board: "arduino:avr:uno",\n    port: "/dev/ttyd2",\n    baud-rate: 115200,\n    flash: true\n  }\n}\n
    \n

    The board is necessary for building and the port is necessary for flashing.

    \n

    Baud rate of the serial port

    \n

    All Arduino boards have at least one serial port (also known as a UART or\nUSART), and some have several. By default, Lingua Franca will assume a default\nbaud rate of 9600. This parameter is tunable by adjusting the baud-rate\nparameter in platform options.

    \n

    Building

    \n

    In order to have arduino-cli compile the generated sketch file, a board must\nbe specified. If no board is set, lfc will run in no-compile mode, meaning\nit will not invoke the target compiler. Whether specified as a target property\nor given directly to arduino-cli, you need an FQBN. You can find the FQBN of\nthe board that you have plugged in using the following command:

    \n

    arduino-cli board list

    \n

    To see the list of all FQBN’s installed on your device (all FQBNs supported by the libraries you have installed), run the following command:

    \n

    arduino-cli board listall

    \n

    You likely need to install support for your board first, which you can do using the following command:

    \n

    arduino-cli core install arduino:[BOARD_FAMILY]

    \n

    The common board families include avr, megaAVR, sam, samd, and mbed.

    \n

    If you specify your FQBN under board in the platform target property, lfc will automatically invoke arduino-cli on the generated sketch. To invoke arduino-cli manually

    \n
      \n
    • \n

      for unthreaded programs (most arduino flavors), run:\narduino-cli compile -b [FQBN] --build-property compiler.c.extra_flags=\"-DLF_UNTHREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\" --build-property compiler.cpp.extra_flags=\"-DLF_UNTHREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\"

      \n
    • \n
    • \n

      for threaded programs (arduino:mbed boards), run:\narduino-cli compile -b [FQBN] --build-property compiler.c.extra_flags=\"-DLF_THREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\" --build-property compiler.cpp.extra_flags=\"-DLF_THREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\"

      \n
    • \n
    \n

    Flashing

    \n

    Arduino’s can be flashed via USB. To find the port oto which your device is connected, run the following command:

    \n

    arduino-cli board list

    \n

    You can either use arduino-cli to flash the code onto your device after the sketch file has been built, or you can set flash:true and provide a port in your Lingua Franca file. To flash manually using arduino-cli, enter the subdirectory of src-gen in which the generated sketch file is located and run:

    \n

    arduino-cli upload -p PORT -b FQBN .

    ","headings":[{"value":"Overview","depth":1},{"value":"Prerequisites","depth":2},{"value":"Writing a simple Arduino Program","depth":1},{"value":"Platform Options","depth":2},{"value":"Baud rate of the serial port","depth":3},{"value":"Building","depth":1},{"value":"Flashing","depth":1}],"frontmatter":{"permalink":"/docs/handbook/arduino","title":"Arduino","oneline":"Developing LF Programs on Arduino.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Zephyr","oneline":"Developing LF Programs for Zephyr RTOS.","permalink":"/docs/handbook/zephyr"}}}},"pageContext":{"id":"4-arduino","slug":"/docs/handbook/arduino","repoPath":"/packages/documentation/copy/en/embedded/Arduino.md","nextID":"3f41a520-2093-55ff-a5e7-36ec8f69920a","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/arduino","result":{"data":{"markdownRemark":{"id":"77295cb2-a818-5f3d-9d3c-9ad07392edf2","excerpt":"Overview To run Lingua Franca on an Arduino-compatible microcontroller, you can use the\nC target with the platform target property set to arduino. The Lingua…","html":"

    Overview

    \n

    To run Lingua Franca on an Arduino-compatible microcontroller, you can use the\nC target with the platform target property set to arduino. The Lingua\nFranca compiler will then not output ordinary C code in a CMake project, but\ngenerate a .ino sketch that can be compiled and deployed using arduino-cli.\nTo flash the compiled sketch onto a board, you need specify a Fully Qualified\nBoard Name (FQBN) as well as the port to which your board is connected. On this\npage, we explain exactly how to do this.

    \n

    Prerequisites

    \n
      \n
    • You need a development system that runs on macOS or Linux (there is currently no Windows support).
    • \n
    • Arduino CLI must be installed on your development system. Confirm that your installation works by running:
    • \n
    \n
    arduino-cli version
    \n

    Writing a simple Arduino Program

    \n

    The most basic Arduino program (Blink) can be defined in LF like so:

    \n
    target C {\n  platform: "Arduino"\n}\n\nmain reactor Blink {\n  timer t1(0, 1 sec)\n  timer t2(500 msec, 1 sec)\n  reaction(startup) {=\n    pinMode(LED_BUILTIN, OUTPUT);\n  =}\n\n  reaction (t1) {=\n    digitalWrite(LED_BUILTIN, HIGH);\n  =}\n\n  reaction (t2) {=\n    digitalWrite(LED_BUILTIN, LOW);\n  =}\n}\n
    \n

    The key difference between writing ordinary Arduino code and writing LF code is\nthat there is no setup() and loop() function. In LF, execution is\nevent-driven, with reactive code being triggered by events, such as the\nexpiration of a timer. Those pieces of reactive code, specified as reactions,\ncan invoke Arduino library functions, just like one would do in the definition\nof an Arduino loop(). For any setup that might be needed at the beginning of\nexecution, a reaction triggered by the built-in startup trigger can be used.

    \n

    Platform Options

    \n

    The platform property can also be used so specify more details. Along with name: \"arduino\",\nyou can specify which board, port, and baud-rate you are using. If you want the program\nto be flashed onto the board automatically after compilation, specify flash: true.\nHere a target declaration that specifies all of these options:

    \n
    target C {\n  platform: {\n    name: "arduino",\n    board: "arduino:avr:uno",\n    port: "/dev/ttyd2",\n    baud-rate: 115200,\n    flash: true\n  }\n}\n
    \n

    The board is necessary for building and the port is necessary for flashing.

    \n

    Baud rate of the serial port

    \n

    All Arduino boards have at least one serial port (also known as a UART or\nUSART), and some have several. By default, Lingua Franca will assume a default\nbaud rate of 9600. This parameter is tunable by adjusting the baud-rate\nparameter in platform options.

    \n

    Building

    \n

    In order to have arduino-cli compile the generated sketch file, a board must\nbe specified. If no board is set, lfc will run in no-compile mode, meaning\nit will not invoke the target compiler. Whether specified as a target property\nor given directly to arduino-cli, you need an FQBN. You can find the FQBN of\nthe board that you have plugged in using the following command:

    \n

    arduino-cli board list

    \n

    To see the list of all FQBN’s installed on your device (all FQBNs supported by the libraries you have installed), run the following command:

    \n

    arduino-cli board listall

    \n

    You likely need to install support for your board first, which you can do using the following command:

    \n

    arduino-cli core install arduino:[BOARD_FAMILY]

    \n

    The common board families include avr, megaAVR, sam, samd, and mbed.

    \n

    If you specify your FQBN under board in the platform target property, lfc will automatically invoke arduino-cli on the generated sketch. To invoke arduino-cli manually

    \n
      \n
    • \n

      for unthreaded programs (most arduino flavors), run:\narduino-cli compile -b [FQBN] --build-property compiler.c.extra_flags=\"-DLF_UNTHREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\" --build-property compiler.cpp.extra_flags=\"-DLF_UNTHREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\"

      \n
    • \n
    • \n

      for threaded programs (arduino:mbed boards), run:\narduino-cli compile -b [FQBN] --build-property compiler.c.extra_flags=\"-DLF_THREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\" --build-property compiler.cpp.extra_flags=\"-DLF_THREADED -DPLATFORM_ARDUINO -DINITIAL_EVENT_QUEUE_SIZE=10 -DINITIAL_REACT_QUEUE_SIZE=10\"

      \n
    • \n
    \n

    Flashing

    \n

    Arduino’s can be flashed via USB. To find the port oto which your device is connected, run the following command:

    \n

    arduino-cli board list

    \n

    You can either use arduino-cli to flash the code onto your device after the sketch file has been built, or you can set flash:true and provide a port in your Lingua Franca file. To flash manually using arduino-cli, enter the subdirectory of src-gen in which the generated sketch file is located and run:

    \n

    arduino-cli upload -p PORT -b FQBN .

    ","headings":[{"value":"Overview","depth":1},{"value":"Prerequisites","depth":2},{"value":"Writing a simple Arduino Program","depth":1},{"value":"Platform Options","depth":2},{"value":"Baud rate of the serial port","depth":3},{"value":"Building","depth":1},{"value":"Flashing","depth":1}],"frontmatter":{"permalink":"/docs/handbook/arduino","title":"Arduino","oneline":"Developing LF Programs on Arduino.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Zephyr","oneline":"Developing LF Programs for Zephyr RTOS.","permalink":"/docs/handbook/zephyr"}}}},"pageContext":{"id":"4-arduino","slug":"/docs/handbook/arduino","repoPath":"/packages/documentation/copy/en/embedded/Arduino.md","nextID":"3f41a520-2093-55ff-a5e7-36ec8f69920a","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/causality-loops/page-data.json b/page-data/docs/handbook/causality-loops/page-data.json index 40119db0a..9bc178855 100644 --- a/page-data/docs/handbook/causality-loops/page-data.json +++ b/page-data/docs/handbook/causality-loops/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/causality-loops","result":{"data":{"markdownRemark":{"id":"2ca49a4a-51a0-5d4e-b91d-fc76ff59751a","excerpt":"Cycles The interconnection pattern for a collection of reactors can form a cycle, but some care is required. Consider the following example: $start(Cycle)$ $end…","html":"

    Cycles

    \n

    The interconnection pattern for a collection of reactors can form a cycle, but some care is required. Consider the following example:

    \n

    $start(Cycle)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(x) {=\n    # ... something here ...\n  =}\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n

    $end(Cycle)$

    \n

    This program yields the following diagram:

    \n\"Lingua\n

    The diagram highlights a causality loop in the program. At each tag, in reactor B, the first reaction has to execute before the second if it is enabled, a precedence indicated with the red dashed arrow. But the first can’t execute until the reaction of A has executed, and that reaction cannot execute until the second reaction B has executed. There is no way to satisfy these requirements, so the tools refuse to generated code.

    \n

    Cycles with Delays

    \n

    One way to break the causality loop and get an executable program is to introduce a logical delay into the loop, as shown below:

    \n

    $start(CycleWithDelay)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(x) {=\n    # ... something here ...\n  =}\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x after 0\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n

    $end(CycleWithDelay)$

    \n\"Lingua\n

    Here, we have used a delay of 0, which results in a delay of one microstep. We could equally well have specified a positive time value.

    \n

    Reaction Order

    \n

    Frequently, a program will have such cycles, but you don’t want a logical delay in the loop. To get a cycle without logical delays, the reactions need to be reordered, as shown below:

    \n

    $start(CycleReordered)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n  reaction(x) {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n

    $end(CycleReordered)$

    \n\"Lingua\n

    There is no longer any causality loop.

    ","headings":[{"value":"Cycles","depth":2},{"value":"Cycles with Delays","depth":2},{"value":"Reaction Order","depth":2}],"frontmatter":{"permalink":"/docs/handbook/causality-loops","title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Methods","oneline":"Methods in Lingua Franca.","permalink":"/docs/handbook/methods"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","permalink":"/docs/handbook/extending-reactors"}}}},"pageContext":{"id":"1-causality-loops","slug":"/docs/handbook/causality-loops","repoPath":"/packages/documentation/copy/en/topics/Causality Loops.md","previousID":"c2fc2a0b-a389-5c48-9cb0-c079bf2e6034","nextID":"d4ade317-2d91-582b-8cca-37743461f687","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/causality-loops","result":{"data":{"markdownRemark":{"id":"2ca49a4a-51a0-5d4e-b91d-fc76ff59751a","excerpt":"Cycles The interconnection pattern for a collection of reactors can form a cycle, but some care is required. Consider the following example: $start(Cycle)$ $end…","html":"

    Cycles

    \n

    The interconnection pattern for a collection of reactors can form a cycle, but some care is required. Consider the following example:

    \n

    $start(Cycle)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(x) {=\n    # ... something here ...\n  =}\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n

    $end(Cycle)$

    \n

    This program yields the following diagram:

    \n\"Lingua\n

    The diagram highlights a causality loop in the program. At each tag, in reactor B, the first reaction has to execute before the second if it is enabled, a precedence indicated with the red dashed arrow. But the first can’t execute until the reaction of A has executed, and that reaction cannot execute until the second reaction B has executed. There is no way to satisfy these requirements, so the tools refuse to generated code.

    \n

    Cycles with Delays

    \n

    One way to break the causality loop and get an executable program is to introduce a logical delay into the loop, as shown below:

    \n

    $start(CycleWithDelay)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(x) {=\n    # ... something here ...\n  =}\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x after 0\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(x) {=\n    // ... something here ...\n  =}\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x after 0;\n  b.y -> a.x;\n}\n
    \n

    $end(CycleWithDelay)$

    \n\"Lingua\n

    Here, we have used a delay of 0, which results in a delay of one microstep. We could equally well have specified a positive time value.

    \n

    Reaction Order

    \n

    Frequently, a program will have such cycles, but you don’t want a logical delay in the loop. To get a cycle without logical delays, the reactions need to be reordered, as shown below:

    \n

    $start(CycleReordered)$

    \n
    target C;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Cpp;\nreactor A {\n  input x:int;\n  output y:int;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:int;\n  output y:int;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target Python;\nreactor A {\n  input x;\n  output y;\n  reaction(x) -> y {=\n    # ... something here ...\n  =}\n}\nreactor B {\n  input x;\n  output y;\n  reaction(startup) -> y {=\n    # ... something here ...\n  =}\n  reaction(x) {=\n    # ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n
    target TypeScript\nreactor A {\n  input x:number\n  output y:number\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:number\n  output y:number\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A()\n  b = new B()\n  a.y -> b.x\n  b.y -> a.x\n}\n
    \n
    target Rust;\nreactor A {\n  input x:u32;\n  output y:u32;\n  reaction(x) -> y {=\n    // ... something here ...\n  =}\n}\nreactor B {\n  input x:u32;\n  output y:u32;\n  reaction(startup) -> y {=\n    // ... something here ...\n  =}\n  reaction(x) {=\n    // ... something here ...\n  =}\n}\nmain reactor {\n  a = new A();\n  b = new B();\n  a.y -> b.x;\n  b.y -> a.x;\n}\n
    \n

    $end(CycleReordered)$

    \n\"Lingua\n

    There is no longer any causality loop.

    ","headings":[{"value":"Cycles","depth":2},{"value":"Cycles with Delays","depth":2},{"value":"Reaction Order","depth":2}],"frontmatter":{"permalink":"/docs/handbook/causality-loops","title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Methods","oneline":"Methods in Lingua Franca.","permalink":"/docs/handbook/methods"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","permalink":"/docs/handbook/extending-reactors"}}}},"pageContext":{"id":"1-causality-loops","slug":"/docs/handbook/causality-loops","repoPath":"/packages/documentation/copy/en/topics/Causality Loops.md","previousID":"c2fc2a0b-a389-5c48-9cb0-c079bf2e6034","nextID":"d4ade317-2d91-582b-8cca-37743461f687","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/code-extension/page-data.json b/page-data/docs/handbook/code-extension/page-data.json index 6f820c716..8f79e9f34 100644 --- a/page-data/docs/handbook/code-extension/page-data.json +++ b/page-data/docs/handbook/code-extension/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/code-extension","result":{"data":{"markdownRemark":{"id":"7ff25653-23b9-5682-8f53-43f7664a8487","excerpt":"The Lingua Franca extension for Visual Studio Code (VS Code) provides syntax-directed editing capability, compilation, and diagram synthesis for Lingua Franca…","html":"

    The Lingua Franca extension for Visual Studio Code (VS Code) provides syntax-directed editing capability, compilation, and diagram synthesis for Lingua Franca programs.

    \n

    Usage

    \n

    Creating a new file

    \n

    To create a new LF file, go to File > New File… and select New Lingua Franca File. When saving the file, save it in a directory called src to make sure that generated code is placed conveniently in an adjacent src-gen directory. For instance, for a file called Foo.lf, the directory structure after building should look something like this:

    \n
    bin/Foo\nsrc/\n└ Foo.lf\nsrc-gen/Foo
    \n

    Rendering diagrams

    \n

    To show the diagram for the currently active Lingua Franca file, click on the diagrams icon at the upper right:

    \n\n \n \"diagrams\n \n

    Compilation

    \n

    To compile the .lf source, open the command palette (Ctrl + Shift + P) and then enter Lingua Franca: Build.

    \n

    Running

    \n

    You can also build and immediately afterwards run your code by opening the command palette (Ctrl + Shift + P) and then entering Lingua Franca: Build and Run.\nRunning the code can also be done from the VS Code terminal by executing the generated file in ./bin.

    \n

    Notes

    \n

    For Python Users

    \n

    Users who edit LF programs with a Python target will benefit the most from Python linting by installing Pylint 2.12.2 or later.

    ","headings":[{"value":"Usage","depth":2},{"value":"Creating a new file","depth":3},{"value":"Rendering diagrams","depth":3},{"value":"Compilation","depth":3},{"value":"Running","depth":3},{"value":"Notes","depth":2},{"value":"For Python Users","depth":3}],"frontmatter":{"permalink":"/docs/handbook/code-extension","title":"Code Extension","oneline":"Visual Studio Code Extension for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Epoch IDE","oneline":"Epoch IDE for Lingua Franca.","permalink":"/docs/handbook/epoch-ide"}}}},"pageContext":{"id":"2-code-extension","slug":"/docs/handbook/code-extension","repoPath":"/packages/documentation/copy/en/tools/Code Extension.md","nextID":"bf08510b-3948-5b7f-bceb-5636ab49d351","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/code-extension","result":{"data":{"markdownRemark":{"id":"7ff25653-23b9-5682-8f53-43f7664a8487","excerpt":"The Lingua Franca extension for Visual Studio Code (VS Code) provides syntax-directed editing capability, compilation, and diagram synthesis for Lingua Franca…","html":"

    The Lingua Franca extension for Visual Studio Code (VS Code) provides syntax-directed editing capability, compilation, and diagram synthesis for Lingua Franca programs.

    \n

    Usage

    \n

    Creating a new file

    \n

    To create a new LF file, go to File > New File… and select New Lingua Franca File. When saving the file, save it in a directory called src to make sure that generated code is placed conveniently in an adjacent src-gen directory. For instance, for a file called Foo.lf, the directory structure after building should look something like this:

    \n
    bin/Foo\nsrc/\n└ Foo.lf\nsrc-gen/Foo
    \n

    Rendering diagrams

    \n

    To show the diagram for the currently active Lingua Franca file, click on the diagrams icon at the upper right:

    \n\n \n \"diagrams\n \n

    Compilation

    \n

    To compile the .lf source, open the command palette (Ctrl + Shift + P) and then enter Lingua Franca: Build.

    \n

    Running

    \n

    You can also build and immediately afterwards run your code by opening the command palette (Ctrl + Shift + P) and then entering Lingua Franca: Build and Run.\nRunning the code can also be done from the VS Code terminal by executing the generated file in ./bin.

    \n

    Notes

    \n

    For Python Users

    \n

    Users who edit LF programs with a Python target will benefit the most from Python linting by installing Pylint 2.12.2 or later.

    ","headings":[{"value":"Usage","depth":2},{"value":"Creating a new file","depth":3},{"value":"Rendering diagrams","depth":3},{"value":"Compilation","depth":3},{"value":"Running","depth":3},{"value":"Notes","depth":2},{"value":"For Python Users","depth":3}],"frontmatter":{"permalink":"/docs/handbook/code-extension","title":"Code Extension","oneline":"Visual Studio Code Extension for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Epoch IDE","oneline":"Epoch IDE for Lingua Franca.","permalink":"/docs/handbook/epoch-ide"}}}},"pageContext":{"id":"2-code-extension","slug":"/docs/handbook/code-extension","repoPath":"/packages/documentation/copy/en/tools/Code Extension.md","nextID":"bf08510b-3948-5b7f-bceb-5636ab49d351","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/command-line-tools/page-data.json b/page-data/docs/handbook/command-line-tools/page-data.json index d907c4ba2..0387445ff 100644 --- a/page-data/docs/handbook/command-line-tools/page-data.json +++ b/page-data/docs/handbook/command-line-tools/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/command-line-tools","result":{"data":{"markdownRemark":{"id":"d1faa7cd-c94d-58d6-8fee-1ad5d56dae5e","excerpt":"Usage Set up a Lingua Franca project by putting your program in a file with the .lf extension,\nsuch as Example.lf and putting that file with a directory called…","html":"

    Usage

    \n

    Set up a Lingua Franca project by putting your program in a file with the .lf extension,\nsuch as Example.lf and putting that file with a directory called src.\nThen compile the program:

    \n
    $ lfc src/Example.lf\n
    \n

    This will create two directories in parallel with the src directory, src-gen and bin. If your target language is a compiled one (like C and C++), then the bin directory should contain an executable that you can run:

    \n
    $ bin/Example\n
    \n

    To see the options that can be given to lfc, get help:

    \n
    $ lfc --help\n
    ","headings":[{"value":"Usage","depth":2}],"frontmatter":{"permalink":"/docs/handbook/command-line-tools","title":"Command Line Tools","oneline":"Command-line tools for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Epoch IDE","oneline":"Epoch IDE for Lingua Franca.","permalink":"/docs/handbook/epoch-ide"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Troubleshooting","oneline":"Troubleshooting page for Lingua Franca tools.","permalink":"/docs/handbook/troubleshooting"}}}},"pageContext":{"id":"2-command-line-tools","slug":"/docs/handbook/command-line-tools","repoPath":"/packages/documentation/copy/en/tools/Command Line Tools.md","previousID":"bf08510b-3948-5b7f-bceb-5636ab49d351","nextID":"0f46feff-0811-5262-9ee0-d7dc527800b9","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/command-line-tools","result":{"data":{"markdownRemark":{"id":"d1faa7cd-c94d-58d6-8fee-1ad5d56dae5e","excerpt":"Usage Set up a Lingua Franca project by putting your program in a file with the .lf extension,\nsuch as Example.lf and putting that file with a directory called…","html":"

    Usage

    \n

    Set up a Lingua Franca project by putting your program in a file with the .lf extension,\nsuch as Example.lf and putting that file with a directory called src.\nThen compile the program:

    \n
    $ lfc src/Example.lf\n
    \n

    This will create two directories in parallel with the src directory, src-gen and bin. If your target language is a compiled one (like C and C++), then the bin directory should contain an executable that you can run:

    \n
    $ bin/Example\n
    \n

    To see the options that can be given to lfc, get help:

    \n
    $ lfc --help\n
    ","headings":[{"value":"Usage","depth":2}],"frontmatter":{"permalink":"/docs/handbook/command-line-tools","title":"Command Line Tools","oneline":"Command-line tools for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Epoch IDE","oneline":"Epoch IDE for Lingua Franca.","permalink":"/docs/handbook/epoch-ide"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Troubleshooting","oneline":"Troubleshooting page for Lingua Franca tools.","permalink":"/docs/handbook/troubleshooting"}}}},"pageContext":{"id":"2-command-line-tools","slug":"/docs/handbook/command-line-tools","repoPath":"/packages/documentation/copy/en/tools/Command Line Tools.md","previousID":"bf08510b-3948-5b7f-bceb-5636ab49d351","nextID":"0f46feff-0811-5262-9ee0-d7dc527800b9","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/composing-reactors/page-data.json b/page-data/docs/handbook/composing-reactors/page-data.json index f3d677b23..b02ec3f33 100644 --- a/page-data/docs/handbook/composing-reactors/page-data.json +++ b/page-data/docs/handbook/composing-reactors/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/composing-reactors","result":{"data":{"markdownRemark":{"id":"2fc4ce5a-22b6-5d88-8f16-224e8c04aa2e","excerpt":"$page-showing-target$ Contained Reactors Reactors can contain instances of other reactors defined in the same file or in an imported file. Assume the Count and…","html":"

    $page-showing-target$

    \n

    Contained Reactors

    \n

    Reactors can contain instances of other reactors defined in the same file or in an imported file. Assume the Count and Scale reactors defined in Parameters and State Variables are stored in files Count.lf and Scale.lf, respectively,\nand that the TestCount reactor from Time and Timers is stored in TestCount.lf. Then the following program composes one instance of each of the three:

    \n

    $start(RegressionTest)$

    \n
    target C {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Cpp {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Python {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target TypeScript {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, numInputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Rust {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n

    $end(RegressionTest)$

    \n

    Diagrams

    \n

    As soon as programs consist of more than one reactor, it becomes particularly useful to reference the diagrams that are automatically created and displayed by the Lingua Franca IDEs. The diagram for the above program is as follows:

    \n\"Lingua\n

    In this diagram, the timer is represented by a clock-like icon, the reactions by chevron shapes, and the $shutdown$ event by a diamond. If there were a $startup$ event in this program, it would appear as a circle.

    \n

    Creating Reactor Instances

    \n

    An instance is created with the syntax:

    \n
        <instance_name> = new <class_name>(<parameters>)\n
    \n

    A bank with several instances can be created in one such statement, as explained in the banks of reactors documentation.

    \n

    The <parameters> argument is a comma-separated list of assignments:

    \n
        <parameter_name> = <value>, ...\n
    \n

    Like the default value for parameters, <value> can be a numeric constant, a string enclosed in quotation marks, a time value such as 10 msec, target-language code enclosed in {= ... =}, or any of the list forms described in Expressions.

    \n

    Connections

    \n

    Connections between ports are specified with the syntax:

    \n
        <source_port_reference> -> <destination_port_reference>\n
    \n

    where the port references are either <instance_name>.<port_name> or just <port_name>, where the latter form is used for connections that cross hierarchical boundaries, as illustrated in the next section.

    \n

    On the left and right of a connection statement, you can put a comma-separated list. For example, the above pair of connections can be written,

    \n
        c.y, s.y -> s.x, t.x\n
    \n

    A constraint is that the total number of channels on the left match the total number on the right.\nIn addition, some targets require the types of all the ports to be the same.

    \n

    A destination port (on the right) can only be connected to a single source port (on the left). However, a source port may be connected to multiple destinations, as in the following example:

    \n
    \n
    reactor A {\n  output y\n}\nreactor B {\n  input x\n}\nmain reactor {\n  a = new A()\n  b1 = new B()\n  b2 = new B()\n  a.y -> b1.x\n  a.y -> b2.x\n}\n
    \n
    \n
    \n
    reactor A {\n  output y:int\n}\nreactor B {\n  input x:int\n}\nmain reactor {\n  a = new A()\n  b1 = new B()\n  b2 = new B()\n  a.y -> b1.x\n  a.y -> b2.x\n}\n
    \n
    \n\"Lingua\n

    Lingua Franca provides a convenient shortcut for such multicast connections, where the above two lines can be replaced by one as follows:

    \n
      (a.y)+ -> b1.x, b2.x\n
    \n

    The enclosing ( ... )+ means to repeat the enclosed comma-separated list of sources however many times is needed to provide inputs to all the sinks on the right of the connection ->.

    \n

    Import Statement

    \n

    An import statement has the form:

    \n
      import <classname> as <alias> from "<path>"\n
    \n

    where <classname> and <alias> can be a comma-separated list to import multiple reactors from the same file. The <path> specifies another .lf file relative to the location of the current file. The as <alias> portion is optional and specifies alternative class names to use in the $new$ statements.

    \n

    Hierarchy

    \n

    Reactors can be composed in arbitrarily deep hierarchies. For example, the following program combines the Count and Scale reactors within on Container:

    \n

    $start(Hierarchy)$

    \n
    target C\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: int = 2) {\n  output y: int\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target Cpp\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: int(2)) {\n  output y: int\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target Python\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride=2) {\n  output y\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target TypeScript\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: number = 2) {\n  output y: number\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, numInputs=11)\n  c.y -> t.x\n}\n
    \n
    target Rust\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: u32 = 2) {\n  output y: u32\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n

    $end(Hierarchy)$

    \n\"Lingua\n

    The Container has a parameter named stride, whose value is passed to the factor parameter of the Scale reactor. The line

    \n
      s.y -> y;\n
    \n

    establishes a connection across levels of the hierarchy. This propagates the output of a contained reactor to the output of the container. A similar notation may be used to propagate the input of a container to the input of a contained reactor,

    \n
      x -> s.x;\n
    \n

    Connections with Logical Delays

    \n

    Connections may include a logical delay using the $after$ keyword, as follows:

    \n
      <source_port_reference> -> <destination_port_reference> after <time_value>\n
    \n

    where <time_value> can be any of the forms described in Expressions.

    \n

    The $after$ keyword specifies that the logical time of the event delivered to the destination port will be larger than the logical time of the reaction that wrote to source port. The time value is required to be non-negative, but it can be zero, in which case the input event at the receiving end will be one microstep later than the event that triggered it.

    \n

    Physical Connections

    \n

    A subtle and rarely used variant of the -> connection is a physical connection, denoted ~>. For example:

    \n
    main reactor {\n  a = new A();\n  b = new B();\n  a.y ~> b.x;\n}\n
    \n

    This is rendered in by the diagram synthesizer as follows:

    \n\"Lingua\n

    In such a connection, the logical time at the recipient is derived from the local physical clock rather than being equal to the logical time at the sender. The physical time will always exceed the logical time of the sender (unless fast is set to true), so this type of connection incurs a nondeterministic positive logical time delay. Physical connections are useful sometimes in Distributed-Execution in situations where the nondeterministic logical delay is tolerable. Such connections are more efficient because timestamps need not be transmitted and messages do not need to flow through through a centralized coordinator (if a centralized coordinator is being used).

    ","headings":[{"value":"Contained Reactors","depth":2},{"value":"Diagrams","depth":2},{"value":"Creating Reactor Instances","depth":2},{"value":"Connections","depth":2},{"value":"Import Statement","depth":2},{"value":"Hierarchy","depth":2},{"value":"Connections with Logical Delays","depth":2},{"value":"Physical Connections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/composing-reactors","title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","permalink":"/docs/handbook/time-and-timers"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Reactions","oneline":"Reactions in Lingua Franca.","permalink":"/docs/handbook/reactions"}}}},"pageContext":{"id":"1-composing-reactors","slug":"/docs/handbook/composing-reactors","repoPath":"/packages/documentation/copy/en/topics/Composing Reactors.md","previousID":"bba867ed-95b9-5017-b4f1-350e621e99da","nextID":"25ac2513-8979-52dd-9176-b0db61f55dc9","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/composing-reactors","result":{"data":{"markdownRemark":{"id":"2fc4ce5a-22b6-5d88-8f16-224e8c04aa2e","excerpt":"$page-showing-target$ Contained Reactors Reactors can contain instances of other reactors defined in the same file or in an imported file. Assume the Count and…","html":"

    $page-showing-target$

    \n

    Contained Reactors

    \n

    Reactors can contain instances of other reactors defined in the same file or in an imported file. Assume the Count and Scale reactors defined in Parameters and State Variables are stored in files Count.lf and Scale.lf, respectively,\nand that the TestCount reactor from Time and Timers is stored in TestCount.lf. Then the following program composes one instance of each of the three:

    \n

    $start(RegressionTest)$

    \n
    target C {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Cpp {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Python {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target TypeScript {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, numInputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n
    target Rust {\n  timeout: 1 sec,\n  fast: true\n}\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nmain reactor RegressionTest {\n  c = new Count()\n  s = new Scale(factor=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> s.x\n  s.y -> t.x\n}\n
    \n

    $end(RegressionTest)$

    \n

    Diagrams

    \n

    As soon as programs consist of more than one reactor, it becomes particularly useful to reference the diagrams that are automatically created and displayed by the Lingua Franca IDEs. The diagram for the above program is as follows:

    \n\"Lingua\n

    In this diagram, the timer is represented by a clock-like icon, the reactions by chevron shapes, and the $shutdown$ event by a diamond. If there were a $startup$ event in this program, it would appear as a circle.

    \n

    Creating Reactor Instances

    \n

    An instance is created with the syntax:

    \n
        <instance_name> = new <class_name>(<parameters>)\n
    \n

    A bank with several instances can be created in one such statement, as explained in the banks of reactors documentation.

    \n

    The <parameters> argument is a comma-separated list of assignments:

    \n
        <parameter_name> = <value>, ...\n
    \n

    Like the default value for parameters, <value> can be a numeric constant, a string enclosed in quotation marks, a time value such as 10 msec, target-language code enclosed in {= ... =}, or any of the list forms described in Expressions.

    \n

    Connections

    \n

    Connections between ports are specified with the syntax:

    \n
        <source_port_reference> -> <destination_port_reference>\n
    \n

    where the port references are either <instance_name>.<port_name> or just <port_name>, where the latter form is used for connections that cross hierarchical boundaries, as illustrated in the next section.

    \n

    On the left and right of a connection statement, you can put a comma-separated list. For example, the above pair of connections can be written,

    \n
        c.y, s.y -> s.x, t.x\n
    \n

    A constraint is that the total number of channels on the left match the total number on the right.\nIn addition, some targets require the types of all the ports to be the same.

    \n

    A destination port (on the right) can only be connected to a single source port (on the left). However, a source port may be connected to multiple destinations, as in the following example:

    \n
    \n
    reactor A {\n  output y\n}\nreactor B {\n  input x\n}\nmain reactor {\n  a = new A()\n  b1 = new B()\n  b2 = new B()\n  a.y -> b1.x\n  a.y -> b2.x\n}\n
    \n
    \n
    \n
    reactor A {\n  output y:int\n}\nreactor B {\n  input x:int\n}\nmain reactor {\n  a = new A()\n  b1 = new B()\n  b2 = new B()\n  a.y -> b1.x\n  a.y -> b2.x\n}\n
    \n
    \n\"Lingua\n

    Lingua Franca provides a convenient shortcut for such multicast connections, where the above two lines can be replaced by one as follows:

    \n
      (a.y)+ -> b1.x, b2.x\n
    \n

    The enclosing ( ... )+ means to repeat the enclosed comma-separated list of sources however many times is needed to provide inputs to all the sinks on the right of the connection ->.

    \n

    Import Statement

    \n

    An import statement has the form:

    \n
      import <classname> as <alias> from "<path>"\n
    \n

    where <classname> and <alias> can be a comma-separated list to import multiple reactors from the same file. The <path> specifies another .lf file relative to the location of the current file. The as <alias> portion is optional and specifies alternative class names to use in the $new$ statements.

    \n

    Hierarchy

    \n

    Reactors can be composed in arbitrarily deep hierarchies. For example, the following program combines the Count and Scale reactors within on Container:

    \n

    $start(Hierarchy)$

    \n
    target C\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: int = 2) {\n  output y: int\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target Cpp\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: int(2)) {\n  output y: int\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target Python\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride=2) {\n  output y\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n
    target TypeScript\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: number = 2) {\n  output y: number\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, numInputs=11)\n  c.y -> t.x\n}\n
    \n
    target Rust\nimport Count from "Count.lf"\nimport Scale from "Scale.lf"\nimport TestCount from "TestCount.lf"\nreactor Container(stride: u32 = 2) {\n  output y: u32\n  c = new Count()\n  s = new Scale(factor=stride)\n  c.y -> s.x\n  s.y -> y\n}\nmain reactor Hierarchy {\n  c = new Container(stride=4)\n  t = new TestCount(stride=4, num_inputs=11)\n  c.y -> t.x\n}\n
    \n

    $end(Hierarchy)$

    \n\"Lingua\n

    The Container has a parameter named stride, whose value is passed to the factor parameter of the Scale reactor. The line

    \n
      s.y -> y;\n
    \n

    establishes a connection across levels of the hierarchy. This propagates the output of a contained reactor to the output of the container. A similar notation may be used to propagate the input of a container to the input of a contained reactor,

    \n
      x -> s.x;\n
    \n

    Connections with Logical Delays

    \n

    Connections may include a logical delay using the $after$ keyword, as follows:

    \n
      <source_port_reference> -> <destination_port_reference> after <time_value>\n
    \n

    where <time_value> can be any of the forms described in Expressions.

    \n

    The $after$ keyword specifies that the logical time of the event delivered to the destination port will be larger than the logical time of the reaction that wrote to source port. The time value is required to be non-negative, but it can be zero, in which case the input event at the receiving end will be one microstep later than the event that triggered it.

    \n

    Physical Connections

    \n

    A subtle and rarely used variant of the -> connection is a physical connection, denoted ~>. For example:

    \n
    main reactor {\n  a = new A();\n  b = new B();\n  a.y ~> b.x;\n}\n
    \n

    This is rendered in by the diagram synthesizer as follows:

    \n\"Lingua\n

    In such a connection, the logical time at the recipient is derived from the local physical clock rather than being equal to the logical time at the sender. The physical time will always exceed the logical time of the sender (unless fast is set to true), so this type of connection incurs a nondeterministic positive logical time delay. Physical connections are useful sometimes in Distributed-Execution in situations where the nondeterministic logical delay is tolerable. Such connections are more efficient because timestamps need not be transmitted and messages do not need to flow through through a centralized coordinator (if a centralized coordinator is being used).

    ","headings":[{"value":"Contained Reactors","depth":2},{"value":"Diagrams","depth":2},{"value":"Creating Reactor Instances","depth":2},{"value":"Connections","depth":2},{"value":"Import Statement","depth":2},{"value":"Hierarchy","depth":2},{"value":"Connections with Logical Delays","depth":2},{"value":"Physical Connections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/composing-reactors","title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","permalink":"/docs/handbook/time-and-timers"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Reactions","oneline":"Reactions in Lingua Franca.","permalink":"/docs/handbook/reactions"}}}},"pageContext":{"id":"1-composing-reactors","slug":"/docs/handbook/composing-reactors","repoPath":"/packages/documentation/copy/en/topics/Composing Reactors.md","previousID":"bba867ed-95b9-5017-b4f1-350e621e99da","nextID":"25ac2513-8979-52dd-9176-b0db61f55dc9","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/containerized-execution/page-data.json b/page-data/docs/handbook/containerized-execution/page-data.json index 895fe817f..f7c7816a1 100644 --- a/page-data/docs/handbook/containerized-execution/page-data.json +++ b/page-data/docs/handbook/containerized-execution/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/containerized-execution","result":{"data":{"markdownRemark":{"id":"2548d58b-6c82-5c27-83b8-e05a47521264","excerpt":"For the C target at least, the Lingua Franca code generator is able to generate a Dockerfile when it generates the C source files. To enable this, include the…","html":"

    For the C target at least, the Lingua Franca code generator is able to generate a Dockerfile when it generates the C source files. To enable this, include the docker property in your target specification, as follows:

    \n
    target C {\n    docker: true\n};\n
    \n

    The generated Docker file has the same name as the LF file except that the extension is .Dockerfile and will be put in the src-gen directory. You can also specify options. Currently, only the base image (FROM) can be customized, but this will be extended to allow further customization is the future. To customize the Docker file, instead of just true above, which gives default options, specify the options as in the following example:

    \n
    target C {\n    docker: {FROM: "alpine:latest"}\n};\n
    \n

    This specifies that the base image is the latest version of alpine, a very small Linux. In fact, alpine:latest is the default value for this option, so you only need to specify this option if you need something other than alpine:latest.

    \n

    How to use this depends on whether your application is federated. You will need to install Docker if you haven’t already in order to use this.

    \n

    Unfederated Execution

    \n

    Using docker build and docker run

    \n

    Suppose your LF source file is Foo.lf. When you run lfc or use the IDE to generate code, a file called Foo.Dockerfile will appear in the src_gen directory. You can use this file to build a Docker image as follows. First, make sure you are in the same directory as the source file. Then issue the command:

    \n
       docker build -t foo -f src-gen/Foo.Dockerfile .\n
    \n

    This will create a Docker image with tag foo. The tag is required to be all lower-case letters. By convention, we advise using the LF source file name, converted to lower case.

    \n

    You can then use this tag to run the image in a container:

    \n
        docker run -t --rm foo\n
    \n

    The -t option creates a pseudo terminal, which is necessary for you to see any output produced by your program to stdout. If your program also reads from stdin, then you will need to give the -i option as well, or combine the two as it.

    \n

    The --rm option is important. This removes the container upon completion of the run. If you omit this option, the container will continue to exist even after your program has terminated. You can alternatively remove the container after the run using docker rm.

    \n

    If you wish for your program to run in the background, give a -d option as well (for “detached”). In this case, you will not see any output from your run.

    \n

    The above run command can include any supported command-line arguments to the LF program. For example, to specify a logical timeout, you can do this:

    \n
        docker run -t --rm foo --timeout 20 sec\n
    \n

    Using docker compose up

    \n

    When you use lfc to compile Foo.lf, a file called docker-compose.yml will also appear in the same directory where Foo.Dockerfile is located. cd to that directory. Then, use docker compose up to automatically build and run the container. Once the container finishes execution, use docker compose down in the same directory where docker-compose.yml is located to remove the container.

    \n

    Federated Execution

    \n

    Using docker build and docker run

    \n

    For a federated Lingua Franca program, one Dockerfile is created for each federate plus an additional one for the RTI. The Dockerfile for the RTI will be generated at src-gen/RTI. You will need to run docker build for each of these. For example, to build the image for the RTI, you can do this:

    \n
    docker build -t distributedcountcontainerized_rti -f src-gen/DistributedCountContainerized_RTI.Dockerfile .\n
    \n

    This is for the DistributedCountContainerized.lf, a federated that automatically runs in multiple Docker containers (one for the RTI and one for each federate) in continuous integration.

    \n

    Now, there are several options for how to proceed. One is to create a named network on which to run your federation. For example, to create a network named lf, do this:

    \n
        docker network create lf\n
    \n

    You can then run the RTI on this network and assign the RTI a name that the federates can use to find the RTI:

    \n
        docker run -t --rm --network lf --name distributedcount-rti distributedcount_rti\n
    \n

    Here, the assigned name is not quite the same as the tag that was specified when building the image (the last argument is the tag of the image to run in a container) because a host name is not allowed to have an underscore in it.

    \n

    Currently, you will also have to specify this host name in the LF source file so that the federates know where to find the RTI. E.g., in DistributedCount.lf, change the end of the file to read as follows:

    \n
    federated reactor DistributedCount at distributedcount-rti {\n    c = new Count();\n    p = new Print();\n    c.out -> p.in after 200 msec;\n}\n
    \n

    Notice the at distributedcount-rti, which must match the name you use when running the RTI. FIXME: We should find a way to make this more automatic!

    \n

    In two other terminals, you can now run the other federates on the same network:

    \n
    docker run -t --rm --network lf distributedcount_c\n
    \n

    and

    \n
    docker run -t --rm --network lf distributedcount_p\n
    \n

    Using docker compose up

    \n

    For a federated Lingua Franca program, once you use lfc to compile Foo.lf, on top of the docker-compose.yml for the reactors, an additional docker-compose.yml will be generated for the RTI and placed in src-gen/RTI.

    \n

    To run the federated program, open two terminals. In the first terminal, go to src-gen/RTI and use docker compose up to build and run the containerized RTI. Wait until the RTI is booted up. Then, open a second terminal and cd to the top level folder of the program (this is the folder that contains one folder for each of the federated reactors). You should see a docker-compose.yml there. Run docker compose up to build and run the containers.

    \n

    Once the program finished executing, run docker compose down in both the folder that contains the docker-compose.yml for the RTI as well as the folder that contains the docker-compose.yml for the reactors to remove the containers.

    ","headings":[{"value":"Unfederated Execution","depth":2},{"value":"Using docker build and docker run","depth":3},{"value":"Using docker compose up","depth":3},{"value":"Federated Execution","depth":2},{"value":"Using docker build and docker run","depth":3},{"value":"Using docker compose up","depth":3}],"frontmatter":{"permalink":"/docs/handbook/containerized-execution","title":"Containerized Execution","oneline":"Containerized Execution using Docker","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Tracing","oneline":"Tracing (preliminary)","permalink":"/docs/handbook/tracing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Security","oneline":"Secure Federated Execution","permalink":"/docs/handbook/security"}}}},"pageContext":{"id":"3-containerized-execution","slug":"/docs/handbook/containerized-execution","repoPath":"/packages/documentation/copy/en/reference/Containerized Execution.md","previousID":"3f4000b5-1133-5c34-807d-29c05884f149","nextID":"b02df86d-9ef8-5f8e-ba32-437934fba499","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/containerized-execution","result":{"data":{"markdownRemark":{"id":"2548d58b-6c82-5c27-83b8-e05a47521264","excerpt":"For the C target at least, the Lingua Franca code generator is able to generate a Dockerfile when it generates the C source files. To enable this, include the…","html":"

    For the C target at least, the Lingua Franca code generator is able to generate a Dockerfile when it generates the C source files. To enable this, include the docker property in your target specification, as follows:

    \n
    target C {\n    docker: true\n};\n
    \n

    The generated Docker file has the same name as the LF file except that the extension is .Dockerfile and will be put in the src-gen directory. You can also specify options. Currently, only the base image (FROM) can be customized, but this will be extended to allow further customization is the future. To customize the Docker file, instead of just true above, which gives default options, specify the options as in the following example:

    \n
    target C {\n    docker: {FROM: "alpine:latest"}\n};\n
    \n

    This specifies that the base image is the latest version of alpine, a very small Linux. In fact, alpine:latest is the default value for this option, so you only need to specify this option if you need something other than alpine:latest.

    \n

    How to use this depends on whether your application is federated. You will need to install Docker if you haven’t already in order to use this.

    \n

    Unfederated Execution

    \n

    Using docker build and docker run

    \n

    Suppose your LF source file is src/Foo.lf. When you run lfc or use the IDE to generate code, a file called Dockerfile will appear in the src_gen/Foo directory, see Structure of an LF project for more info. You can build a Docker image as follows. First, navigate into the directory where Dockerfile is located. Then issue the command:

    \n
       docker build -t foo .\n
    \n

    This will create a Docker image with tag foo. The tag is required to be all lower-case letters. By convention, we advise using the LF source file name, converted to lower case.

    \n

    You can then use this tag to run the image in a container:

    \n
        docker run -t --rm foo\n
    \n

    The -t option creates a pseudo terminal, which is necessary for you to see any output produced by your program to stdout. If your program also reads from stdin, then you will need to give the -i option as well, or combine the two as it.

    \n

    The --rm option is important. This removes the container upon completion of the run. If you omit this option, the container will continue to exist even after your program has terminated. You can alternatively remove the container after the run using docker rm.

    \n

    If you wish for your program to run in the background, give a -d option as well (for “detached”). In this case, you will not see any output from your run.

    \n

    The above run command can include any supported command-line arguments to the LF program. For example, to specify a logical timeout, you can do this:

    \n
        docker run -t --rm foo --timeout 20 sec\n
    \n

    Using docker compose up

    \n

    When you use lfc to compile Foo.lf, a file called docker-compose.yml will also appear in the same directory where Foo.Dockerfile is located. cd to that directory. Then, use docker compose up to automatically build and run the container. Once the container finishes execution, use docker compose down in the same directory where docker-compose.yml is located to remove the container.

    \n

    Federated Execution

    \n

    Using docker build and docker run

    \n

    For a federated Lingua Franca program, one Dockerfile is created for each federate plus an additional one for the RTI. The Dockerfile for the RTI will be generated at src-gen/RTI. You will need to run docker build for each of these. For example, to build the image for the RTI, you can do this:

    \n
    docker build -t distributedcountcontainerized_rti -f src-gen/DistributedCountContainerized_RTI.Dockerfile .\n
    \n

    This is for the DistributedCountContainerized.lf, a federated that automatically runs in multiple Docker containers (one for the RTI and one for each federate) in continuous integration.

    \n

    Now, there are several options for how to proceed. One is to create a named network on which to run your federation. For example, to create a network named lf, do this:

    \n
        docker network create lf\n
    \n

    You can then run the RTI on this network and assign the RTI a name that the federates can use to find the RTI:

    \n
        docker run -t --rm --network lf --name distributedcount-rti distributedcount_rti\n
    \n

    Here, the assigned name is not quite the same as the tag that was specified when building the image (the last argument is the tag of the image to run in a container) because a host name is not allowed to have an underscore in it.

    \n

    Currently, you will also have to specify this host name in the LF source file so that the federates know where to find the RTI. E.g., in DistributedCount.lf, change the end of the file to read as follows:

    \n
    federated reactor DistributedCount at distributedcount-rti {\n    c = new Count();\n    p = new Print();\n    c.out -> p.in after 200 msec;\n}\n
    \n

    Notice the at distributedcount-rti, which must match the name you use when running the RTI. FIXME: We should find a way to make this more automatic!

    \n

    In two other terminals, you can now run the other federates on the same network:

    \n
    docker run -t --rm --network lf distributedcount_c\n
    \n

    and

    \n
    docker run -t --rm --network lf distributedcount_p\n
    \n

    Using docker compose up

    \n

    For a federated Lingua Franca program, once you use lfc to compile Foo.lf, on top of the docker-compose.yml for the reactors, an additional docker-compose.yml will be generated for the RTI and placed in src-gen/RTI.

    \n

    To run the federated program, open two terminals. In the first terminal, go to src-gen/RTI and use docker compose up to build and run the containerized RTI. Wait until the RTI is booted up. Then, open a second terminal and cd to the top level folder of the program (this is the folder that contains one folder for each of the federated reactors). You should see a docker-compose.yml there. Run docker compose up to build and run the containers.

    \n

    Once the program finished executing, run docker compose down in both the folder that contains the docker-compose.yml for the RTI as well as the folder that contains the docker-compose.yml for the reactors to remove the containers.

    ","headings":[{"value":"Unfederated Execution","depth":2},{"value":"Using docker build and docker run","depth":3},{"value":"Using docker compose up","depth":3},{"value":"Federated Execution","depth":2},{"value":"Using docker build and docker run","depth":3},{"value":"Using docker compose up","depth":3}],"frontmatter":{"permalink":"/docs/handbook/containerized-execution","title":"Containerized Execution","oneline":"Containerized Execution using Docker","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Tracing","oneline":"Tracing (preliminary)","permalink":"/docs/handbook/tracing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Security","oneline":"Secure Federated Execution","permalink":"/docs/handbook/security"}}}},"pageContext":{"id":"3-containerized-execution","slug":"/docs/handbook/containerized-execution","repoPath":"/packages/documentation/copy/en/reference/Containerized Execution.md","previousID":"3f4000b5-1133-5c34-807d-29c05884f149","nextID":"b02df86d-9ef8-5f8e-ba32-437934fba499","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/contributing/page-data.json b/page-data/docs/handbook/contributing/page-data.json index a95951b8e..1a69d959e 100644 --- a/page-data/docs/handbook/contributing/page-data.json +++ b/page-data/docs/handbook/contributing/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/contributing","result":{"data":{"markdownRemark":{"id":"cb27d929-c4ae-5578-9f34-e2f268594fd2","excerpt":"The preferred way to contribute to Lingua Franca is to issue pull requests through GitHub.\nSee the Contributing document for more details.","html":"

    The preferred way to contribute to Lingua Franca is to issue pull requests through GitHub.\nSee the Contributing document for more details.

    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/contributing","title":"Contributing","oneline":"Contribute to Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","permalink":"/docs/handbook/developer-setup"}}}},"pageContext":{"id":"5-contributing","slug":"/docs/handbook/contributing","repoPath":"/packages/documentation/copy/en/developer/Contributing.md","nextID":"81b7347d-2806-53c1-91ad-c7b12d062d3c","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/contributing","result":{"data":{"markdownRemark":{"id":"cb27d929-c4ae-5578-9f34-e2f268594fd2","excerpt":"The preferred way to contribute to Lingua Franca is to issue pull requests through GitHub.\nSee the Contributing document for more details.","html":"

    The preferred way to contribute to Lingua Franca is to issue pull requests through GitHub.\nSee the Contributing document for more details.

    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/contributing","title":"Contributing","oneline":"Contribute to Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","permalink":"/docs/handbook/developer-setup"}}}},"pageContext":{"id":"5-contributing","slug":"/docs/handbook/contributing","repoPath":"/packages/documentation/copy/en/developer/Contributing.md","nextID":"81b7347d-2806-53c1-91ad-c7b12d062d3c","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/deadlines/page-data.json b/page-data/docs/handbook/deadlines/page-data.json index d5a441096..380b30df5 100644 --- a/page-data/docs/handbook/deadlines/page-data.json +++ b/page-data/docs/handbook/deadlines/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/deadlines","result":{"data":{"markdownRemark":{"id":"f38ee330-34ee-5bea-906f-ebea05b6c4bd","excerpt":"$page-showing-target$ Lingua Franca includes a notion of a deadline, which is a constraint on the relation between logical time and physical time. Specifically…","html":"

    $page-showing-target$

    \n

    Lingua Franca includes a notion of a deadline, which is a constraint on the relation between logical time and physical time. Specifically, a program may specify that the invocation of a reaction must occur within some physical time interval of the logical time of the message. If a reaction is invoked at logical time 12 noon, for example, and the reaction has a deadline of one hour, then the reaction is required to be invoked before the physical-time clock of the execution platform reaches 1 PM. If the deadline is violated, then the specified deadline handler is invoked instead of the reaction.

    \n

    Purposes for Deadlines

    \n

    A deadline in an LF program serves two purposes. First, it can guide scheduling in that a scheduler may prioritize reactions with deadlines over those without or those with longer deadlines. For this purpose, if a reaction has a deadline, then all upstream reactions on which it depends (without logical delay) inherit its deadline. Hence, those upstream reactions will also be given higher priority.

    \n

    Second, the deadline mechanism provides a fault handler, a section of code to invoke when the deadline requirement is violated. Because invocation of the fault handler depends on factors beyond the control of the LF program, an LF program with deadlines becomes nondeterministic. The behavior of the program depends on the exact timing of the execution.

    \n

    There remains the question of when the fault handler should be invoked. By default, deadlines in LF are lazy, meaning that the fault handler is invoked at the logical time of the event triggering the reaction whose deadline is missed. Specifically, the possible violation of a deadline is not checked until the reaction with the deadline is ready to execute. Only then is the determination made whether to invoke the regular reaction or the fault handler.

    \n

    Deadline Specification

    \n

    A deadline is specified as follows:

    \n

    $start(Deadline)$

    \n
    target C;\nreactor Deadline {\n  input x:int;\n  output d:int; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    printf("Normal reaction.\\n");\n  =} deadline(10 msec) {=\n    printf("Deadline violation detected.\\n");\n    lf_set(d, x->value);\n  =}\n}\n
    \n
    target Cpp;\nreactor Deadline {\n  input x:int;\n  output d:int; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    std::cout << "Normal reaction." << std::endl;\n  =} deadline(10ms) {=\n    std::cout << "Deadline violation detected." << std::endl;\n    d.set(*x.get());\n  =}\n}\n
    \n
    target Python;\nreactor Deadline {\n  input x;\n  output d; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    print("Normal reaction.")\n  =} deadline(10 msec) {=\n    print("Deadline violation detected.")\n    d.set(x.value)\n  =}\n}\n
    \n
    target TypeScript\nreactor Deadline {\n  input x:number\n  output d:number // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    console.log("Normal reaction.")\n  =} deadline(10 msec) {=\n    console.log("Deadline violation detected.")\n    d = x\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Deadline.lf\n
    \n

    $end(Deadline)$

    \n

    This reactor specifies a deadline of 10 milliseconds (this can be a parameter of the reactor). If the reaction to x is triggered later in physical time than 10 msec past the timestamp of x, then the second body of code is executed instead of the first. That second body of code has access to anything the first body of code has access to, including the input x and the output d. The output can be used to notify the rest of the system that a deadline violation occurred. This reactor can be tested as follows:

    \n

    $start(DeadlineTest)$

    \n
    target C\nimport Deadline from "Deadline.lf"\npreamble {=\n  #include "platform.h"\n=}\nmain reactor {\n  logical action a\n  d = new Deadline()\n  reaction(startup) -> d.x, a {=\n    lf_set(d.x, 0);\n    lf_schedule(a, 0);\n  =}\n  reaction(a) -> d.x {=\n    lf_set(d.x, 0);\n    lf_sleep(MSEC(20));\n  =}\n  reaction(d.d) {=\n    printf("Deadline reactor produced an output.\\n");\n  =}\n}\n
    \n
    target Cpp;\nimport Deadline from "Deadline.lf";\nmain reactor {\n  logical action a;\n  d = new Deadline();\n  reaction(startup) -> d.x, a {=\n    d.x.set(0);\n    a.schedule(0ms);\n  =}\n  reaction(a) -> d.x {=\n    d.x.set(0);\n    std::this_thread::sleep_for(20ms);\n  =}\n  reaction(d.d) {=\n    std::cout << "Deadline reactor produced an output." << std::endl;\n  =}\n}\n
    \n
    target Python;\nimport Deadline from "Deadline.lf";\npreamble {= import time =}\nmain reactor {\n  logical action a;\n  d = new Deadline();\n  reaction(startup) -> d.x, a {=\n    d.x.set(0)\n    a.schedule(0)\n  =}\n  reaction(a) -> d.x {=\n    d.x.set(0)\n    time.sleep(0.02)\n  =}\n  reaction(d.d) {=\n    print("Deadline reactor produced an output.")\n  =}\n}\n
    \n
    target TypeScript\nimport Deadline from "Deadline.lf"\nmain reactor {\n  logical action a\n  d = new Deadline()\n  reaction(startup) -> d.x, a {=\n    d.x = 0\n    actions.a.schedule(TimeValue.zero(), null)\n  =}\n  reaction(a) -> d.x {=\n    d.x = 0\n    for (const later = util.getCurrentPhysicalTime().add(TimeValue.msecs(20));\n      util.getCurrentPhysicalTime() < later;) {\n      // Take time...\n    }\n  =}\n  reaction(d.d) {=\n    console.log("Deadline reactor produced an output.")\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/DeadlineTest.lf\n
    \n

    $end(DeadlineTest)$

    \n\"Lingua\n

    Running this program will result in the following output:

    \n
    Normal reaction.\nDeadline violation detected.\nDeadline reactor produced an output.
    \n

    The first reaction of the Deadline reactor does not violate the deadline, but the second does. Notice that the sleep in the $main$ reactor occurs after setting the output, but because of the deterministic semantics of LF, this does not matter. The actual value of an output cannot be known until every reaction that sets that output completes its execution. Since this reaction takes at least 20 msec to complete, the deadline is assured of being violated.

    \n

    Notice that the deadline is annotated in the diagram with a small clock symbol.

    \n
    \n

    Notice that the deadline violation here is caused by an invocation of lf_sleep, defined in \"platform.h\" (see Libraries Available to Programmers).\nIt is not generally advisable for a reaction to sleep because this can block other reactions from executing.\nBut this is exactly what we are trying to accomplish here in order to force a deadline to be violated.

    \n
    \n

    Deadline Violations During Execution

    \n

    Whether a deadline violation occurs is checked only before invoking the reaction with a deadline. What if the reaction itself runs for long enough that the deadline gets violated during the reaction execution? For this purpose, a target-language function is provided to check whether a deadline is violated during execution of a reaction with a deadline.

    \n
    \n

    NOTE: As of this writing, this function is only implemented in the C target.

    \n
    \n

    Consider this example:

    \n

    $start(CheckDeadline)$

    \n
    target C;\nreactor Count {\n  output out:int;\n  reaction(startup) -> out {=\n    int count = 0;\n    while (!lf_check_deadline(self, true)) {\n      count++;\n    }\n    lf_set(out, count);\n  =} deadline (3 msec) {=\n    printf("Stopped counting.\\n");\n  =}\n}\nmain reactor {\n  c = new Count();\n  reaction(c.out) {=\n    printf("Counted to %d\\n", c.out->value);\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/py/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/CheckDeadline.lf\n
    \n

    $end(CheckDeadline)$

    \n
    \n

    The Count reactor has a single reaction with a deadline of 3 msec.\nIf the deadline is not already violated when this reaction becomes enabled (at startup), then the reaction begins executing a loop. In each iteration of the loop, it calls lf_check_deadline(self, true), which returns true if the deadline has been violated and false otherwise. Hence, this reaction will increment the count variable as many times as possible before the deadline is violated and, at\nthat point, will exit the loop and produce on the output the count. Running this program will produce something like this:

    \n
    Stopped counting.\nCounted to 20257
    \n

    This is a (rather trivial) example of an anytime computation. Such computations proceed to improve results until time runs out and then produce the most improved result.

    \n

    The arguments to the lf_check_deadline are the self struct and a boolean that indicates whether the deadline violation handler should be invoked upon detecting a deadline violation. Because the argument is true above, the handler is invoked and Stopped counting is printed.

    \n
    ","headings":[{"value":"Purposes for Deadlines","depth":2},{"value":"Deadline Specification","depth":2},{"value":"Deadline Violations During Execution","depth":2}],"frontmatter":{"permalink":"/docs/handbook/deadlines","title":"Deadlines","oneline":"Deadlines in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Modal Reactors","oneline":"Modal Reactors","permalink":"/docs/handbook/modal-models"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","permalink":"/docs/handbook/multiports-and-banks"}}}},"pageContext":{"id":"1-deadlines","slug":"/docs/handbook/deadlines","repoPath":"/packages/documentation/copy/en/topics/Deadlines.md","previousID":"ddeb2577-9554-5362-9ed2-abba8f412fc1","nextID":"9ff63bbf-2bdf-553e-a96d-52355866ec94","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/deadlines","result":{"data":{"markdownRemark":{"id":"f38ee330-34ee-5bea-906f-ebea05b6c4bd","excerpt":"$page-showing-target$ Lingua Franca includes a notion of a deadline, which is a constraint on the relation between logical time and physical time. Specifically…","html":"

    $page-showing-target$

    \n

    Lingua Franca includes a notion of a deadline, which is a constraint on the relation between logical time and physical time. Specifically, a program may specify that the invocation of a reaction must occur within some physical time interval of the logical time of the message. If a reaction is invoked at logical time 12 noon, for example, and the reaction has a deadline of one hour, then the reaction is required to be invoked before the physical-time clock of the execution platform reaches 1 PM. If the deadline is violated, then the specified deadline handler is invoked instead of the reaction.

    \n

    Purposes for Deadlines

    \n

    A deadline in an LF program serves two purposes. First, it can guide scheduling in that a scheduler may prioritize reactions with deadlines over those without or those with longer deadlines. For this purpose, if a reaction has a deadline, then all upstream reactions on which it depends (without logical delay) inherit its deadline. Hence, those upstream reactions will also be given higher priority.

    \n

    Second, the deadline mechanism provides a fault handler, a section of code to invoke when the deadline requirement is violated. Because invocation of the fault handler depends on factors beyond the control of the LF program, an LF program with deadlines becomes nondeterministic. The behavior of the program depends on the exact timing of the execution.

    \n

    There remains the question of when the fault handler should be invoked. By default, deadlines in LF are lazy, meaning that the fault handler is invoked at the logical time of the event triggering the reaction whose deadline is missed. Specifically, the possible violation of a deadline is not checked until the reaction with the deadline is ready to execute. Only then is the determination made whether to invoke the regular reaction or the fault handler.

    \n

    Deadline Specification

    \n

    A deadline is specified as follows:

    \n

    $start(Deadline)$

    \n
    target C;\nreactor Deadline {\n  input x:int;\n  output d:int; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    printf("Normal reaction.\\n");\n  =} deadline(10 msec) {=\n    printf("Deadline violation detected.\\n");\n    lf_set(d, x->value);\n  =}\n}\n
    \n
    target Cpp;\nreactor Deadline {\n  input x:int;\n  output d:int; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    std::cout << "Normal reaction." << std::endl;\n  =} deadline(10ms) {=\n    std::cout << "Deadline violation detected." << std::endl;\n    d.set(*x.get());\n  =}\n}\n
    \n
    target Python;\nreactor Deadline {\n  input x;\n  output d; // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    print("Normal reaction.")\n  =} deadline(10 msec) {=\n    print("Deadline violation detected.")\n    d.set(x.value)\n  =}\n}\n
    \n
    target TypeScript\nreactor Deadline {\n  input x:number\n  output d:number // Produced if the deadline is violated.\n  reaction(x) -> d {=\n    console.log("Normal reaction.")\n  =} deadline(10 msec) {=\n    console.log("Deadline violation detected.")\n    d = x\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Deadline.lf\n
    \n

    $end(Deadline)$

    \n

    This reactor specifies a deadline of 10 milliseconds (this can be a parameter of the reactor). If the reaction to x is triggered later in physical time than 10 msec past the timestamp of x, then the second body of code is executed instead of the first. That second body of code has access to anything the first body of code has access to, including the input x and the output d. The output can be used to notify the rest of the system that a deadline violation occurred. This reactor can be tested as follows:

    \n

    $start(DeadlineTest)$

    \n
    target C\nimport Deadline from "Deadline.lf"\npreamble {=\n  #include "platform.h"\n=}\nmain reactor {\n  logical action a\n  d = new Deadline()\n  reaction(startup) -> d.x, a {=\n    lf_set(d.x, 0);\n    lf_schedule(a, 0);\n  =}\n  reaction(a) -> d.x {=\n    lf_set(d.x, 0);\n    lf_sleep(MSEC(20));\n  =}\n  reaction(d.d) {=\n    printf("Deadline reactor produced an output.\\n");\n  =}\n}\n
    \n
    target Cpp;\nimport Deadline from "Deadline.lf";\nmain reactor {\n  logical action a;\n  d = new Deadline();\n  reaction(startup) -> d.x, a {=\n    d.x.set(0);\n    a.schedule(0ms);\n  =}\n  reaction(a) -> d.x {=\n    d.x.set(0);\n    std::this_thread::sleep_for(20ms);\n  =}\n  reaction(d.d) {=\n    std::cout << "Deadline reactor produced an output." << std::endl;\n  =}\n}\n
    \n
    target Python;\nimport Deadline from "Deadline.lf";\npreamble {= import time =}\nmain reactor {\n  logical action a;\n  d = new Deadline();\n  reaction(startup) -> d.x, a {=\n    d.x.set(0)\n    a.schedule(0)\n  =}\n  reaction(a) -> d.x {=\n    d.x.set(0)\n    time.sleep(0.02)\n  =}\n  reaction(d.d) {=\n    print("Deadline reactor produced an output.")\n  =}\n}\n
    \n
    target TypeScript\nimport Deadline from "Deadline.lf"\nmain reactor {\n  logical action a\n  d = new Deadline()\n  reaction(startup) -> d.x, a {=\n    d.x = 0\n    actions.a.schedule(TimeValue.zero(), null)\n  =}\n  reaction(a) -> d.x {=\n    d.x = 0\n    for (const later = util.getCurrentPhysicalTime().add(TimeValue.msecs(20));\n      util.getCurrentPhysicalTime() < later;) {\n      // Take time...\n    }\n  =}\n  reaction(d.d) {=\n    console.log("Deadline reactor produced an output.")\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/DeadlineTest.lf\n
    \n

    $end(DeadlineTest)$

    \n\"Lingua\n

    Running this program will result in the following output:

    \n
    Normal reaction.\nDeadline violation detected.\nDeadline reactor produced an output.
    \n

    The first reaction of the Deadline reactor does not violate the deadline, but the second does. Notice that the sleep in the $main$ reactor occurs after setting the output, but because of the deterministic semantics of LF, this does not matter. The actual value of an output cannot be known until every reaction that sets that output completes its execution. Since this reaction takes at least 20 msec to complete, the deadline is assured of being violated.

    \n

    Notice that the deadline is annotated in the diagram with a small clock symbol.

    \n
    \n

    Notice that the deadline violation here is caused by an invocation of lf_sleep, defined in \"platform.h\" (see Libraries Available to Programmers).\nIt is not generally advisable for a reaction to sleep because this can block other reactions from executing.\nBut this is exactly what we are trying to accomplish here in order to force a deadline to be violated.

    \n
    \n

    Deadline Violations During Execution

    \n

    Whether a deadline violation occurs is checked only before invoking the reaction with a deadline. What if the reaction itself runs for long enough that the deadline gets violated during the reaction execution? For this purpose, a target-language function is provided to check whether a deadline is violated during execution of a reaction with a deadline.

    \n
    \n

    NOTE: As of this writing, this function is only implemented in the C target.

    \n
    \n

    Consider this example:

    \n

    $start(CheckDeadline)$

    \n
    target C;\nreactor Count {\n  output out:int;\n  reaction(startup) -> out {=\n    int count = 0;\n    while (!lf_check_deadline(self, true)) {\n      count++;\n    }\n    lf_set(out, count);\n  =} deadline (3 msec) {=\n    printf("Stopped counting.\\n");\n  =}\n}\nmain reactor {\n  c = new Count();\n  reaction(c.out) {=\n    printf("Counted to %d\\n", c.out->value);\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/py/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/CheckDeadline.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/CheckDeadline.lf\n
    \n

    $end(CheckDeadline)$

    \n
    \n

    The Count reactor has a single reaction with a deadline of 3 msec.\nIf the deadline is not already violated when this reaction becomes enabled (at startup), then the reaction begins executing a loop. In each iteration of the loop, it calls lf_check_deadline(self, true), which returns true if the deadline has been violated and false otherwise. Hence, this reaction will increment the count variable as many times as possible before the deadline is violated and, at\nthat point, will exit the loop and produce on the output the count. Running this program will produce something like this:

    \n
    Stopped counting.\nCounted to 20257
    \n

    This is a (rather trivial) example of an anytime computation. Such computations proceed to improve results until time runs out and then produce the most improved result.

    \n

    The arguments to the lf_check_deadline are the self struct and a boolean that indicates whether the deadline violation handler should be invoked upon detecting a deadline violation. Because the argument is true above, the handler is invoked and Stopped counting is printed.

    \n
    ","headings":[{"value":"Purposes for Deadlines","depth":2},{"value":"Deadline Specification","depth":2},{"value":"Deadline Violations During Execution","depth":2}],"frontmatter":{"permalink":"/docs/handbook/deadlines","title":"Deadlines","oneline":"Deadlines in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Modal Reactors","oneline":"Modal Reactors","permalink":"/docs/handbook/modal-models"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","permalink":"/docs/handbook/multiports-and-banks"}}}},"pageContext":{"id":"1-deadlines","slug":"/docs/handbook/deadlines","repoPath":"/packages/documentation/copy/en/topics/Deadlines.md","previousID":"ddeb2577-9554-5362-9ed2-abba8f412fc1","nextID":"9ff63bbf-2bdf-553e-a96d-52355866ec94","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/developer-setup/page-data.json b/page-data/docs/handbook/developer-setup/page-data.json index be7734fa3..1e80539a5 100644 --- a/page-data/docs/handbook/developer-setup/page-data.json +++ b/page-data/docs/handbook/developer-setup/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/developer-setup","result":{"data":{"markdownRemark":{"id":"fcfe37e2-bc0b-50b8-8e1b-490aa315a726","excerpt":"Prerequisites Java 17 (download from Oracle) Cloning the Repository Please run the following commands to clone the repository and its submodules. Submodules are…","html":"

    Prerequisites

    \n\n

    Cloning the Repository

    \n

    Please run the following commands to clone the repository and its submodules.

    \n
    git clone git@github.com:lf-lang/lingua-franca.git\ncd lingua-franca\ngit submodule update --init --recursive\n
    \n

    Submodules are checked out over HTTPS by default. In case you want to commit to a submodule and use SSH instead, you can simply change the remote. For example, to change the remote of the reactor-c submodule, you can do this:

    \n
    cd core/src/main/resources/lib/c/reactor-c\ngit remote remove origin\ngit remote add origin git@github.com:lf-lang/reactor-c.git\n
    \n

    Building the command line tools

    \n

    We use Gradle for building the code within our repository.

    \n

    For an easy start, the bin/ directory contains scripts for building and running our command line tools, including the compiler lfc.\nTry to run ./bin/lfc-dev --version.\nThis will first build lfc and then execute it through Gradle.

    \n

    To build the entire repository, you can simply run ./gradlew build.\nThis will build all tools and also run all formatting checks and unit tests.\nNote that this does not run our integration tests.\nFor more details on our testing infrastructure, please refer to the Regression Test section.

    \n

    If you only want to build without running any tests, you can use ./gradlew assemble instead.\nBoth the assemble and the build task will create a distribution package containing our command line tools in build/distribution.\nThere is also an installed version of this package in build/install/lf-cli/.\nIf you run build/install/lf-cli/bin/lfc this will run lfc as it was last build.\nThus, you can choose if you want to use bin/lfc-dev, which first builds lfc using the latest code and then runs it, or if you prefer to run ./gradlew build and then separately invoke build/install/lf-cli/bin/lfc.

    \n

    IDE Integration

    \n

    You can use any editor or IDE that you like to work with the code base.\nHowever, we would suggest to choose an IDE that comes with good Java (and\nideally Kotlin) support and that integrates well with Gradle.\nWe recommend to use our IntelliJ setup.

    \n

    Building IDEs

    \n

    Currently, we provide two IDEs that support Lingua Franca programs.\nTheir source code is located in external repositories.\nWe have a Lingua Franca extension for VS code and an Eclipse based IDE called Epoch.\nPlease refer to the READMEs for build instructions.

    ","headings":[{"value":"Prerequisites","depth":2},{"value":"Cloning the Repository","depth":2},{"value":"Building the command line tools","depth":2},{"value":"IDE Integration","depth":2},{"value":"Building IDEs","depth":2}],"frontmatter":{"permalink":"/docs/handbook/developer-setup","title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","permalink":"/docs/handbook/intellij"}}}},"pageContext":{"id":"5-developer-setup","slug":"/docs/handbook/developer-setup","repoPath":"/packages/documentation/copy/en/developer/Downloading and Building.md","previousID":"008e847f-8ee6-513a-afab-0995ffce336b","nextID":"1d9f0442-2300-5615-9c04-6ee5f2c33793","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/developer-setup","result":{"data":{"markdownRemark":{"id":"fcfe37e2-bc0b-50b8-8e1b-490aa315a726","excerpt":"Prerequisites Java 17 (download from Oracle) Cloning the Repository Please run the following commands to clone the repository and its submodules. Submodules are…","html":"

    Prerequisites

    \n\n

    Cloning the Repository

    \n

    Please run the following commands to clone the repository and its submodules.

    \n
    git clone git@github.com:lf-lang/lingua-franca.git\ncd lingua-franca\ngit submodule update --init --recursive\n
    \n

    Submodules are checked out over HTTPS by default. In case you want to commit to a submodule and use SSH instead, you can simply change the remote. For example, to change the remote of the reactor-c submodule, you can do this:

    \n
    cd core/src/main/resources/lib/c/reactor-c\ngit remote remove origin\ngit remote add origin git@github.com:lf-lang/reactor-c.git\n
    \n

    Building the command line tools

    \n

    We use Gradle for building the code within our repository.

    \n

    For an easy start, the bin/ directory contains scripts for building and running our command line tools, including the compiler lfc.\nTry to run ./bin/lfc-dev --version.\nThis will first build lfc and then execute it through Gradle.

    \n

    To build the entire repository, you can simply run ./gradlew build.\nThis will build all tools and also run all formatting checks and unit tests.\nNote that this does not run our integration tests.\nFor more details on our testing infrastructure, please refer to the Regression Test section.

    \n

    If you only want to build without running any tests, you can use ./gradlew assemble instead.\nBoth the assemble and the build task will create a distribution package containing our command line tools in build/distribution.\nThere is also an installed version of this package in build/install/lf-cli/.\nIf you run build/install/lf-cli/bin/lfc this will run lfc as it was last build.\nThus, you can choose if you want to use bin/lfc-dev, which first builds lfc using the latest code and then runs it, or if you prefer to run ./gradlew build and then separately invoke build/install/lf-cli/bin/lfc.

    \n

    IDE Integration

    \n

    You can use any editor or IDE that you like to work with the code base.\nHowever, we would suggest to choose an IDE that comes with good Java (and\nideally Kotlin) support and that integrates well with Gradle.\nWe recommend to use our IntelliJ setup.

    \n

    Building IDEs

    \n

    Currently, we provide two IDEs that support Lingua Franca programs.\nTheir source code is located in external repositories.\nWe have a Lingua Franca extension for VS code and an Eclipse based IDE called Epoch.\nPlease refer to the READMEs for build instructions.

    ","headings":[{"value":"Prerequisites","depth":2},{"value":"Cloning the Repository","depth":2},{"value":"Building the command line tools","depth":2},{"value":"IDE Integration","depth":2},{"value":"Building IDEs","depth":2}],"frontmatter":{"permalink":"/docs/handbook/developer-setup","title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","permalink":"/docs/handbook/intellij"}}}},"pageContext":{"id":"5-developer-setup","slug":"/docs/handbook/developer-setup","repoPath":"/packages/documentation/copy/en/developer/Downloading and Building.md","previousID":"008e847f-8ee6-513a-afab-0995ffce336b","nextID":"1d9f0442-2300-5615-9c04-6ee5f2c33793","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/distributed-execution/page-data.json b/page-data/docs/handbook/distributed-execution/page-data.json index 78234c8df..ae7299526 100644 --- a/page-data/docs/handbook/distributed-execution/page-data.json +++ b/page-data/docs/handbook/distributed-execution/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/distributed-execution","result":{"data":{"markdownRemark":{"id":"01e8e05d-d3c6-5a55-a16a-4c1baf83a83c","excerpt":"Federated execution is not supported in $target-language$. $page-showing-target$ NOTE: Distributed execution of Lingua Franca programs is at an early stage of…","html":"
    \n

    Federated execution is not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    NOTE: Distributed execution of Lingua Franca programs is at an early stage of development with many missing capabilities and a rather brittle execution. It is ready for experimentation, but not yet for deployment of serious systems. The capability has been tested on macOS and Linux, and there are no plans currently to support Windows systems.

    \n

    A distributed Lingua Franca program is called a federation. Each reactor within the main reactor is called a federate. The LF compiler generates a separate program for each federate and synthesizes the code for the federates to communicate. The federates can be distributed across networks and eventually will be able to be written in different target languages, although this is not yet supported.

    \n

    In addition to the federates, there is a program called the RTI, for runtime infrastructure, that coordinates startup and shutdown and may, if the coordination is centralized, mediate communication. The RTI needs to be compiled and installed separately on the system before any federation can execute.

    \n

    It is possible to encapsulate federates in Docker containers for deployment.\nSee containerized execution.

    \n

    Installation of the RTI

    \n

    Federated execution requires installation of a separate stand-alone program called the Runtime Infrastructure or RTI. At the current time, the only way to install this is from source files:

    \n
    git clone https://github.com/lf-lang/reactor-c.git\ncd reactor-c/core/federated/RTI/\nmkdir build && cd build\ncmake ../\nmake\nsudo make install\n
    \n

    The above will create a program called RTI and install it at /usr/local/bin/RTI. Once this program is available in your path, you can compile and execute federated Lingua Franca programs using Epoch, VS Code, or the command-line tools. For more details, see the README file.

    \n

    Minimal Example

    \n

    A minimal federated execution is specified by using the $federated$ keyword instead of $main$ for the main federate. An example is given below:

    \n

    $start(Federated)$

    \n
    target C\nreactor Count {\n  output out: int\n  state c: int = 0\n  timer t(0, 1 sec)\n  reaction(t) -> out {=\n    lf_set(out, self->c++);\n  =}\n}\nreactor Print {\n  input in: int\n  reaction(in) {=\n    lf_print("Received: %d at (%lld, %d)", in->value,\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new Print()\n  c.out -> p.in\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Federated.lf\n
    \n
    target Python\nreactor Count {\n  output out\n  state c = 0\n  timer t(0, 1 sec)\n  reaction(t) -> out {=\n    out.set(self.c)\n    self.c += 1\n  =}\n}\nreactor Print {\n  input inp\n  reaction(inp) {=\n    print(\n        f"Received: {inp.value} "\n        f"at ({lf.time.logical_elapsed()}, {lf.tag().microstep})"\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new Print()\n  c.out -> p.inp\n}\n
    \n
    target TypeScript {\n  timeout: 1 msec  // FIXME: This should work with timeout: 0 msec.\n}\nreactor Source {\n  output out: string\n  reaction(startup) -> out {=\n    out = "Hello World!";\n  =}\n}\nreactor Destination {\n  input inp: string\n  reaction(inp) {=\n    console.log("Received: " + inp);\n  =}\n}\nfederated reactor Federated {\n  s = new Source()\n  d = new Destination()\n  s.out -> d.inp\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Federated.lf\n
    \n

    $end(Federated)$

    \n

    The $federated$ keyword tells the code generator that the program is to be split into several distinct programs, one for each top level reactor.

    \n

    When you run the code generator on src/Federated.lf containing the above code, the following three programs will appear:

    \n
    \n
    bin/Federated\nbin/Federated_s\nbin/Federated_d
    \n
    \n
    \n
    bin/Federated\nsrc-gen/Federated/s/Federated_s.py\nsrc-gen/Federated/d/Federated_d.py
    \n
    \n
    \n
    bin/Federated\nsrc-gen/dist/Federated/Federated_s.js\nsrc-gen/dist/Federated/Federated_d.js
    \n
    \n

    The root name, Federated, is the name of the .lf file from which these are generated (and the name of the main reactor, which is required to match if it is specified). The suffixes “_s” and “_d” come from the names of the top-level instances. There will always be one federate for each top-level reactor instance.

    \n

    To run the program, you can simply run bin/Federated, which is a bash script that launches the RTI and two other programs, Federated_s and Federated_d.\nAlternatively, you can manually execute the RTI followed by the two federate programs by starting them on the command line. It is best to use three separate terminal windows (so that outputs from the three programs do not get jumbled together) to execute the following commands:

    \n
    \n
    RTI -n 2\nbin/Federated_s\nbin/Federated_d\n
    \n
    \n
    \n
    RTI -n 2\npython3 src-gen/Federated/s/Federated_s.py\npython3 src-gen/Federated/d/Federated_d.py\n
    \n
    \n
    \n
    RTI -n 2\nnode src-gen/Federated/dist/Federated_s.js\nnode src-gen/Federated/dist/Federated_d.js\n
    \n
    \n

    The -n argument to the RTI specifies that there it should expect two federates to join the federation.

    \n

    Upon running the program, you will see information printed about the starting and ending of the federation, and buried in the output will be this line:

    \n
    \n
    Federate 1: Received: Hello World!
    \n

    The prefix Federate 1 is automatically added by the built-in lf_print function to help disambiguate the outputs from multiple concurrent federates.

    \n
    \n
    \n
    Received: Hello World!
    \n
    \n
    \n
    Received: Hello World!
    \n
    \n

    Federation ID

    \n

    You may have several federations running on the same machine(s) or even several instances of the same federation. In this case, it is necessary to distinguish between the federations. To accomplish this, you can pass a -i or --id parameter to the RTI and its federates with an identifier that is unique to the particular federation. For example,

    \n
    \n
    RTI -n 2 -i myFederation\nbin/Federated_s -i myFederation\nbin/Federated_d -i myFederation\n
    \n
    \n
    \n
    RTI -n 2 -i myFederation\npython3 src-gen/Federated/s/Federated_s.py -i myFederation\npython3 src-gen/Federated/d/Federated_d.py -i myFederation\n
    \n
    \n
    \n
    RTI -n 2 -i myFederation\nnode src-gen/Federated/dist/Federated_s.js -i myFederation\nnode src-gen/Federated/dist/Federated_d.js -i myFederation\n
    \n
    \n

    Each federate must have the same ID as the RTI in order to join the federation.\nThe bash script that executes each of the components of the federation automatically generates a unique federation ID each time you run it.

    \n

    Coordinated Start

    \n

    When the above programs execute, each federate registers with the RTI. When all expected federates have registered, the RTI broadcasts to the federates the logical time at which they should start execution. Hence, all federates start at the same logical time.

    \n

    The starting logical time is determined as follows. When each federate starts executing, it sends its current physical time (drawn from its real-time clock) to the RTI. When the RTI has heard from all the federates, it chooses the largest of these physical times, adds a fixed offset (currently one second), and broadcasts the resulting time to each federate.

    \n

    When a federate receives the starting time from the RTI, if it is running in realtime mode (the default), then it will wait until its local physical clock matches or exceeds that starting time. Thus, to the extent that the machines have synchronized clocks, the federates will all start executing at roughly the same physical time, a physical time close to the starting logical time.

    \n

    Coordinated Shutdown

    \n

    Coordinating the shutdown of a distributed program is discussed in Termination.

    \n

    Communication Between Federates

    \n

    When one federate sends data to another, by default, the timestamp at the receiver will match the timestamp at the sender. You can also specify a logical delay on the communication using the after keyword. For example, if we had instead specified

    \n
    \ts.out -> p.in after 200 msec;\n
    \n

    then the timestamp at the receiving end will be incremented by 200 msec compared to the timestamp at the sender.

    \n

    The preservation of timestamps across federates implies some constraints (unless you use physical connections). How these constraints are managed depends on whether you choose centralized or decentralized coordination.

    \n

    Centralized Coordination

    \n

    In the centralized mode of coordination (the default), the RTI regulates the advancement of time in each of the federates in order to ensure that the logical time semantics of Lingua Franca is respected. If the p federate above has an event with timestamp t that it wants to react to (it is the earliest event in its event queue), then it needs to get the OK from the RTI to advance its logical time to t. The RTI grants this time advance only when it can assure that p has received all messages that it will ever receive with timestamps t or less.

    \n

    First, note that, by default, logical time on each federate never advances ahead of physical time, as reported by its local physical clock. Consider the consequences for the above connection. Suppose the timestamp of the message sent by s is t. This message cannot be sent before the local clock at s reaches t and also cannot be sent before the RTI grants to s a time advance to t. Since s has no federates upstream of it, the RTI will always grant it such a time advance (in fact, it does not even wait for a response from the RTI).

    \n

    Suppose that the communication latency is L. That is, it takes L time units (in physical time) for a message to traverse the network. Then the p federate will not see the message from s before physical time t + L, where this physical time is measured by the physical clock on s’s host. If that clock differs from the clock on p’s host by E, then p will see the message at physical time t + E + L, as measured by its own clock. Let the value of the after specification (200 msec above) be a. Then the timestamp of the received message is t + a. The relationship between logical and physical times at the receiving end (the p federate), therefore, will depend on the relationship between a and E + L. If, for example, E + L > a, then federate p will lag behind physical time by at least E + L - a.

    \n

    Assume the RTI has granted a time advance to t to federate s. Hence, s is able to send a message with timestamp t. The RTI now cannot grant any time advance to p that is greater than or equal to t + a until the message has been delivered to p. In centralized coordination, all messages flow through the RTI, so the RTI will deliver a Tag Advance Grant (TAG) message to p only after it has delivered the message.

    \n

    If a > E + L, then the existence of this communication does not cause p’s logical time to lag behind physical time. This means that if we were to modify p to have a physical action, the RTI will be able to immediately grant a TAG to p to advance the timestamp of that physical action. However, if a < E + L, then the RTI will delay granting a time advance to p by at least E + L - a. Hence, E + L - a represents an additional latency in the processing of physical actions! This latency could present a problem for meeting deadlines. For this reason, if there are physical actions or deadlines at a federate that receives network messages, it is desirable to have $after$ delays on the connection to that federate larger than any expected E + L. This way, there is no additional latency to processing physical actions at this federate and no additional risk of missing deadlines.

    \n

    If, in addition, the physical clocks on the hosts are allowed to drift with respect to one another, then E can grow without bound, and hence the lag between logical time and physical time in processing events can grow without bound. This is mitigated either by hosts that themselves realize some clock synchronization algorithm, such as NTP or PTP, or by utilizing Lingua Franca’s own built in clock synchronization. If the federates lack physical actions and deadlines, however, then unsynchronized clocks present no semantic problem if you are using centralized coordination. However, because of logical time chases physical time, federates will slow to match the slowest clock of federates upstream of them.

    \n

    With centralized coordination, all messages (except those on physical connections) go through the RTI. This can create a bottleneck and a single point of failure. To avoid this bottleneck, you can use decentralized coordination.

    \n

    Decentralized Coordination

    \n

    The default coordination between mechanisms is centralized, equivalent to specifying the target property:

    \n
      coordination: centralized\n
    \n

    An alternative is decentralized coordination, which extends a technique realized in PTIDES and Google Spanner, a globally distributed database system:

    \n
      coordination: decentralized\n
    \n

    With decentralized coordination, the RTI coordinates startup, shutdown, and clock synchronization, but is otherwise not involved in the execution of the distributed program.

    \n

    In decentralized coordination, each federate and some reactions have a safe-to-process (STP) offset. When one federate communicates with another, it does so directly through a dedicated socket without going through the RTI. Moreover, it does not consult the RTI to advance logical time. Instead, it can advance its logical time to t when its physical clock matches or exceeds t + STP.

    \n

    By default, the STP is zero. An STP of zero is OK for any federate where either every logical connection into the federate has a sufficiently large $after$ clause, or the federate has only one upstream federate sending it messages and it has no local timers or actions. The value of the $after$ delay on each connection must exceed the sum of the clock synchronization error E, a bound L on the network latency, and the time lag on the sender D (the physical time at which it sends the message minus the timestamp of the message). The sender’s time lag D can be enforced by using a $deadline$. For example:

    \n

    $start(DecentralizedTimerAfter)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerAfter.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(\n        f"Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerAfter.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerAfter.lf\n
    \n

    $end(DecentralizedTimerAfter)$

    \n

    This example inherits from the Federated example above.\nIn this example, as long as the messages from federate c arrive at federate p within 10 msec, all messages will be processed in tag order, as with an unfederated program.

    \n

    An alternative to the $after$ delays is to add an STP offset to downstream federates, as in the following example:

    \n

    $start(DecentralizedTimerSTP)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer(STP_offset: time = 10 msec) extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerSTP.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer(STP_offset = 10 msec) extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(\n        "Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerSTP.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerSTP.lf\n
    \n

    $end(DecentralizedTimerSTP)$

    \n

    Here, a parameter named STP_offset (not case sensitive) gives a time value, and the federate waits this specified amount of time (physical time) beyond a logical time t before advancing its logical time to t. In the above example, reactions to the timer events will be delayed by the amount specified by the STP_offset parameter. Just as with the use of $after$, if the STP_offset exceeds the sum of network latency, clock synchronization error, and execution times, then all events will be processed in tag order.

    \n

    Of course, the assumptions about network latency, etc., can be violated in practice. Analogous to a deadline violation, Lingua Franca provides a mechanism for handling such a violation by providing an STP violation handler. The pattern is:

    \n
    reaction(in) {=\n    // User code\n=} STP (0) {=\n    // Error handling code\n=}\n
    \n

    If the tag at which this reaction is to be invoked (the value returned by lf_tag()) exceeds the tag of an incoming message in (the current tag has already advanced beyond the intended tag of in), then the STP violation handler will be invoked instead of the normal reaction. Within the body of the STP handler, the code can access the intended tag of in using in->intended_tag, which has two fields, a timestamp in->intended_tag.time and a microstep in->intended_tag.microstep. The code can then ascertain the severity of the error and act accordingly. For example:

    \n

    $start(DecentralizedTimerAfterHandler)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count from "Federated.lf"\nreactor PrintTimer {\n  timer t(0, 1 sec)\n  input in: int\n  reaction(in) {=\n    lf_print("Received: %d at (%lld, %d)", in->value,\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =} STP(0) {=\n    lf_print("****** STP violation handler invoked at (%lld, %d). "\n        "Intended tag was (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep,\n        in->intended_tag.time - lf_time_start(), in->intended_tag.microstep\n    );\n  =}\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerAfterHandler.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count from "Federated.lf"\nreactor PrintTimer {\n  timer t(0, 1 sec)\n  input inp\n  reaction(inp) {=\n    print(\n        f"Received: {inp.value} "\n        f"at ({lf.time.logical_elapsed()}, {lf.tag().microstep})"\n    )\n  =} STP(0) {=\n    print(\n        "****** STP violation handler invoked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep}). "\n        "Intended tag was "\n        f"({inp.intended_tag.time - lf.time.start()}, {inp.intended_tag.microstep})."\n    )\n  =}\n  reaction(t) {=\n    print(\n        "Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerAfterHandler.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerAfterHandler.lf\n
    \n

    $end(DecentralizedTimerAfterHandler)$

    \n

    For more advanced users, the LF API provides two functions that can be used to dynamically adjust the STP:

    \n
    interval_t lf_get_stp_offset();\nvoid lf_set_stp_offset(interval_t offset);\n
    \n

    Using these functions, however, is a pretty advanced operation.

    \n

    Physical Connections

    \n

    Coordinating the execution of the federates so that timestamps are preserved is tricky. If your application does not require the deterministic execution that results from preserving the timestamps, then you can alternatively specify a physical connection as follows:

    \n
    source.out ~> print.in;\n
    \n

    The tilde specifies that the timestamp of the sender should be discarded. A new timestamp will be assigned at the receiving end based on the local physical clock, much like a physical action. To distinguish it from a physical connection, the normal connection is called a logical connection.

    \n

    There are a number of subtleties with physical connections. One is that if you specify an after clause, for example like this:

    \n
    count.out ~> print.in after 10 msec;\n
    \n

    then what does this mean? At the receiving end, the timestamp assigned to the incoming event will be the current physical time plus 10 msec.

    \n

    Prerequisites for Distributed Execution

    \n

    In the above example, all of the generated programs expect to run on localhost. This is the default. With these defaults, every federate has to run on the same machine as the RTI because localhost is not a host that is visible from other machines on the network. In order to run federates or the RTI on remote machines, you can specify a domain name or IP address for the RTI and/or federates.

    \n

    In order for a federated execution to work, there is some setup required on the machines to be used. First, each machine must be running on ssh server. On a Linux machine, this is typically done with a command like this:

    \n
      sudo systemctl <start|enable> ssh.service\n
    \n

    Enable means to always start the service at startup, whereas start means to just start it this once. On macOS, open System Preferences from the Apple menu and click on the “Sharing” preference panel. Select the checkbox next to “Remote Login” to enable it.

    \n

    It will also be much more convenient if the launcher does not have to enter passwords to gain access to the remote machine. This can be accomplished by installing your public key (typically found in ~/.ssh/id_rsa.pub) in ~/.ssh/authorized_keys on the remote host.

    \n

    Second, the RTI must be installed on the remote machine. See instructions for installation the RTI.

    \n

    Specifying RTI Hosts

    \n

    You can specify a domain name on which the RTI should run as follows:

    \n
    federated reactor DistributedCount at www.example.com {\n  ...\n}\n
    \n

    You can alternatively specify an IP address (either IPv4 or IPv6):

    \n
    federated reactor DistributedCount at 10.0.0.198 { ... }\n
    \n

    By default, the RTI starts a socket server on port 15045, if that port is available, and increments the port number by 1 until it finds an available port. The number of increments is limited by a target-specific number. In the C target, in rti.h, STARTING_PORT defines the number 15045 and PORT_RANGE_LIMIT limits the range of ports attempted (currently 1024).

    \n

    You can also specify a port for the RTI to use as follows:

    \n
    federated reactor DistributedCount at 10.0.0.198:8080 { ... }\n
    \n

    If you specify a specific port, then it will use that port if it is available and fail otherwise. The above changes this to port 8080.

    \n

    Note that if the machine uses DHCP to obtain its address, then the generated code may not work in the future since the address of the machine may change in the future.

    \n

    Address 0.0.0.0: The default host, localhost is used if no address is specified. Using localhost requires that the generated federates run on the local machine. This is ideal for testing. If you use 0.0.0.0, then you are also specifying that the local machine (the one performing the code generation) will be the host, but now the process(es) running on this local machine can establish connections with processes on remote machines. The code generator will determine the IP address of the local machine, and any other hosts that need to communicate with reactors on the local host will use the current IP address of that local host at the time of code generation.

    \n

    Specifying Federate Hosts

    \n

    A federate may be mapped to a particular remote machine using a syntax like this:

    \n
      count = new Count() at user@host:port/path;\n
    \n

    The port is ignored in centralized mode because all communication is routed through the RTI, but in decentralized mode it will specify the port on which a socket server listens for incoming connections from other federates.

    \n

    If any federate has such a remote designator, then a Federation_distribute.sh shell script will be generated. This script will distribute the generated code for the RTI to the remote machine at the specified directory.

    \n

    You can also specify a user name on the remote machine for cases where the username will not match whoever launches the federation:

    \n
    federated reactor DistributedCount at user@10.0.0.198:8080 { ... }\n
    \n

    The general form of the host designation is

    \n
    federated reactor DistributedCount at user@host:port/path { ... }\n
    \n

    where user@, :port, and /path are all optional. The path specifies the directory on the remote machine (relative to the home directory of the user) where the generated code will be put. The host should be an IPv4 address (e.g. 93.184.216.34), IPv6 address (e.g. 2606:2800:220:1:248:1893:25c8:1946), or a domain name (e.g. www.example.com). It can also be localhost or 0.0.0.0. The host can be remote as long as it is accessible from the machine where the programs will be started.

    \n

    If user@ is not given, then it is assumed that the username on the remote host is the same as on the machine that launches the programs. If :port is not given, then it defaults to port 15045. If /path is not given, then ~user/LinguaFrancaRemote will be the root directory on the remote machine.

    \n

    Clock Synchronization

    \n

    Both centralized and decentralized coordination have some reliance on clock synchronization. First, the RTI determines the start time of all federates, and the actually physical start time will differ by the extent that their physical clocks differ. This is particularly problematic if clocks differ by hours or more, which is certainly possible. If the hosts on which you are running run a clock synchronization algorithm, such as NTP or PTP, then you may not need to be concerned about this at all. Windows, Mac, and most versions of Linux, by default, run NTP, which synchronizes their clocks to some remote host. NTP is not particularly precise, however, so clock synchronization error can be hundreds of milliseconds or larger. PTP protocols are much more precise, so if your hosts derive their physical clocks from a PTP implementation, then you probably don’t need to do anything further. Unfortunately, as of this writing, even though almost all networking hardware provides support for PTP, few operating systems utilize it. We expect this to change when people have finally understood the value of precise clock synchronization.

    \n

    If your host is not running any clock synchronization, or if it is running only NTP and your application needs tighter latencies, then Lingua Franca’s own built-in clock synchronization may provide better precision, depending on your network conditions. Like NTP, it realizes a software-only protocol, which are much less precise than hardware-supported protocols such as PTP, but if your hosts are on the same local area network, then network conditions may be such that the performance of LF clock synchronization will be much better than NTP. If your network is equipped with PTP, you will want to disable the clock synchronization in Lingua Franca by specifying in your target properties the following:

    \n
      clock-sync: off\n
    \n

    When a federation is mapped onto multiple machines, then, by default, any federate mapped to a machine that is not the one running the RTI will attempt during startup to synchronize its clock with the one on the machine running the RTI. The determination of whether the federate is running on the same machine is determined by comparing the string that comes after the at clause between the federate and the RTI. If they differ at all, then they will be treated as if the federate is running on a different machine even if it is actually running on the same machine. This default behavior can be obtained by either specifying nothing in the target properties or saying:

    \n
      clock-sync: initial\n
    \n

    This results in clock synchronization being done during startup only. To account for the possibility of your clocks drifting during execution of the program, you can alternatively specify:

    \n
      clock-sync: on\n
    \n

    With this specification, in addition to synchronization during startup, synchronization will be redone periodically during program execution.

    \n

    Clock Synchronization Options

    \n

    A number of options can be specified using the clock-sync-options target parameter. For example:

    \n
      clock-sync-options: {local-federates-on: true, test-offset: 200 msec}\n
    \n

    The supported options are:

    \n
      \n
    • \n

      local-federates-on: Should be true or false. By default, if a federate is mapped to the same host as the RTI (using the at keyword), then clock synchronization is turned off. This assumes that the federate will be using the same clock as the RTI, so there is no point in performing clock synchronization. However, sometimes it is useful to force clock synchronization to be run even in this case, for example to test the performance of clock synchronization. To force clock synchronization on in this case, set this option to true.

      \n
    • \n
    • \n

      test-offset: The value should be a time value with units, e.g. 200 msec. This will establish an artificial fixed offset for each federate’s clock of one plus the federate ID times the time value given. For example, with the value 200 msec, a fixed offset of 200 milliseconds will be set on the clock for federate 0, 400 msec on the clock of federate 1, etc.

      \n
    • \n
    • \n

      period: A time value (with units) that specifies how often runtime clock synchronization will be performed if it is turned on. The default is 5 msec.

      \n
    • \n
    • \n

      attenuation: A positive integer specifying a divisor applied to the estimated clock error during runtime clock synchronization when adjusting the clock offset. The default is 10. Making this number bigger reduces each adjustment to the clock. Making the number equal to 1 means that each round of clock synchronization fully applies its estimated clock synchronization error.

      \n
    • \n
    • \n

      trials: The number of rounds of message exchange with the RTI in each clock synchronization round. This defaults to 10.

      \n
    • \n
    \n
    ","headings":[{"value":"Installation of the RTI","depth":2},{"value":"Minimal Example","depth":2},{"value":"Federation ID","depth":2},{"value":"Coordinated Start","depth":2},{"value":"Coordinated Shutdown","depth":2},{"value":"Communication Between Federates","depth":2},{"value":"Centralized Coordination","depth":2},{"value":"Decentralized Coordination","depth":2},{"value":"Physical Connections","depth":2},{"value":"Prerequisites for Distributed Execution","depth":2},{"value":"Specifying RTI Hosts","depth":2},{"value":"Specifying Federate Hosts","depth":2},{"value":"Clock Synchronization","depth":2},{"value":"Clock Synchronization Options","depth":3}],"frontmatter":{"permalink":"/docs/handbook/distributed-execution","title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Preambles","oneline":"Defining preambles in Lingua Franca.","permalink":"/docs/handbook/preambles"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Termination","oneline":"Terminating a Lingua Franca execution.","permalink":"/docs/handbook/termination"}}}},"pageContext":{"id":"1-distributed-execution","slug":"/docs/handbook/distributed-execution","repoPath":"/packages/documentation/copy/en/topics/Distributed Execution.md","previousID":"d9c76683-1fe5-55e0-b223-8e21c125f9cd","nextID":"ddf59040-674f-5833-a630-a62f39d0106e","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/distributed-execution","result":{"data":{"markdownRemark":{"id":"01e8e05d-d3c6-5a55-a16a-4c1baf83a83c","excerpt":"Federated execution is not supported in $target-language$. $page-showing-target$ NOTE: Distributed execution of Lingua Franca programs is at an early stage of…","html":"
    \n

    Federated execution is not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    NOTE: Distributed execution of Lingua Franca programs is at an early stage of development with many missing capabilities and a rather brittle execution. It is ready for experimentation, but not yet for deployment of serious systems. The capability has been tested on macOS and Linux, and there are no plans currently to support Windows systems.

    \n

    A distributed Lingua Franca program is called a federation. Each reactor within the main reactor is called a federate. The LF compiler generates a separate program for each federate and synthesizes the code for the federates to communicate. The federates can be distributed across networks and eventually will be able to be written in different target languages, although this is not yet supported.

    \n

    In addition to the federates, there is a program called the RTI, for runtime infrastructure, that coordinates startup and shutdown and may, if the coordination is centralized, mediate communication. The RTI needs to be compiled and installed separately on the system before any federation can execute.

    \n

    It is possible to encapsulate federates in Docker containers for deployment.\nSee containerized execution.

    \n

    Installation of the RTI

    \n

    Federated execution requires installation of a separate stand-alone program called the Runtime Infrastructure or RTI. At the current time, the only way to install this is from source files:

    \n
    git clone https://github.com/lf-lang/reactor-c.git\ncd reactor-c/core/federated/RTI/\nmkdir build && cd build\ncmake ../\nmake\nsudo make install\n
    \n

    The above will create a program called RTI and install it at /usr/local/bin/RTI. Once this program is available in your path, you can compile and execute federated Lingua Franca programs using Epoch, VS Code, or the command-line tools. For more details, see the README file.

    \n

    Minimal Example

    \n

    A minimal federated execution is specified by using the $federated$ keyword instead of $main$ for the main federate. An example is given below:

    \n

    $start(Federated)$

    \n
    target C\nreactor Count {\n  output out: int\n  state c: int = 0\n  timer t(0, 1 sec)\n  reaction(t) -> out {=\n    lf_set(out, self->c++);\n  =}\n}\nreactor Print {\n  input in: int\n  reaction(in) {=\n    lf_print("Received: %d at (%lld, %d)", in->value,\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new Print()\n  c.out -> p.in\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Federated.lf\n
    \n
    target Python\nreactor Count {\n  output out\n  state c = 0\n  timer t(0, 1 sec)\n  reaction(t) -> out {=\n    out.set(self.c)\n    self.c += 1\n  =}\n}\nreactor Print {\n  input inp\n  reaction(inp) {=\n    print(\n        f"Received: {inp.value} "\n        f"at ({lf.time.logical_elapsed()}, {lf.tag().microstep})"\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new Print()\n  c.out -> p.inp\n}\n
    \n
    target TypeScript {\n  timeout: 1 msec  // FIXME: This should work with timeout: 0 msec.\n}\nreactor Source {\n  output out: string\n  reaction(startup) -> out {=\n    out = "Hello World!";\n  =}\n}\nreactor Destination {\n  input inp: string\n  reaction(inp) {=\n    console.log("Received: " + inp);\n  =}\n}\nfederated reactor Federated {\n  s = new Source()\n  d = new Destination()\n  s.out -> d.inp\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Federated.lf\n
    \n

    $end(Federated)$

    \n

    The $federated$ keyword tells the code generator that the program is to be split into several distinct programs, one for each top level reactor.

    \n

    When you run the code generator on src/Federated.lf containing the above code, the following three programs will appear:

    \n
    \n
    bin/Federated\nbin/Federated_s\nbin/Federated_d
    \n
    \n
    \n
    bin/Federated\nsrc-gen/Federated/s/Federated_s.py\nsrc-gen/Federated/d/Federated_d.py
    \n
    \n
    \n
    bin/Federated\nsrc-gen/dist/Federated/Federated_s.js\nsrc-gen/dist/Federated/Federated_d.js
    \n
    \n

    The root name, Federated, is the name of the .lf file from which these are generated (and the name of the main reactor, which is required to match if it is specified). The suffixes “_s” and “_d” come from the names of the top-level instances. There will always be one federate for each top-level reactor instance.

    \n

    To run the program, you can simply run bin/Federated, which is a bash script that launches the RTI and two other programs, Federated_s and Federated_d.\nAlternatively, you can manually execute the RTI followed by the two federate programs by starting them on the command line. It is best to use three separate terminal windows (so that outputs from the three programs do not get jumbled together) to execute the following commands:

    \n
    \n
    RTI -n 2\nbin/Federated_s\nbin/Federated_d\n
    \n
    \n
    \n
    RTI -n 2\npython3 src-gen/Federated/s/Federated_s.py\npython3 src-gen/Federated/d/Federated_d.py\n
    \n
    \n
    \n
    RTI -n 2\nnode src-gen/Federated/dist/Federated_s.js\nnode src-gen/Federated/dist/Federated_d.js\n
    \n
    \n

    The -n argument to the RTI specifies that there it should expect two federates to join the federation.

    \n

    Upon running the program, you will see information printed about the starting and ending of the federation, and buried in the output will be this line:

    \n
    \n
    Federate 1: Received: Hello World!
    \n

    The prefix Federate 1 is automatically added by the built-in lf_print function to help disambiguate the outputs from multiple concurrent federates.

    \n
    \n
    \n
    Received: Hello World!
    \n
    \n
    \n
    Received: Hello World!
    \n
    \n

    Federation ID

    \n

    You may have several federations running on the same machine(s) or even several instances of the same federation. In this case, it is necessary to distinguish between the federations. To accomplish this, you can pass a -i or --id parameter to the RTI and its federates with an identifier that is unique to the particular federation. For example,

    \n
    \n
    RTI -n 2 -i myFederation\nbin/Federated_s -i myFederation\nbin/Federated_d -i myFederation\n
    \n
    \n
    \n
    RTI -n 2 -i myFederation\npython3 src-gen/Federated/s/Federated_s.py -i myFederation\npython3 src-gen/Federated/d/Federated_d.py -i myFederation\n
    \n
    \n
    \n
    RTI -n 2 -i myFederation\nnode src-gen/Federated/dist/Federated_s.js -i myFederation\nnode src-gen/Federated/dist/Federated_d.js -i myFederation\n
    \n
    \n

    Each federate must have the same ID as the RTI in order to join the federation.\nThe bash script that executes each of the components of the federation automatically generates a unique federation ID each time you run it.

    \n

    Coordinated Start

    \n

    When the above programs execute, each federate registers with the RTI. When all expected federates have registered, the RTI broadcasts to the federates the logical time at which they should start execution. Hence, all federates start at the same logical time.

    \n

    The starting logical time is determined as follows. When each federate starts executing, it sends its current physical time (drawn from its real-time clock) to the RTI. When the RTI has heard from all the federates, it chooses the largest of these physical times, adds a fixed offset (currently one second), and broadcasts the resulting time to each federate.

    \n

    When a federate receives the starting time from the RTI, if it is running in realtime mode (the default), then it will wait until its local physical clock matches or exceeds that starting time. Thus, to the extent that the machines have synchronized clocks, the federates will all start executing at roughly the same physical time, a physical time close to the starting logical time.

    \n

    Coordinated Shutdown

    \n

    Coordinating the shutdown of a distributed program is discussed in Termination.

    \n

    Communication Between Federates

    \n

    When one federate sends data to another, by default, the timestamp at the receiver will match the timestamp at the sender. You can also specify a logical delay on the communication using the after keyword. For example, if we had instead specified

    \n
    \ts.out -> p.in after 200 msec;\n
    \n

    then the timestamp at the receiving end will be incremented by 200 msec compared to the timestamp at the sender.

    \n

    The preservation of timestamps across federates implies some constraints (unless you use physical connections). How these constraints are managed depends on whether you choose centralized or decentralized coordination.

    \n

    Centralized Coordination

    \n

    In the centralized mode of coordination (the default), the RTI regulates the advancement of time in each of the federates in order to ensure that the logical time semantics of Lingua Franca is respected. If the p federate above has an event with timestamp t that it wants to react to (it is the earliest event in its event queue), then it needs to get the OK from the RTI to advance its logical time to t. The RTI grants this time advance only when it can assure that p has received all messages that it will ever receive with timestamps t or less.

    \n

    First, note that, by default, logical time on each federate never advances ahead of physical time, as reported by its local physical clock. Consider the consequences for the above connection. Suppose the timestamp of the message sent by s is t. This message cannot be sent before the local clock at s reaches t and also cannot be sent before the RTI grants to s a time advance to t. Since s has no federates upstream of it, the RTI will always grant it such a time advance (in fact, it does not even wait for a response from the RTI).

    \n

    Suppose that the communication latency is L. That is, it takes L time units (in physical time) for a message to traverse the network. Then the p federate will not see the message from s before physical time t + L, where this physical time is measured by the physical clock on s’s host. If that clock differs from the clock on p’s host by E, then p will see the message at physical time t + E + L, as measured by its own clock. Let the value of the after specification (200 msec above) be a. Then the timestamp of the received message is t + a. The relationship between logical and physical times at the receiving end (the p federate), therefore, will depend on the relationship between a and E + L. If, for example, E + L > a, then federate p will lag behind physical time by at least E + L - a.

    \n

    Assume the RTI has granted a time advance to t to federate s. Hence, s is able to send a message with timestamp t. The RTI now cannot grant any time advance to p that is greater than or equal to t + a until the message has been delivered to p. In centralized coordination, all messages flow through the RTI, so the RTI will deliver a Tag Advance Grant (TAG) message to p only after it has delivered the message.

    \n

    If a > E + L, then the existence of this communication does not cause p’s logical time to lag behind physical time. This means that if we were to modify p to have a physical action, the RTI will be able to immediately grant a TAG to p to advance the timestamp of that physical action. However, if a < E + L, then the RTI will delay granting a time advance to p by at least E + L - a. Hence, E + L - a represents an additional latency in the processing of physical actions! This latency could present a problem for meeting deadlines. For this reason, if there are physical actions or deadlines at a federate that receives network messages, it is desirable to have $after$ delays on the connection to that federate larger than any expected E + L. This way, there is no additional latency to processing physical actions at this federate and no additional risk of missing deadlines.

    \n

    If, in addition, the physical clocks on the hosts are allowed to drift with respect to one another, then E can grow without bound, and hence the lag between logical time and physical time in processing events can grow without bound. This is mitigated either by hosts that themselves realize some clock synchronization algorithm, such as NTP or PTP, or by utilizing Lingua Franca’s own built in clock synchronization. If the federates lack physical actions and deadlines, however, then unsynchronized clocks present no semantic problem if you are using centralized coordination. However, because of logical time chases physical time, federates will slow to match the slowest clock of federates upstream of them.

    \n

    With centralized coordination, all messages (except those on physical connections) go through the RTI. This can create a bottleneck and a single point of failure. To avoid this bottleneck, you can use decentralized coordination.

    \n

    Decentralized Coordination

    \n

    The default coordination between mechanisms is centralized, equivalent to specifying the target property:

    \n
      coordination: centralized\n
    \n

    An alternative is decentralized coordination, which extends a technique realized in PTIDES and Google Spanner, a globally distributed database system:

    \n
      coordination: decentralized\n
    \n

    With decentralized coordination, the RTI coordinates startup, shutdown, and clock synchronization, but is otherwise not involved in the execution of the distributed program.

    \n

    In decentralized coordination, each federate and some reactions have a safe-to-process (STP) offset. When one federate communicates with another, it does so directly through a dedicated socket without going through the RTI. Moreover, it does not consult the RTI to advance logical time. Instead, it can advance its logical time to t when its physical clock matches or exceeds t + STP.

    \n

    By default, the STP is zero. An STP of zero is OK for any federate where either every logical connection into the federate has a sufficiently large $after$ clause, or the federate has only one upstream federate sending it messages and it has no local timers or actions. The value of the $after$ delay on each connection must exceed the sum of the clock synchronization error E, a bound L on the network latency, and the time lag on the sender D (the physical time at which it sends the message minus the timestamp of the message). The sender’s time lag D can be enforced by using a $deadline$. For example:

    \n

    $start(DecentralizedTimerAfter)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerAfter.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(\n        f"Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerAfter.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerAfter.lf\n
    \n

    $end(DecentralizedTimerAfter)$

    \n

    This example inherits from the Federated example above.\nIn this example, as long as the messages from federate c arrive at federate p within 10 msec, all messages will be processed in tag order, as with an unfederated program.

    \n

    An alternative to the $after$ delays is to add an STP offset to downstream federates, as in the following example:

    \n

    $start(DecentralizedTimerSTP)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer(STP_offset: time = 10 msec) extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerSTP.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count, Print from "Federated.lf"\nreactor PrintTimer(STP_offset = 10 msec) extends Print {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(\n        "Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerSTP.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerSTP.lf\n
    \n

    $end(DecentralizedTimerSTP)$

    \n

    Here, a parameter named STP_offset (not case sensitive) gives a time value, and the federate waits this specified amount of time (physical time) beyond a logical time t before advancing its logical time to t. In the above example, reactions to the timer events will be delayed by the amount specified by the STP_offset parameter. Just as with the use of $after$, if the STP_offset exceeds the sum of network latency, clock synchronization error, and execution times, then all events will be processed in tag order.

    \n

    Of course, the assumptions about network latency, etc., can be violated in practice. Analogous to a deadline violation, Lingua Franca provides a mechanism for handling such a violation by providing an STP violation handler. The pattern is:

    \n
    reaction(in) {=\n    // User code\n=} STP (0) {=\n    // Error handling code\n=}\n
    \n

    If the tag at which this reaction is to be invoked (the value returned by lf_tag()) exceeds the tag of an incoming message in (the current tag has already advanced beyond the intended tag of in), then the STP violation handler will be invoked instead of the normal reaction. Within the body of the STP handler, the code can access the intended tag of in using in->intended_tag, which has two fields, a timestamp in->intended_tag.time and a microstep in->intended_tag.microstep. The code can then ascertain the severity of the error and act accordingly. For example:

    \n

    $start(DecentralizedTimerAfterHandler)$

    \n
    target C {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count from "Federated.lf"\nreactor PrintTimer {\n  timer t(0, 1 sec)\n  input in: int\n  reaction(in) {=\n    lf_print("Received: %d at (%lld, %d)", in->value,\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =} STP(0) {=\n    lf_print("****** STP violation handler invoked at (%lld, %d). "\n        "Intended tag was (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep,\n        in->intended_tag.time - lf_time_start(), in->intended_tag.microstep\n    );\n  =}\n  reaction(t) {=\n    lf_print("Timer ticked at (%lld, %d).",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.in after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/DecentralizedTimerAfterHandler.lf\n
    \n
    target Python {\n  timeout: 5 sec,\n  coordination: decentralized\n}\nimport Count from "Federated.lf"\nreactor PrintTimer {\n  timer t(0, 1 sec)\n  input inp\n  reaction(inp) {=\n    print(\n        f"Received: {inp.value} "\n        f"at ({lf.time.logical_elapsed()}, {lf.tag().microstep})"\n    )\n  =} STP(0) {=\n    print(\n        "****** STP violation handler invoked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep}). "\n        "Intended tag was "\n        f"({inp.intended_tag.time - lf.time.start()}, {inp.intended_tag.microstep})."\n    )\n  =}\n  reaction(t) {=\n    print(\n        "Timer ticked at "\n        f"({lf.time.logical_elapsed()}, {lf.tag().microstep})."\n    )\n  =}\n}\nfederated reactor {\n  c = new Count()\n  p = new PrintTimer()\n  c.out -> p.inp after 10 msec\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/DecentralizedTimerAfterHandler.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/DecentralizedTimerAfterHandler.lf\n
    \n

    $end(DecentralizedTimerAfterHandler)$

    \n

    For more advanced users, the LF API provides two functions that can be used to dynamically adjust the STP:

    \n
    interval_t lf_get_stp_offset();\nvoid lf_set_stp_offset(interval_t offset);\n
    \n

    Using these functions, however, is a pretty advanced operation.

    \n

    Physical Connections

    \n

    Coordinating the execution of the federates so that timestamps are preserved is tricky. If your application does not require the deterministic execution that results from preserving the timestamps, then you can alternatively specify a physical connection as follows:

    \n
    source.out ~> print.in;\n
    \n

    The tilde specifies that the timestamp of the sender should be discarded. A new timestamp will be assigned at the receiving end based on the local physical clock, much like a physical action. To distinguish it from a physical connection, the normal connection is called a logical connection.

    \n

    There are a number of subtleties with physical connections. One is that if you specify an after clause, for example like this:

    \n
    count.out ~> print.in after 10 msec;\n
    \n

    then what does this mean? At the receiving end, the timestamp assigned to the incoming event will be the current physical time plus 10 msec.

    \n

    Prerequisites for Distributed Execution

    \n

    In the above example, all of the generated programs expect to run on localhost. This is the default. With these defaults, every federate has to run on the same machine as the RTI because localhost is not a host that is visible from other machines on the network. In order to run federates or the RTI on remote machines, you can specify a domain name or IP address for the RTI and/or federates.

    \n

    In order for a federated execution to work, there is some setup required on the machines to be used. First, each machine must be running on ssh server. On a Linux machine, this is typically done with a command like this:

    \n
      sudo systemctl <start|enable> ssh.service\n
    \n

    Enable means to always start the service at startup, whereas start means to just start it this once. On macOS, open System Preferences from the Apple menu and click on the “Sharing” preference panel. Select the checkbox next to “Remote Login” to enable it.

    \n

    It will also be much more convenient if the launcher does not have to enter passwords to gain access to the remote machine. This can be accomplished by installing your public key (typically found in ~/.ssh/id_rsa.pub) in ~/.ssh/authorized_keys on the remote host.

    \n

    Second, the RTI must be installed on the remote machine. See instructions for installation the RTI.

    \n

    Specifying RTI Hosts

    \n

    You can specify a domain name on which the RTI should run as follows:

    \n
    federated reactor DistributedCount at www.example.com {\n  ...\n}\n
    \n

    You can alternatively specify an IP address (either IPv4 or IPv6):

    \n
    federated reactor DistributedCount at 10.0.0.198 { ... }\n
    \n

    By default, the RTI starts a socket server on port 15045, if that port is available, and increments the port number by 1 until it finds an available port. The number of increments is limited by a target-specific number. In the C target, in rti.h, STARTING_PORT defines the number 15045 and PORT_RANGE_LIMIT limits the range of ports attempted (currently 1024).

    \n

    You can also specify a port for the RTI to use as follows:

    \n
    federated reactor DistributedCount at 10.0.0.198:8080 { ... }\n
    \n

    If you specify a specific port, then it will use that port if it is available and fail otherwise. The above changes this to port 8080.

    \n

    Note that if the machine uses DHCP to obtain its address, then the generated code may not work in the future since the address of the machine may change in the future.

    \n

    Address 0.0.0.0: The default host, localhost is used if no address is specified. Using localhost requires that the generated federates run on the local machine. This is ideal for testing. If you use 0.0.0.0, then you are also specifying that the local machine (the one performing the code generation) will be the host, but now the process(es) running on this local machine can establish connections with processes on remote machines. The code generator will determine the IP address of the local machine, and any other hosts that need to communicate with reactors on the local host will use the current IP address of that local host at the time of code generation.

    \n

    Specifying Federate Hosts

    \n

    A federate may be mapped to a particular remote machine using a syntax like this:

    \n
      count = new Count() at user@host:port/path;\n
    \n

    The port is ignored in centralized mode because all communication is routed through the RTI, but in decentralized mode it will specify the port on which a socket server listens for incoming connections from other federates.

    \n

    If any federate has such a remote designator, then a Federation_distribute.sh shell script will be generated. This script will distribute the generated code for the RTI to the remote machine at the specified directory.

    \n

    You can also specify a user name on the remote machine for cases where the username will not match whoever launches the federation:

    \n
    federated reactor DistributedCount at user@10.0.0.198:8080 { ... }\n
    \n

    The general form of the host designation is

    \n
    federated reactor DistributedCount at user@host:port/path { ... }\n
    \n

    where user@, :port, and /path are all optional. The path specifies the directory on the remote machine (relative to the home directory of the user) where the generated code will be put. The host should be an IPv4 address (e.g. 93.184.216.34), IPv6 address (e.g. 2606:2800:220:1:248:1893:25c8:1946), or a domain name (e.g. www.example.com). It can also be localhost or 0.0.0.0. The host can be remote as long as it is accessible from the machine where the programs will be started.

    \n

    If user@ is not given, then it is assumed that the username on the remote host is the same as on the machine that launches the programs. If :port is not given, then it defaults to port 15045. If /path is not given, then ~user/LinguaFrancaRemote will be the root directory on the remote machine.

    \n

    Clock Synchronization

    \n

    Both centralized and decentralized coordination have some reliance on clock synchronization. First, the RTI determines the start time of all federates, and the actually physical start time will differ by the extent that their physical clocks differ. This is particularly problematic if clocks differ by hours or more, which is certainly possible. If the hosts on which you are running run a clock synchronization algorithm, such as NTP or PTP, then you may not need to be concerned about this at all. Windows, Mac, and most versions of Linux, by default, run NTP, which synchronizes their clocks to some remote host. NTP is not particularly precise, however, so clock synchronization error can be hundreds of milliseconds or larger. PTP protocols are much more precise, so if your hosts derive their physical clocks from a PTP implementation, then you probably don’t need to do anything further. Unfortunately, as of this writing, even though almost all networking hardware provides support for PTP, few operating systems utilize it. We expect this to change when people have finally understood the value of precise clock synchronization.

    \n

    If your host is not running any clock synchronization, or if it is running only NTP and your application needs tighter latencies, then Lingua Franca’s own built-in clock synchronization may provide better precision, depending on your network conditions. Like NTP, it realizes a software-only protocol, which are much less precise than hardware-supported protocols such as PTP, but if your hosts are on the same local area network, then network conditions may be such that the performance of LF clock synchronization will be much better than NTP. If your network is equipped with PTP, you will want to disable the clock synchronization in Lingua Franca by specifying in your target properties the following:

    \n
      clock-sync: off\n
    \n

    When a federation is mapped onto multiple machines, then, by default, any federate mapped to a machine that is not the one running the RTI will attempt during startup to synchronize its clock with the one on the machine running the RTI. The determination of whether the federate is running on the same machine is determined by comparing the string that comes after the at clause between the federate and the RTI. If they differ at all, then they will be treated as if the federate is running on a different machine even if it is actually running on the same machine. This default behavior can be obtained by either specifying nothing in the target properties or saying:

    \n
      clock-sync: initial\n
    \n

    This results in clock synchronization being done during startup only. To account for the possibility of your clocks drifting during execution of the program, you can alternatively specify:

    \n
      clock-sync: on\n
    \n

    With this specification, in addition to synchronization during startup, synchronization will be redone periodically during program execution.

    \n

    Clock Synchronization Options

    \n

    A number of options can be specified using the clock-sync-options target parameter. For example:

    \n
      clock-sync-options: {local-federates-on: true, test-offset: 200 msec}\n
    \n

    The supported options are:

    \n
      \n
    • \n

      local-federates-on: Should be true or false. By default, if a federate is mapped to the same host as the RTI (using the at keyword), then clock synchronization is turned off. This assumes that the federate will be using the same clock as the RTI, so there is no point in performing clock synchronization. However, sometimes it is useful to force clock synchronization to be run even in this case, for example to test the performance of clock synchronization. To force clock synchronization on in this case, set this option to true.

      \n
    • \n
    • \n

      test-offset: The value should be a time value with units, e.g. 200 msec. This will establish an artificial fixed offset for each federate’s clock of one plus the federate ID times the time value given. For example, with the value 200 msec, a fixed offset of 200 milliseconds will be set on the clock for federate 0, 400 msec on the clock of federate 1, etc.

      \n
    • \n
    • \n

      period: A time value (with units) that specifies how often runtime clock synchronization will be performed if it is turned on. The default is 5 msec.

      \n
    • \n
    • \n

      attenuation: A positive integer specifying a divisor applied to the estimated clock error during runtime clock synchronization when adjusting the clock offset. The default is 10. Making this number bigger reduces each adjustment to the clock. Making the number equal to 1 means that each round of clock synchronization fully applies its estimated clock synchronization error.

      \n
    • \n
    • \n

      trials: The number of rounds of message exchange with the RTI in each clock synchronization round. This defaults to 10.

      \n
    • \n
    \n
    ","headings":[{"value":"Installation of the RTI","depth":2},{"value":"Minimal Example","depth":2},{"value":"Federation ID","depth":2},{"value":"Coordinated Start","depth":2},{"value":"Coordinated Shutdown","depth":2},{"value":"Communication Between Federates","depth":2},{"value":"Centralized Coordination","depth":2},{"value":"Decentralized Coordination","depth":2},{"value":"Physical Connections","depth":2},{"value":"Prerequisites for Distributed Execution","depth":2},{"value":"Specifying RTI Hosts","depth":2},{"value":"Specifying Federate Hosts","depth":2},{"value":"Clock Synchronization","depth":2},{"value":"Clock Synchronization Options","depth":3}],"frontmatter":{"permalink":"/docs/handbook/distributed-execution","title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Preambles","oneline":"Defining preambles in Lingua Franca.","permalink":"/docs/handbook/preambles"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Termination","oneline":"Terminating a Lingua Franca execution.","permalink":"/docs/handbook/termination"}}}},"pageContext":{"id":"1-distributed-execution","slug":"/docs/handbook/distributed-execution","repoPath":"/packages/documentation/copy/en/topics/Distributed Execution.md","previousID":"d9c76683-1fe5-55e0-b223-8e21c125f9cd","nextID":"ddf59040-674f-5833-a630-a62f39d0106e","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/eclipse-oomph/page-data.json b/page-data/docs/handbook/eclipse-oomph/page-data.json index b57973abe..55490bc66 100644 --- a/page-data/docs/handbook/eclipse-oomph/page-data.json +++ b/page-data/docs/handbook/eclipse-oomph/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/eclipse-oomph","result":{"data":{"markdownRemark":{"id":"d47b3353-7fbd-5138-a517-921fb5a960c3","excerpt":"Prerequisites Java 17 (download from Oracle) Each target language may have additional requirements. See the Target Language Details page and select your target…","html":"

    Prerequisites

    \n\n

    Note: Eclipse does not currently support Kotlin, the language used for some of the target code generators. If you plan to develop Kotlin code, we recommend using IntelliJ instead of Eclipse.

    \n

    Oomph Setup

    \n

    The Eclipse setup with Oomph allows to automatically create a fully configured Eclipse IDE for the development of Lingua Franca. Note that we recommend installing a new instance of Eclipse even if you already have one for other purposes. There is no problem having multiple Eclipse installations on the same machine, and separate installations help prevent cross-project problems.

    \n
      \n
    1. If you have previously installed Eclipse and you want to start fresh, then remove or move a hidden directory called .p2 in your home directory. I do this:
    2. \n
    \n
    mv ~/.p2 ~/.p2.bak\n
    \n
      \n
    1. \n

      Go to the Eclipse download site (https://www.eclipse.org/downloads/index.php) and download the Eclipse Installer for your platform. The site does not advertise that it ships the Oomph Eclipse Installer but downloading Eclipse with the orange download button will give you the installer.
      \nYou can skip this step if you already have the installer available on your system.

      \n
    2. \n
    3. \n

      Starting the installer for the first time will open a window that looks like the following (if you have previously followed these steps, skip to step 4):
      \n\n \n \n

      \n
    4. \n
    5. \n

      Next, we need to register the Lingua Franca specific setup in Oomph (only the first time you use the installer). Click the green Plus button at the top right corner. Select “GitHub Projects” as catalog and paste the following URL into the “Resource URI” field:\nhttps://raw.githubusercontent.com/icyphy/lingua-franca/master/oomph/LinguaFranca.setup.\nThen press OK.\nNOTE: to check out another branch instead, adjust the URL above accordingly. For instance, in order to install the setup from foo-bar branch, change the URL to https://raw.githubusercontent.com/icyphy/lingua-franca/foo-bar/oomph/LinguaFranca.setup. Also, in the subsequent screen in the wizard, select the particular branch of interest instead of default, which is master.

      \n
    6. \n
    7. \n

      Now Oomph lists the Lingua Franca setup in the ”” directory of the “GitHub Projects” catalog. Check the Lingua Franca entry. A new entry for Lingua Franca will appear in the table at the bottom of the window. Select Lingua Franca and click Next.
      \n\n \n \n

      \n
    8. \n
    9. \n

      Now you can further configure where and how your development Eclipse should be created. Check “Show all variables” to enable all possible configuration options. You can hover over the field labels to get a more detailed explanation of their effects.

      \n
    10. \n
    \n
      \n
    • If you already have cloned the LF repository and you want Eclipse to use this location instead of cloning it into the new IDE environment, you should adjust the “Git clone location rule”.
    • \n
    • Preferably, you have a GitHub account with an SSH key uploaded to GitHub. Otherwise, you should adjust the “Lingua Franca GitHub repository” entry to use the https option in the drop-down menu. See adding an SSH key to your GitHub account.
    • \n
    • If the “JRE 17 location” is empty, you need to install and/or locate a JDK that has at least version 17.\n\n \n \n
    • \n
    \n
      \n
    1. \n

      Click Next to get a summary of what will happen during installation. Click Finish to start.

      \n
    2. \n
    3. \n

      Once the basic installation is complete, your new Eclipse will start. If it fails to clone the GitHub repository, then you should use the back button in the Oomph dialog and change the way you are accessing the repo (ssh or https). See above.
      \nThe setup may also fail to clone the repository via SHH if Eclipse cannot find the private ssh key that matches the public key you uploaded to GitHub. You can configure the location of your private key in Eclipse as follows. In the Eclipse IDE, click the menu entry Window -> Preferences (on Mac Apple-Menu -> Preferences) and navigate to General -> Network Connections -> SSH2 in the tree view on the left and configure the SSH home directory and key names according to your computer. After the repo has been cloned, you can safely close the initial Oomph dialog (if not dismissed automatically). You will see a Welcome page that you can close.

      \n
    4. \n
    5. \n

      In the new Eclipse, it may automatically start building the project, or it may pop up an “Eclipse Updater” dialog. If neither happens, you can click the button with the yellow and blue cycling arrows in the status bar at the bottom. Oomph will perform various operations to configure the Eclipse environment, including the initial code generation for the LF language. This may take some time. Wait until the setup is finished.

      \n
    6. \n
    7. \n

      If you get compile errors, make sure Eclipse is using Java 17. If you skipped the first step above (removing your ~/.p2 directory), then you may have legacy configuration information that causes Eclipse to mysteriously use an earlier version of Java. Lingua Franca requires Java 17, and will get compiler errors if it uses an earlier version. To fix this, go to the menu Project->Properties and select Java Build Path. Remove the entry for JRE System Library [JRE for JavaSE-8] (or similar). Choose Add Library on the right, and choose JRE System Library. You should now be able to choose Workspace default JRE (JRE for JavaSE-17). A resulting rebuild should then compile correctly.

      \n
    8. \n
    9. \n

      When the setup dialog is closed, your LF development IDE is ready. Probably, Eclipse is still compiling some code but when this is finished as well, all error markers on the project should have disappeared. Now, you can start a runtime Eclipse to test the actual Lingua Franca end-user IDE. In the toolbar, click on the small arrow next to the green Start button. There may already be an entry named “Launch Runtime Eclipse”, but probably not. To create it, click on “Run Configurations…“. Expand the “Eclipse Application” entry, select “Launch Runtime Eclipse”, as follows:

      \n
    10. \n
    \n

    \n \n \n

    \n\n
  • \n

    Next, we need to register the Lingua Franca specific setup in Oomph (only the first time you use the installer). Click the green Plus button at the top right corner. Select “GitHub Projects” as catalog and paste the following URL into the “Resource URI” field:\nhttps://raw.githubusercontent.com/icyphy/lingua-franca/master/oomph/LinguaFranca.setup.\nThen press OK.\nNOTE: to check out another branch instead, adjust the URL above accordingly. For instance, in order to install the setup from foo-bar branch, change the URL to https://raw.githubusercontent.com/icyphy/lingua-franca/foo-bar/oomph/LinguaFranca.setup. Also, in the subsequent screen in the wizard, select the particular branch of interest instead of default, which is master.

    \n
  • \n
  • \n

    Now Oomph lists the Lingua Franca setup in the ”” directory of the “GitHub Projects” catalog. Check the Lingua Franca entry. A new entry for Lingua Franca will appear in the table at the bottom of the window. Select Lingua Franca and click Next.
    \n\n \n \n

    \n
  • \n
  • \n

    Now you can further configure where and how your development Eclipse should be created. Check “Show all variables” to enable all possible configuration options. You can hover over the field labels to get a more detailed explanation of their effects.

    \n
  • \n\n
      \n
    • If you already have cloned the LF repository and you want Eclipse to use this location instead of cloning it into the new IDE environment, you should adjust the “Git clone location rule”.
    • \n
    • Preferably, you have a GitHub account with an SSH key uploaded to GitHub. Otherwise, you should adjust the “Lingua Franca GitHub repository” entry to use the https option in the drop-down menu. See adding an SSH key to your GitHub account.
    • \n
    • If the “JRE 17 location” is empty, you need to install and/or locate a JDK that has at least version 17.\n\n \n \n
    • \n
    \n
      \n
    1. \n

      Click Next to get a summary of what will happen during installation. Click Finish to start.

      \n
    2. \n
    3. \n

      Once the basic installation is complete, your new Eclipse will start. If it fails to clone the GitHub repository, then you should use the back button in the Oomph dialog and change the way you are accessing the repo (ssh or https). See above.
      \nThe setup may also fail to clone the repository via SHH if Eclipse cannot find the private ssh key that matches the public key you uploaded to GitHub. You can configure the location of your private key in Eclipse as follows. In the Eclipse IDE, click the menu entry Window -> Preferences (on Mac Apple-Menu -> Preferences) and navigate to General -> Network Connections -> SSH2 in the tree view on the left and configure the SSH home directory and key names according to your computer. After the repo has been cloned, you can safely close the initial Oomph dialog (if not dismissed automatically). You will see a Welcome page that you can close.

      \n
    4. \n
    5. \n

      In the new Eclipse, it may automatically start building the project, or it may pop up an “Eclipse Updater” dialog. If neither happens, you can click the button with the yellow and blue cycling arrows in the status bar at the bottom. Oomph will perform various operations to configure the Eclipse environment, including the initial code generation for the LF language. This may take some time. Wait until the setup is finished.

      \n
    6. \n
    7. \n

      If you get compile errors, make sure Eclipse is using Java 17. If you skipped the first step above (removing your ~/.p2 directory), then you may have legacy configuration information that causes Eclipse to mysteriously use an earlier version of Java. Lingua Franca requires Java 17, and will get compiler errors if it uses an earlier version. To fix this, go to the menu Project->Properties and select Java Build Path. Remove the entry for JRE System Library [JRE for JavaSE-8] (or similar). Choose Add Library on the right, and choose JRE System Library. You should now be able to choose Workspace default JRE (JRE for JavaSE-17). A resulting rebuild should then compile correctly.

      \n
    8. \n
    9. \n

      When the setup dialog is closed, your LF development IDE is ready. Probably, Eclipse is still compiling some code but when this is finished as well, all error markers on the project should have disappeared. Now, you can start a runtime Eclipse to test the actual Lingua Franca end-user IDE. In the toolbar, click on the small arrow next to the green Start button. There may already be an entry named “Launch Runtime Eclipse”, but probably not. To create it, click on “Run Configurations…“. Expand the “Eclipse Application” entry, select “Launch Runtime Eclipse”, as follows:

      \n
    10. \n
    \n

    \n . Single-quoted literals must be exactly one character long —even in Python.\n

  • Boolean literals: true, false, True, False. The latter two are there for Python.
  • \n\n\n
  • \n

    Parameter references, which are simple identifiers (e.g. foo). Any identifier in expression position must refer to a parameter of the enclosing reactor.

    \n
  • \n
  • \n

    Time values, e.g. 1 msec or 10 seconds. The syntax of time values is integer time_unit, where time_unit is one of the following:

    \n
      \n
    • nsec or ns: nanoseconds
    • \n
    • usec or us: microseconds
    • \n
    • msec or ms: milliseconds
    • \n
    • sec, second, or s: seconds
    • \n
    • minute or min: 60 seconds
    • \n
    • hour: 60 minutes
    • \n
    • day: 24 hours
    • \n
    • week: 7 days
    • \n
    \n

    Each of these units also support a plural version (e.g., nsecs, minutes, and days), which means the same thing.

    \n

    The time value 0 need not be given a unit, but for all other values, the unit is required.

    \n

    Time values are compatible with the time type.

    \n
  • \n
  • \n

    Escaped target-language expression, e.g. {= foo() =}. This syntax is used to write any expression which does not fall into one of the other forms described here. The contents are not parsed and are used verbatim in the generated file.

    \n
  • \n\n
    \n

    For instance, to have a 2-dimensional array as a parameter in C:

    \n
    reactor Foo(param:{= int[][] =}({= { {1}, {2} } =})) {\n  ...\n}\n
    \n

    Both int[][] and { {1}, {2} } are C fragments here, not LF.

    \n
    \n
    \n

    For instance, to assign a 2-dimensional list as an initial value to a parameter\nin the Python target:

    \n
    reactor Foo(param({= ((1, 2, 3), (4, 5, 6)) =})) {\n  ...\n}\n
    \n
    \n

    Collections

    \n
    \n

    To avoid the awkwardness of using the code delimiters {= ... =}, Lingua Franca supports initialization of simple arrays and similar structures. The interpretation is slightly different in each target language.

    \n
    \n
    \n

    In C, a parameter or state may be given an array value as in the following example:

    \n
    reactor Foo(param: double[] = {0.0, 1.0, 2.0}) {\n  reaction(startup) {=\n    printf("%f %f %f\\n", self->param[0], self->param[1], self->param[2]);\n  =}\n}\n
    \n

    The parameter named param will become an array of length three. When instantiating this reactor, the default parameter value can be overridden using a similar syntax:

    \n
    main reactor {\n  f = new Foo(param = {3.3, 4.4, 5.5});\n}\n
    \n

    See the Target Language Details for details and alternative syntaxes.

    \n
    \n
    \n

    In C++, initial values for a parameter or state can be used to pass arguments to a constructor, as in the following example:

    \n
      state x: int[](2, 0);\n
    \n

    Here, the type int[] is translated by the code generator into std::vector and the (2, 0) to constructor arguments, as in new std::vector(2,0), which creates a vector of length 2 filled with elements with value 0. See the Target Language Details for details and alternative syntaxes.

    \n
    \n
    \n

    In Python, a parameter or state variable may be assigned a list expression as its initial value, as in the following example:

    \n
    reactor Foo(param = {= [1, 2, 3] =}) {\n  state x = {= [1, 2, 3] =}\n  ...\n}\n
    \n

    The param parameter and x state variable become Python lists.\nTheir elements may be accessed as arrays, for example self.x[i], where i is an array index.

    \n

    The parameter may be overridden with a different list at instantiation:

    \n
    main reactor {\n  f = new Foo(param = {= [3, 4, 5, 6]} )\n}\n
    \n

    See the Target Language Details for more details.

    \n
    \n
    \n

    In TypeScript, a parameter or state variable may be assigned an array expression as its initial value, as in the following example:

    \n
    reactor Foo(param:{=Array<number>=}({= [1, 2, 3] =})) {\n  state x:{=Array<number>=} = {= [0.1, 0.2, 0.3] =}\n}\n
    \n

    See the TypeScript reactor documentation for details and alternative syntaxes.

    \n
    \n
    \n

    FIXME: Rust

    \n
    ","headings":[{"value":"Basic expressions","depth":2},{"value":"Collections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/expressions","title":"Expressions","oneline":"Expressions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Target Language Details","oneline":"Detailed reference for each target langauge.","permalink":"/docs/handbook/target-language-details"}}}},"pageContext":{"id":"3-expressions","slug":"/docs/handbook/expressions","repoPath":"/packages/documentation/copy/en/reference/Expressions.md","nextID":"15e2a8e5-dd68-55e9-839f-18c6d342e73c","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/expressions","result":{"data":{"markdownRemark":{"id":"5002de77-4c7a-5740-bedf-9efa09e5ede6","excerpt":"$page-showing-target$ A subset of LF syntax is used to write expressions, which represent values in the target language. Expressions are used to initialize…","html":"

    $page-showing-target$

    \n

    A subset of LF syntax is used to write expressions, which represent values in the target language. Expressions are used to initialize state variable and to give values to parameters. Arbitrary expressions in the target language can always be given within delimiters {= ... =}, but simple forms do not require the delimiters. These simple forms are documented here.

    \n

    Basic expressions

    \n

    The most basic expression forms, which are supported by all target languages, are the following:

    \n
      \n
    • \n

      Literals:

      \n
        \n
      • Numeric literals, e.g. 1, -120, 1.5, 3.14e10. Note that the sign, if any, is part of the literal and must not be separated by whitespace.
      • \n
      • String literals, e.g. \"abcd\". String literals always use double-quotes, even in languages which support other forms (like Python).
      • \n
      • Character literals. e.g. 'a'. Single-quoted literals must be exactly one character long —even in Python.
      • \n
      • Boolean literals: true, false, True, False. The latter two are there for Python.
      • \n
      \n
    • \n
    • \n

      Parameter references, which are simple identifiers (e.g. foo). Any identifier in expression position must refer to a parameter of the enclosing reactor.

      \n
    • \n
    • \n

      Time values, e.g. 1 msec or 10 seconds. The syntax of time values is integer time_unit, where time_unit is one of the following:

      \n
        \n
      • nsec or ns: nanoseconds
      • \n
      • usec or us: microseconds
      • \n
      • msec or ms: milliseconds
      • \n
      • sec, second, or s: seconds
      • \n
      • minute or min: 60 seconds
      • \n
      • hour: 60 minutes
      • \n
      • day: 24 hours
      • \n
      • week: 7 days
      • \n
      \n

      Each of these units also support a plural version (e.g., nsecs, minutes, and days), which means the same thing.

      \n

      The time value 0 need not be given a unit, but for all other values, the unit is required.

      \n

      Time values are compatible with the time type.

      \n
    • \n
    • \n

      Escaped target-language expression, e.g. {= foo() =}. This syntax is used to write any expression which does not fall into one of the other forms described here. The contents are not parsed and are used verbatim in the generated file.

      \n
    • \n
    \n
    \n

    For instance, to have a 2-dimensional array as a parameter in C:

    \n
    reactor Foo(param:{= int[][] =}({= { {1}, {2} } =})) {\n  ...\n}\n
    \n

    Both int[][] and { {1}, {2} } are C fragments here, not LF.

    \n
    \n
    \n

    For instance, to assign a 2-dimensional list as an initial value to a parameter\nin the Python target:

    \n
    reactor Foo(param({= ((1, 2, 3), (4, 5, 6)) =})) {\n  ...\n}\n
    \n
    \n

    Collections

    \n
    \n

    To avoid the awkwardness of using the code delimiters {= ... =}, Lingua Franca supports initialization of simple arrays and similar structures. The interpretation is slightly different in each target language.

    \n
    \n
    \n

    In C, a parameter or state may be given an array value as in the following example:

    \n
    reactor Foo(param: double[] = {0.0, 1.0, 2.0}) {\n  reaction(startup) {=\n    printf("%f %f %f\\n", self->param[0], self->param[1], self->param[2]);\n  =}\n}\n
    \n

    The parameter named param will become an array of length three. When instantiating this reactor, the default parameter value can be overridden using a similar syntax:

    \n
    main reactor {\n  f = new Foo(param = {3.3, 4.4, 5.5});\n}\n
    \n

    See the Target Language Details for details and alternative syntaxes.

    \n
    \n
    \n

    In C++, initial values for a parameter or state can be used to pass arguments to a constructor, as in the following example:

    \n
      state x: int[](2, 0);\n
    \n

    Here, the type int[] is translated by the code generator into std::vector and the (2, 0) to constructor arguments, as in new std::vector(2,0), which creates a vector of length 2 filled with elements with value 0. See the Target Language Details for details and alternative syntaxes.

    \n
    \n
    \n

    In Python, a parameter or state variable may be assigned a list expression as its initial value, as in the following example:

    \n
    reactor Foo(param = {= [1, 2, 3] =}) {\n  state x = {= [1, 2, 3] =}\n  ...\n}\n
    \n

    The param parameter and x state variable become Python lists.\nTheir elements may be accessed as arrays, for example self.x[i], where i is an array index.

    \n

    The parameter may be overridden with a different list at instantiation:

    \n
    main reactor {\n  f = new Foo(param = {= [3, 4, 5, 6]} )\n}\n
    \n

    See the Target Language Details for more details.

    \n
    \n
    \n

    In TypeScript, a parameter or state variable may be assigned an array expression as its initial value, as in the following example:

    \n
    reactor Foo(param:{=Array<number>=}({= [1, 2, 3] =})) {\n  state x:{=Array<number>=} = {= [0.1, 0.2, 0.3] =}\n}\n
    \n

    See the TypeScript reactor documentation for details and alternative syntaxes.

    \n
    \n
    \n

    FIXME: Rust

    \n
    ","headings":[{"value":"Basic expressions","depth":2},{"value":"Collections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/expressions","title":"Expressions","oneline":"Expressions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Target Language Details","oneline":"Detailed reference for each target langauge.","permalink":"/docs/handbook/target-language-details"}}}},"pageContext":{"id":"3-expressions","slug":"/docs/handbook/expressions","repoPath":"/packages/documentation/copy/en/reference/Expressions.md","nextID":"15e2a8e5-dd68-55e9-839f-18c6d342e73c","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/extending-reactors/page-data.json b/page-data/docs/handbook/extending-reactors/page-data.json index 07500374d..bf51379b5 100644 --- a/page-data/docs/handbook/extending-reactors/page-data.json +++ b/page-data/docs/handbook/extending-reactors/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/extending-reactors","result":{"data":{"markdownRemark":{"id":"2efb41b1-0afb-5eab-9d59-7f064a8b778e","excerpt":"$page-showing-target$ Extending a Base Reactor The Cpp target does not yet support extending reactors. Lingua Franca supports defining a reactor class as an…","html":"

    $page-showing-target$

    \n

    Extending a Base Reactor

    \n
    \n

    The Cpp target does not yet support extending reactors.

    \n
    \n
    \n

    Lingua Franca supports defining a reactor class as an extension (or subclass), as in the following example:

    \n

    $start(Extends)$

    \n
    target C;\nreactor A {\n  input a:int;\n  output out:int;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n}\nreactor B extends A {\n  input b:int;\n  reaction(a, b) -> out {=\n    lf_set(out, a->value + b->value);\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Extends.lf\n// the cpp target currently does not support reactor extends\n
    \n
    target Python;\nreactor A {\n  input a;\n  output out;\n  reaction(a) -> out {=\n    out.set(a.value)\n  =}\n}\nreactor B extends A {\n  input b;\n  reaction(a, b) -> out {=\n    out.set(a.value + b.value)\n  =}\n}\n
    \n
    target TypeScript\nreactor A {\n  input a:number\n  output out:number\n  reaction(a) -> out {=\n    out = a\n  =}\n}\nreactor B extends A {\n  input b:number\n  reaction(a, b) -> out {=\n    out = a + b\n  =}\n}\n
    \n
    target Rust;\nreactor A {\n  input a:u32;\n  output out:u32;\n  reaction(a) -> out {=\n    ctx.set(out, ctx.get(a).unwrap());\n  =}\n}\nreactor B extends A {\n  input b:u32;\n  reaction(a, b) -> out {=\n    ctx.set(out, ctx.get(a).unwrap() + ctx.get(b).unwrap());\n  =}\n}\n
    \n

    $end(Extends)$

    \n\"Lingua\n

    Here, the base class A has a single output that it writes to in reaction to an input. The subclass inherits the input, the output, and the reaction of A, and adds its own input b and reaction. When an input event a arrives, both reactions will be invoked, but, once again, in a well-defined order. The reactions of the base class are invoked before those of the derived class. So in this case, B will overwrite the output produced by A.

    \n

    One limitation is that a subclass cannot have ports, actions, or state variables with the same names as those in the base class. The names must be unique.

    \n

    A subclass can extend more than one base class by just providing a comma-separated list of base classes. If reactions in multiple base classes are triggered at the same tag, they will be invoked in the same order that they appear in the comma-separated list.

    \n
    ","headings":[{"value":"Extending a Base Reactor","depth":2}],"frontmatter":{"permalink":"/docs/handbook/extending-reactors","title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","permalink":"/docs/handbook/causality-loops"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Actions","oneline":"Actions in Lingua Franca.","permalink":"/docs/handbook/actions"}}}},"pageContext":{"id":"1-extending-reactors","slug":"/docs/handbook/extending-reactors","repoPath":"/packages/documentation/copy/en/topics/Extending Reactors.md","previousID":"de5af6b0-de72-5890-9668-c4f000ffdb2c","nextID":"ab880406-6c38-59c6-9a2c-a8f736013224","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/extending-reactors","result":{"data":{"markdownRemark":{"id":"2efb41b1-0afb-5eab-9d59-7f064a8b778e","excerpt":"$page-showing-target$ Extending a Base Reactor The Cpp target does not yet support extending reactors. Lingua Franca supports defining a reactor class as an…","html":"

    $page-showing-target$

    \n

    Extending a Base Reactor

    \n
    \n

    The Cpp target does not yet support extending reactors.

    \n
    \n
    \n

    Lingua Franca supports defining a reactor class as an extension (or subclass), as in the following example:

    \n

    $start(Extends)$

    \n
    target C;\nreactor A {\n  input a:int;\n  output out:int;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n}\nreactor B extends A {\n  input b:int;\n  reaction(a, b) -> out {=\n    lf_set(out, a->value + b->value);\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Extends.lf\n// the cpp target currently does not support reactor extends\n
    \n
    target Python;\nreactor A {\n  input a;\n  output out;\n  reaction(a) -> out {=\n    out.set(a.value)\n  =}\n}\nreactor B extends A {\n  input b;\n  reaction(a, b) -> out {=\n    out.set(a.value + b.value)\n  =}\n}\n
    \n
    target TypeScript\nreactor A {\n  input a:number\n  output out:number\n  reaction(a) -> out {=\n    out = a\n  =}\n}\nreactor B extends A {\n  input b:number\n  reaction(a, b) -> out {=\n    out = a + b\n  =}\n}\n
    \n
    target Rust;\nreactor A {\n  input a:u32;\n  output out:u32;\n  reaction(a) -> out {=\n    ctx.set(out, ctx.get(a).unwrap());\n  =}\n}\nreactor B extends A {\n  input b:u32;\n  reaction(a, b) -> out {=\n    ctx.set(out, ctx.get(a).unwrap() + ctx.get(b).unwrap());\n  =}\n}\n
    \n

    $end(Extends)$

    \n\"Lingua\n

    Here, the base class A has a single output that it writes to in reaction to an input. The subclass inherits the input, the output, and the reaction of A, and adds its own input b and reaction. When an input event a arrives, both reactions will be invoked, but, once again, in a well-defined order. The reactions of the base class are invoked before those of the derived class. So in this case, B will overwrite the output produced by A.

    \n

    One limitation is that a subclass cannot have ports, actions, or state variables with the same names as those in the base class. The names must be unique.

    \n

    A subclass can extend more than one base class by just providing a comma-separated list of base classes. If reactions in multiple base classes are triggered at the same tag, they will be invoked in the same order that they appear in the comma-separated list.

    \n
    ","headings":[{"value":"Extending a Base Reactor","depth":2}],"frontmatter":{"permalink":"/docs/handbook/extending-reactors","title":"Extending Reactors","oneline":"Extending reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","permalink":"/docs/handbook/causality-loops"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Actions","oneline":"Actions in Lingua Franca.","permalink":"/docs/handbook/actions"}}}},"pageContext":{"id":"1-extending-reactors","slug":"/docs/handbook/extending-reactors","repoPath":"/packages/documentation/copy/en/topics/Extending Reactors.md","previousID":"de5af6b0-de72-5890-9668-c4f000ffdb2c","nextID":"ab880406-6c38-59c6-9a2c-a8f736013224","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/features/page-data.json b/page-data/docs/handbook/features/page-data.json index 3acecde0f..281a784ba 100644 --- a/page-data/docs/handbook/features/page-data.json +++ b/page-data/docs/handbook/features/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/features","result":{"data":{"markdownRemark":{"id":"f6a7c690-cb55-559f-8884-d60c4b264ed6","excerpt":"Target/Feature Banks & Multiports Clock Synchronization Federation C Y Y Y C++ Y N N TS Y N N Python Y N N","html":"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
    Target/FeatureBanks & MultiportsClock SynchronizationFederation
    CYYY
    C++YNN
    TSYNN
    PythonYNN
    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/features","title":"Target-Supported Features","oneline":"Which features are supported by which target?","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/features","repoPath":"/packages/documentation/copy/en/preliminary/Target-Supported Features.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/features","result":{"data":{"markdownRemark":{"id":"f6a7c690-cb55-559f-8884-d60c4b264ed6","excerpt":"Target/Feature Banks & Multiports Clock Synchronization Federation C Y Y Y C++ Y N N TS Y N N Python Y N N","html":"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
    Target/FeatureBanks & MultiportsClock SynchronizationFederation
    CYYY
    C++YNN
    TSYNN
    PythonYNN
    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/features","title":"Target-Supported Features","oneline":"Which features are supported by which target?","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/features","repoPath":"/packages/documentation/copy/en/preliminary/Target-Supported Features.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/generic-types-interfaces-inheritance/page-data.json b/page-data/docs/handbook/generic-types-interfaces-inheritance/page-data.json index 1feda41ac..2b282d2b1 100644 --- a/page-data/docs/handbook/generic-types-interfaces-inheritance/page-data.json +++ b/page-data/docs/handbook/generic-types-interfaces-inheritance/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/generic-types-interfaces-inheritance","result":{"data":{"markdownRemark":{"id":"39dfe06d-8d92-5b0f-865d-1ef73107759b","excerpt":"The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals. Generics Reactor classes can be…","html":"

    The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals.

    \n

    Generics

    \n

    Reactor classes can be parameterized with type parameters as follows:

    \n
    reactor Foo<S, T> {\n    input:S;\n    output:T;\n}\n
    \n

    Type Constraints

    \n

    We could like to combine generics with type constraints of the form S extends Bar, where Bar refers to a reactor class or interface. The meaning of extending or implementing a reactor class will mean something slightly different from what this means in the target language — even if it features object orientation (OO).

    \n

    Interfaces

    \n

    While initially being tempted to distinguish interfaces from implementations, in an effort to promote simplicity, we (at least for the moment) propose not to. Only in case reactions and their signatures would be part of an interface and thus should be declared (without supplying an implementation) would there be a material difference between an interface and its implementation. Making reactions and their causality interfaces part of the reactor could prove useful, but it introduces a number of complications:

    \n
      \n
    • \n
    \n

    Inheritance

    \n
      \n
    • A reactor can extend multiple base classes;
    • \n
    • Reactions are inherited in the order of declaration; and
    • \n
    • Equally-named ports and actions between subclass and superclass must also be equally typed.
    • \n
    \n

    Example

    ","headings":[{"value":"Generics","depth":1},{"value":"Type Constraints","depth":2},{"value":"Interfaces","depth":1},{"value":"Inheritance","depth":1},{"value":"Example","depth":2}],"frontmatter":{"permalink":"/docs/handbook/generic-types-interfaces-inheritance","title":"Generic Types, Interfaces, and Inheritance","oneline":"Generic Types, Interfaces, and Inheritance (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/generic-types-interfaces-inheritance","repoPath":"/packages/documentation/copy/en/preliminary/Generic Types, Interfaces, and Inheritance.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/generic-types-interfaces-inheritance","result":{"data":{"markdownRemark":{"id":"39dfe06d-8d92-5b0f-865d-1ef73107759b","excerpt":"The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals. Generics Reactor classes can be…","html":"

    The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals.

    \n

    Generics

    \n

    Reactor classes can be parameterized with type parameters as follows:

    \n
    reactor Foo<S, T> {\n    input:S;\n    output:T;\n}\n
    \n

    Type Constraints

    \n

    We could like to combine generics with type constraints of the form S extends Bar, where Bar refers to a reactor class or interface. The meaning of extending or implementing a reactor class will mean something slightly different from what this means in the target language — even if it features object orientation (OO).

    \n

    Interfaces

    \n

    While initially being tempted to distinguish interfaces from implementations, in an effort to promote simplicity, we (at least for the moment) propose not to. Only in case reactions and their signatures would be part of an interface and thus should be declared (without supplying an implementation) would there be a material difference between an interface and its implementation. Making reactions and their causality interfaces part of the reactor could prove useful, but it introduces a number of complications:

    \n
      \n
    • \n
    \n

    Inheritance

    \n
      \n
    • A reactor can extend multiple base classes;
    • \n
    • Reactions are inherited in the order of declaration; and
    • \n
    • Equally-named ports and actions between subclass and superclass must also be equally typed.
    • \n
    \n

    Example

    ","headings":[{"value":"Generics","depth":1},{"value":"Type Constraints","depth":2},{"value":"Interfaces","depth":1},{"value":"Inheritance","depth":1},{"value":"Example","depth":2}],"frontmatter":{"permalink":"/docs/handbook/generic-types-interfaces-inheritance","title":"Generic Types, Interfaces, and Inheritance","oneline":"Generic Types, Interfaces, and Inheritance (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/generic-types-interfaces-inheritance","repoPath":"/packages/documentation/copy/en/preliminary/Generic Types, Interfaces, and Inheritance.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/generics/page-data.json b/page-data/docs/handbook/generics/page-data.json index 4e4849816..676657f7a 100644 --- a/page-data/docs/handbook/generics/page-data.json +++ b/page-data/docs/handbook/generics/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/generics","result":{"data":{"markdownRemark":{"id":"efe15ca4-8dc6-5216-8a43-0d7ea216c339","excerpt":"Generic reactors are not supported in $target-language$. $page-showing-target$ Generic Reactors Sometimes it is useful to implement a generic pattern without…","html":"
    \n

    Generic reactors are not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    Generic Reactors

    \n

    Sometimes it is useful to implement a generic pattern without knowing the concrete types used. For instance, it could be useful to implement a delay reactor that forwards all values it receives with a fixed delay, regardless of their datatype. For this pattern, it is not required to know the concrete type in advance, and we would like to reuse the same logic for different types. This can be achieved with generic reactors in LF. Consider the following example:

    \n

    $start(GenericDelay)$

    \n
    target C\nreactor Delay<T>(delay: time = 0) {\n  input in: T\n  output out: T\n  logical action a(delay): T\n  reaction(a) -> out {= lf_set(out, a->value); =}\n  reaction(in) -> a {= lf_schedule_copy(a, self->delay, &in->value, 1); =}\n}\nmain reactor {\n  d = new Delay<int>(delay = 100 ms)\n  reaction(startup) -> d.in {= lf_set(d.in, 42); =}\n  reaction(d.out) {=\n    printf("Received %d at time %lld.\\n", d.out->value, lf_time_logical_elapsed());\n  =}\n}\n
    \n
    target Cpp\nreactor Delay<T>(delay: time = 0) {\n  input in: T\n  output out: T\n  logical action a(delay): T\n  reaction(a) -> out {= out.set(a.get()); =}\n  reaction(in) -> a {= a.schedule(in.get(), delay); =}\n}\nmain reactor {\n  d = new Delay<int>(delay = 100 ms)\n  reaction(startup) -> d.in {= d.in.set(42); =}\n  reaction(d.out) {=\n    std::cout << "received " << *d.out.get() << " at time "\n        << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/py/src/GenericDelay.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/GenericDelay.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/GenericDelay.lf\n
    \n

    $end(GenericDelay)$

    \n

    The example above defines a generic reactor Delay which has a type parameter named T. Its input, output and logical action are all of type T. The logic implemented in the reactions is straightforward. The reaction to in schedules the logical action a with the configured delay and the received value. The reaction to a simply forwards this value to the output port at a later tag. The concrete type T, however, is not relevant for this implementation and will be filled in only when the reactor is instantiated. In our example, the main reactor instantiates Delay, specifying int as the type to be assigned to T. As a consequence, we can set an integer on d’s input port and receive an integer on its output. If we wanted instead to delay a string, we can do this as follows:

    \n

    $start(GenericString)$

    \n
    target C\nimport Delay from "GenericDelay.lf"\nmain reactor {\n  d = new Delay<string>(delay = 100 ms)\n  reaction(startup) -> d.in {=\n    lf_set(d.in, "foo");\n  =}\n  reaction(d.out) {=\n    printf("Received %s at time %lld.\\n", d.out->value, lf_time_logical_elapsed());\n  =}\n}\n
    \n
    target Cpp\nimport Delay from "GenericDelay.lf"\nmain reactor {\n  d = new Delay<{= std::string =}>(delay = 100 ms)\n  reaction(startup) -> d.in {=\n    d.in.set("foo");\n  =}\n  reaction(d.out) {=\n    std::cout << "received " << *d.out.get() << " at time "\n        << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/py/src/GenericString.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/GenericString.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/GenericString.lf\n
    \n

    $end(GenericString)$

    \n

    Reactor definitions may also specify multiple type parameters. Moreover, type parameters are not limited to ports and actions, but can also be used in state variables, parameters, or methods. For instance, we can define the following reactor:

    \n
    reactor Generic<T, U, V>(bar: T) {\n  state baz: U\n\n  input in: V\n\n  method (x: T, y: U): V {= /* ... */ =}\n}\n
    \n

    This reactor could be instantiated for example like this:

    \n
    g = new Generic<float, int, bool>()\n
    \n
    ","headings":[{"value":"Generic Reactors","depth":2}],"frontmatter":{"permalink":"/docs/handbook/generics","title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","permalink":"/docs/handbook/multiports-and-banks"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Preambles","oneline":"Defining preambles in Lingua Franca.","permalink":"/docs/handbook/preambles"}}}},"pageContext":{"id":"1-generic-reactors","slug":"/docs/handbook/generics","repoPath":"/packages/documentation/copy/en/topics/Generics.md","previousID":"9ff63bbf-2bdf-553e-a96d-52355866ec94","nextID":"d9c76683-1fe5-55e0-b223-8e21c125f9cd","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/generics","result":{"data":{"markdownRemark":{"id":"efe15ca4-8dc6-5216-8a43-0d7ea216c339","excerpt":"Generic reactors are not supported in $target-language$. $page-showing-target$ Generic Reactors Sometimes it is useful to implement a generic pattern without…","html":"
    \n

    Generic reactors are not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    Generic Reactors

    \n

    Sometimes it is useful to implement a generic pattern without knowing the concrete types used. For instance, it could be useful to implement a delay reactor that forwards all values it receives with a fixed delay, regardless of their datatype. For this pattern, it is not required to know the concrete type in advance, and we would like to reuse the same logic for different types. This can be achieved with generic reactors in LF. Consider the following example:

    \n

    $start(GenericDelay)$

    \n
    target C\nreactor Delay<T>(delay: time = 0) {\n  input in: T\n  output out: T\n  logical action a(delay): T\n  reaction(a) -> out {= lf_set(out, a->value); =}\n  reaction(in) -> a {= lf_schedule_copy(a, self->delay, &in->value, 1); =}\n}\nmain reactor {\n  d = new Delay<int>(delay = 100 ms)\n  reaction(startup) -> d.in {= lf_set(d.in, 42); =}\n  reaction(d.out) {=\n    printf("Received %d at time %lld.\\n", d.out->value, lf_time_logical_elapsed());\n  =}\n}\n
    \n
    target Cpp\nreactor Delay<T>(delay: time = 0) {\n  input in: T\n  output out: T\n  logical action a(delay): T\n  reaction(a) -> out {= out.set(a.get()); =}\n  reaction(in) -> a {= a.schedule(in.get(), delay); =}\n}\nmain reactor {\n  d = new Delay<int>(delay = 100 ms)\n  reaction(startup) -> d.in {= d.in.set(42); =}\n  reaction(d.out) {=\n    std::cout << "received " << *d.out.get() << " at time "\n        << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/py/src/GenericDelay.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/GenericDelay.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/GenericDelay.lf\n
    \n

    $end(GenericDelay)$

    \n

    The example above defines a generic reactor Delay which has a type parameter named T. Its input, output and logical action are all of type T. The logic implemented in the reactions is straightforward. The reaction to in schedules the logical action a with the configured delay and the received value. The reaction to a simply forwards this value to the output port at a later tag. The concrete type T, however, is not relevant for this implementation and will be filled in only when the reactor is instantiated. In our example, the main reactor instantiates Delay, specifying int as the type to be assigned to T. As a consequence, we can set an integer on d’s input port and receive an integer on its output. If we wanted instead to delay a string, we can do this as follows:

    \n

    $start(GenericString)$

    \n
    target C\nimport Delay from "GenericDelay.lf"\nmain reactor {\n  d = new Delay<string>(delay = 100 ms)\n  reaction(startup) -> d.in {=\n    lf_set(d.in, "foo");\n  =}\n  reaction(d.out) {=\n    printf("Received %s at time %lld.\\n", d.out->value, lf_time_logical_elapsed());\n  =}\n}\n
    \n
    target Cpp\nimport Delay from "GenericDelay.lf"\nmain reactor {\n  d = new Delay<{= std::string =}>(delay = 100 ms)\n  reaction(startup) -> d.in {=\n    d.in.set("foo");\n  =}\n  reaction(d.out) {=\n    std::cout << "received " << *d.out.get() << " at time "\n        << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/py/src/GenericString.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/GenericString.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/GenericString.lf\n
    \n

    $end(GenericString)$

    \n

    Reactor definitions may also specify multiple type parameters. Moreover, type parameters are not limited to ports and actions, but can also be used in state variables, parameters, or methods. For instance, we can define the following reactor:

    \n
    reactor Generic<T, U, V>(bar: T) {\n  state baz: U\n\n  input in: V\n\n  method (x: T, y: U): V {= /* ... */ =}\n}\n
    \n

    This reactor could be instantiated for example like this:

    \n
    g = new Generic<float, int, bool>()\n
    \n
    ","headings":[{"value":"Generic Reactors","depth":2}],"frontmatter":{"permalink":"/docs/handbook/generics","title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","permalink":"/docs/handbook/multiports-and-banks"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Preambles","oneline":"Defining preambles in Lingua Franca.","permalink":"/docs/handbook/preambles"}}}},"pageContext":{"id":"1-generic-reactors","slug":"/docs/handbook/generics","repoPath":"/packages/documentation/copy/en/topics/Generics.md","previousID":"9ff63bbf-2bdf-553e-a96d-52355866ec94","nextID":"d9c76683-1fe5-55e0-b223-8e21c125f9cd","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/import-system/page-data.json b/page-data/docs/handbook/import-system/page-data.json index 20a9eda93..fa8539684 100644 --- a/page-data/docs/handbook/import-system/page-data.json +++ b/page-data/docs/handbook/import-system/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/import-system","result":{"data":{"markdownRemark":{"id":"4f193470-c244-5392-8875-25389d7cd383","excerpt":"The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals. Current Implementation of…","html":"

    The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals.

    \n

    Current Implementation of Imports

    \n

    The import functionality in Lingua Franca is limited to:

    \n
    import HelloWorld.lf
    \n

    This can be useful if the .lf file is located in the same directory as the file containing the main reactor.

    \n

    However, several shortcomings exist in this current system which we shall discuss next.

    \n

    Duplicate Reactor Names

    \n

    Reactors with the same name can cause issues. For example:

    \n
    import CatsAndPuppies.lf // Contains a Puppy reactor\nimport MeanPuppies.lf   // Contains another Puppy reactor
    \n

    There is no way for the LF program to distinguish between the two Puppy reactors.

    \n

    Note. With a relatively trivial extension to the current LF import mechanism, it is possible to detect duplicates, but there is no way to circumvent them in the current LF program (i.e., the original names might have to be changed).

    \n

    Selective Importing

    \n

    Selective importing is not possible. For example, using

    \n
    import CatsAndPuppies.lf
    \n

    will import all the reactors contained in the .lf file. It would be desirable to selectively import a subset of reactors in another .lf file.

    \n

    Qualified Paths

    \n

    Currently, there is no elegant way of importing modules that are not in the same directory.

    \n

    Renaming

    \n

    All the reactors imported will have the name originally given to them by the original programmer. It might make sense to rename them for the current LF program.

    \n

    Packages

    \n

    With the current import solution that only uses files, implementing packages in Lingua Franca is not feasible.

    \n

    Proposed Solution

    \n

    With inspirations from Python, we propose the following import mechanism:

    \n
    "import" LF_Trunc_File/module ("," LF_Trunc_File/module)* \n              | "from" LFTruncFile/module "import" reactor ["as" name]\n                ("," reactor ["as" name] )*\n              | "from" LF_Trunc_File/module "import" "*" 
    \n

    Before discussing some examples, let’s discuss LF_Trunc_File/module. First and foremost, LF_Truc_File stands for Lingua Franca Truncated File, which is a name.lf file with the .lf removed. Therefore, the legacy support for import can be carried over as:

    \n
    import HelloWorld
    \n

    Second, the module would introduce the notion of packages to Lingua Franca. The content of a module can be located in any path. To enable this facility, modules provide a Lingua Franca Meta file (LFM) that introduces the package name, and the absolute or relative paths of all the LF files that are included in that package. For example:

    \n
    // CatsAndPuppies.LFM\npackage CatsAndPuppies // Optional. The file name would be interpreted as the package name.\nimport /home/user/linguafranca/pets/Cats.lf // Absolute paths\nimport pets/Puppies.lf // Relative paths
    \n

    For a package to be accessible, the LFM file needs to be discoverable. For example, it can be automatically added to the current directory or “installed” in a known Lingua Franca path (e.g., /usr/local/LF/packages or /home/user/linguafranca/packages).

    \n

    With that in mind, let’s discuss some examples on how this might work next.\nThe content of the HelloWorld.lf example is as follows:

    \n
    target C; \nreactor SayHello {\n    timer t;\n    reaction(t) {=\n        printf("Hello World.\\n");\n    =}\n}\nmain reactor HelloWorldTest {\n    a = new HelloWorld();\n}
    \n

    Let us create a Greetings.lf program based on HelloWorld.

    \n
    target C; \nimport HelloWorld\n\nmain reactor Greetings {\n    a = new SayHello();\n}
    \n

    To generate code for Greetings.lf, Lingua Franca first searches for a HelloWorld.lf file in the same directory as Greetings.lf. If not found, it will look for a HelloWorld.LFM in the known paths. If none is found, an error is raised.

    \n

    Now we can demonstrate selective import. For example:

    \n
    target C; \nfrom HelloWorld import SayHello\n\nmain reactor Greetings {\n    a = new SayHello();\n}
    \n

    Finally, renaming can be done by using the as predicate:

    \n
    target C; \nfrom HelloWorld import SayHello as SayGreetings\n\nmain reactor Greetings {\n    a = new SayHeGreetings();\n}
    ","headings":[{"value":"Current Implementation of Imports","depth":1},{"value":"Duplicate Reactor Names","depth":2},{"value":"Selective Importing","depth":2},{"value":"Qualified Paths","depth":2},{"value":"Renaming","depth":2},{"value":"Packages","depth":2},{"value":"Proposed Solution","depth":1}],"frontmatter":{"permalink":"/docs/handbook/import-system","title":"Import System","oneline":"Import System (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/import-system","repoPath":"/packages/documentation/copy/en/preliminary/Import System.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/import-system","result":{"data":{"markdownRemark":{"id":"4f193470-c244-5392-8875-25389d7cd383","excerpt":"The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals. Current Implementation of…","html":"

    The following topics are meant as collections of design ideas, with the purpose of refining them into concrete design proposals.

    \n

    Current Implementation of Imports

    \n

    The import functionality in Lingua Franca is limited to:

    \n
    import HelloWorld.lf
    \n

    This can be useful if the .lf file is located in the same directory as the file containing the main reactor.

    \n

    However, several shortcomings exist in this current system which we shall discuss next.

    \n

    Duplicate Reactor Names

    \n

    Reactors with the same name can cause issues. For example:

    \n
    import CatsAndPuppies.lf // Contains a Puppy reactor\nimport MeanPuppies.lf   // Contains another Puppy reactor
    \n

    There is no way for the LF program to distinguish between the two Puppy reactors.

    \n

    Note. With a relatively trivial extension to the current LF import mechanism, it is possible to detect duplicates, but there is no way to circumvent them in the current LF program (i.e., the original names might have to be changed).

    \n

    Selective Importing

    \n

    Selective importing is not possible. For example, using

    \n
    import CatsAndPuppies.lf
    \n

    will import all the reactors contained in the .lf file. It would be desirable to selectively import a subset of reactors in another .lf file.

    \n

    Qualified Paths

    \n

    Currently, there is no elegant way of importing modules that are not in the same directory.

    \n

    Renaming

    \n

    All the reactors imported will have the name originally given to them by the original programmer. It might make sense to rename them for the current LF program.

    \n

    Packages

    \n

    With the current import solution that only uses files, implementing packages in Lingua Franca is not feasible.

    \n

    Proposed Solution

    \n

    With inspirations from Python, we propose the following import mechanism:

    \n
    "import" LF_Trunc_File/module ("," LF_Trunc_File/module)* \n              | "from" LFTruncFile/module "import" reactor ["as" name]\n                ("," reactor ["as" name] )*\n              | "from" LF_Trunc_File/module "import" "*" 
    \n

    Before discussing some examples, let’s discuss LF_Trunc_File/module. First and foremost, LF_Truc_File stands for Lingua Franca Truncated File, which is a name.lf file with the .lf removed. Therefore, the legacy support for import can be carried over as:

    \n
    import HelloWorld
    \n

    Second, the module would introduce the notion of packages to Lingua Franca. The content of a module can be located in any path. To enable this facility, modules provide a Lingua Franca Meta file (LFM) that introduces the package name, and the absolute or relative paths of all the LF files that are included in that package. For example:

    \n
    // CatsAndPuppies.LFM\npackage CatsAndPuppies // Optional. The file name would be interpreted as the package name.\nimport /home/user/linguafranca/pets/Cats.lf // Absolute paths\nimport pets/Puppies.lf // Relative paths
    \n

    For a package to be accessible, the LFM file needs to be discoverable. For example, it can be automatically added to the current directory or “installed” in a known Lingua Franca path (e.g., /usr/local/LF/packages or /home/user/linguafranca/packages).

    \n

    With that in mind, let’s discuss some examples on how this might work next.\nThe content of the HelloWorld.lf example is as follows:

    \n
    target C; \nreactor SayHello {\n    timer t;\n    reaction(t) {=\n        printf("Hello World.\\n");\n    =}\n}\nmain reactor HelloWorldTest {\n    a = new HelloWorld();\n}
    \n

    Let us create a Greetings.lf program based on HelloWorld.

    \n
    target C; \nimport HelloWorld\n\nmain reactor Greetings {\n    a = new SayHello();\n}
    \n

    To generate code for Greetings.lf, Lingua Franca first searches for a HelloWorld.lf file in the same directory as Greetings.lf. If not found, it will look for a HelloWorld.LFM in the known paths. If none is found, an error is raised.

    \n

    Now we can demonstrate selective import. For example:

    \n
    target C; \nfrom HelloWorld import SayHello\n\nmain reactor Greetings {\n    a = new SayHello();\n}
    \n

    Finally, renaming can be done by using the as predicate:

    \n
    target C; \nfrom HelloWorld import SayHello as SayGreetings\n\nmain reactor Greetings {\n    a = new SayHeGreetings();\n}
    ","headings":[{"value":"Current Implementation of Imports","depth":1},{"value":"Duplicate Reactor Names","depth":2},{"value":"Selective Importing","depth":2},{"value":"Qualified Paths","depth":2},{"value":"Renaming","depth":2},{"value":"Packages","depth":2},{"value":"Proposed Solution","depth":1}],"frontmatter":{"permalink":"/docs/handbook/import-system","title":"Import System","oneline":"Import System (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/import-system","repoPath":"/packages/documentation/copy/en/preliminary/Import System.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/inputs-and-outputs/page-data.json b/page-data/docs/handbook/inputs-and-outputs/page-data.json index 2a8998448..366df8dc7 100644 --- a/page-data/docs/handbook/inputs-and-outputs/page-data.json +++ b/page-data/docs/handbook/inputs-and-outputs/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/inputs-and-outputs","result":{"data":{"markdownRemark":{"id":"28c65d97-e18a-53ae-8b3a-7e4caea32061","excerpt":"$page-showing-target$ In this section, we will endow reactors with inputs and outputs. Input and Output Declarations Input and output declarations have the form…","html":"

    $page-showing-target$

    \n

    In this section, we will endow reactors with inputs and outputs.

    \n

    Input and Output Declarations

    \n

    Input and output declarations have the form:

    \n
    \n
      input <name>:<type>\n  output <name>:<type>\n
    \n
    \n
    \n
      input <name>\n  output <name>\n
    \n
    \n

    For example, the following reactor doubles its input and sends the result to the output:

    \n

    $start(Double)$

    \n
    target C\nreactor Double {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    lf_set(y, x->value * 2);\n  =}\n}\n
    \n
    target Cpp\nreactor Double {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    if (x.is_present()){\n        y.set(*x.get() * 2);\n    }\n  =}\n}\n
    \n
    target Python\nreactor Double {\n  input x\n  output y\n  reaction(x) -> y {=\n    y.set(x.value * 2)\n  =}\n}\n
    \n
    target TypeScript\nreactor Double {\n  input x: number\n  output y: number\n  reaction(x) -> y {=\n    y = value * 2\n  =}\n}\n
    \n
    target Rust\nreactor Double {\n  input x: u32\n  output y: u32\n  reaction(x) -> y {=\n    ctx.set(y, ctx.get(x).unwrap() * 2);\n  =}\n}\n
    \n

    $end(Double)$

    \n

    Notice how the input value is accessed and how the output value is set. This is done differently for each target language. See the Target Language Details for detailed documentation of these mechanisms.\nSetting an output within a reaction will trigger downstream reactions at the same Logical Time that the reaction is invoked (or, more precisely, at the same tag). If a particular output port is set more than once at any tag, the last set value will be the one that downstream reactions see. Since the order in which reactions of a reactor are invoked at a logical time is deterministic, and whether inputs are present depends only on their timestamps, the final value set for an output will also be deterministic.

    \n
    \n

    The type of a port is a type in the target language plus the special type $time$. A type may also be specified using a code block, delimited by the same delimiters {= ... =} that separate target language code from Lingua Franca code in reactions. Any valid target-language type designator can be given within these delimiters.

    \n
    \n

    The $reaction$ declaration above indicates that an input event on port x is a trigger and that an output event on port y is a (potential) effect. A reaction can declare more than one trigger or effect by just listing them separated by commas (See Reactions for details). For example, the following reactor has two triggers and tests each input for presence before using it:

    \n

    $start(Destination)$

    \n
    target C\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    int sum = 0;\n    if (x->is_present) {\n      sum += x->value;\n    }\n    if (y->is_present) {\n      sum += y->value;\n    }\n    printf("Received %d.\\n", sum);\n  =}\n}\n
    \n
    target Cpp\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    int sum = 0;\n    if (x.is_present()) {\n      sum += *x.get();\n    }\n    if (y.is_present()) {\n      sum += *y.get();\n    }\n    std::cout << "Received: " << sum << std::endl;\n  =}\n}\n
    \n
    target Python\nreactor Destination {\n  input x\n  input y\n  reaction(x, y) {=\n    sum = 0\n    if x.is_present:\n      sum += x.value\n    if y.is_present:\n      sum += y.value\n    print(f"Received {sum}")\n  =}\n}\n
    \n
    target TypeScript\nreactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    let sum = 0\n    if (x !== undefined) {\n      sum += x\n    }\n    if (y !== undefined) {\n      sum += y\n    }\n    console.log(`Received ${sum}.`)\n  =}\n}\n
    \n
    target Rust\nreactor Destination {\n  input x: u32\n  input y: u32\n  reaction(x, y) {=\n    let mut sum = 0;\n    if let Some(x) = ctx.get(x) {\n      sum += x;\n    }\n    if let Some(y) = ctx.get(y) {\n      sum += y;\n    }\n    println!("Received {}.", sum);\n  =}\n}\n
    \n

    $end(Destination)$

    \n

    NOTE: if a reaction fails to test for the presence of an input and reads its value anyway, then the result it will get is target dependent.\nIn the C target, the value read will be the most recently seen input value, or, if no input event has occurred at an earlier logical time, then zero or NULL, depending on the data type of the input.\nIn the C++ target, a smart pointer is returned for present values and nullptr if the value is not present.\nIn the Python target, the value will be None if the input is not present.\nIn the TS target, the value will be undefined if the input is not present, a legitimate value in TypeScript.\nFIXME.

    \n

    Setting an Output Multiple Times

    \n

    If one or more reactions set an output multiple times at the same tag, then only the last value set will be seen by any downstream reactors.

    \n

    If a reaction wishes to test whether an output has been previously set at the current tag by some other reaction, it can test it in the same way it tests inputs for presence.

    \n

    Mutable Inputs

    \n

    Normally, a reaction does not modify the value of an input. An input is said to be immutable. The degree to which this is enforced varies by target language. Most of the target languages make it rather difficult to enforce, so the programmer needs to avoid modifying the input. Modifying an input value may lead to nondeterministic results.

    \n

    Occasionally, it is useful to modify an input. For example, the input may be a large data structure, and a reaction may wish to make a small modification and forward the result to an output. To accomplish this, the programmer should declare the input $mutable$ as follows:

    \n
    \n
      mutable input <name>:<type>\n
    \n
    \n
    \n
      mutable input <name>\n
    \n
    \n

    This is a directive to the code generator indicating that reactions that read this input may also modify the value of the input. The code generator will attempt to optimize the scheduling to avoid copying the input value, but this may not be possible, in which case it will automatically insert a copy operation, making it safe to modify the input. The target-specific reference documentation has more details about how this works.

    ","headings":[{"value":"Input and Output Declarations","depth":2},{"value":"Setting an Output Multiple Times","depth":2},{"value":"Mutable Inputs","depth":2}],"frontmatter":{"permalink":"/docs/handbook/inputs-and-outputs","title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"A First Reactor","oneline":"Writing your first Lingua Franca reactor.","permalink":"/docs/handbook/a-first-reactor"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","permalink":"/docs/handbook/parameters-and-state-variables"}}}},"pageContext":{"id":"1-inputs-and-outputs","slug":"/docs/handbook/inputs-and-outputs","repoPath":"/packages/documentation/copy/en/topics/Inputs and Outputs.md","previousID":"7ab4c09f-77ee-57d7-ac47-116061262590","nextID":"20781702-b6a5-5a16-b4d8-b4c45cd76fa3","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/inputs-and-outputs","result":{"data":{"markdownRemark":{"id":"28c65d97-e18a-53ae-8b3a-7e4caea32061","excerpt":"$page-showing-target$ In this section, we will endow reactors with inputs and outputs. Input and Output Declarations Input and output declarations have the form…","html":"

    $page-showing-target$

    \n

    In this section, we will endow reactors with inputs and outputs.

    \n

    Input and Output Declarations

    \n

    Input and output declarations have the form:

    \n
    \n
      input <name>:<type>\n  output <name>:<type>\n
    \n
    \n
    \n
      input <name>\n  output <name>\n
    \n
    \n

    For example, the following reactor doubles its input and sends the result to the output:

    \n

    $start(Double)$

    \n
    target C\nreactor Double {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    lf_set(y, x->value * 2);\n  =}\n}\n
    \n
    target Cpp\nreactor Double {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    if (x.is_present()){\n        y.set(*x.get() * 2);\n    }\n  =}\n}\n
    \n
    target Python\nreactor Double {\n  input x\n  output y\n  reaction(x) -> y {=\n    y.set(x.value * 2)\n  =}\n}\n
    \n
    target TypeScript\nreactor Double {\n  input x: number\n  output y: number\n  reaction(x) -> y {=\n    y = value * 2\n  =}\n}\n
    \n
    target Rust\nreactor Double {\n  input x: u32\n  output y: u32\n  reaction(x) -> y {=\n    ctx.set(y, ctx.get(x).unwrap() * 2);\n  =}\n}\n
    \n

    $end(Double)$

    \n

    Notice how the input value is accessed and how the output value is set. This is done differently for each target language. See the Target Language Details for detailed documentation of these mechanisms.\nSetting an output within a reaction will trigger downstream reactions at the same Logical Time that the reaction is invoked (or, more precisely, at the same tag). If a particular output port is set more than once at any tag, the last set value will be the one that downstream reactions see. Since the order in which reactions of a reactor are invoked at a logical time is deterministic, and whether inputs are present depends only on their timestamps, the final value set for an output will also be deterministic.

    \n
    \n

    The type of a port is a type in the target language plus the special type $time$. A type may also be specified using a code block, delimited by the same delimiters {= ... =} that separate target language code from Lingua Franca code in reactions. Any valid target-language type designator can be given within these delimiters.

    \n
    \n

    The $reaction$ declaration above indicates that an input event on port x is a trigger and that an output event on port y is a (potential) effect. A reaction can declare more than one trigger or effect by just listing them separated by commas (See Reactions for details). For example, the following reactor has two triggers and tests each input for presence before using it:

    \n

    $start(Destination)$

    \n
    target C\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    int sum = 0;\n    if (x->is_present) {\n      sum += x->value;\n    }\n    if (y->is_present) {\n      sum += y->value;\n    }\n    printf("Received %d.\\n", sum);\n  =}\n}\n
    \n
    target Cpp\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    int sum = 0;\n    if (x.is_present()) {\n      sum += *x.get();\n    }\n    if (y.is_present()) {\n      sum += *y.get();\n    }\n    std::cout << "Received: " << sum << std::endl;\n  =}\n}\n
    \n
    target Python\nreactor Destination {\n  input x\n  input y\n  reaction(x, y) {=\n    sum = 0\n    if x.is_present:\n      sum += x.value\n    if y.is_present:\n      sum += y.value\n    print(f"Received {sum}")\n  =}\n}\n
    \n
    target TypeScript\nreactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    let sum = 0\n    if (x !== undefined) {\n      sum += x\n    }\n    if (y !== undefined) {\n      sum += y\n    }\n    console.log(`Received ${sum}.`)\n  =}\n}\n
    \n
    target Rust\nreactor Destination {\n  input x: u32\n  input y: u32\n  reaction(x, y) {=\n    let mut sum = 0;\n    if let Some(x) = ctx.get(x) {\n      sum += x;\n    }\n    if let Some(y) = ctx.get(y) {\n      sum += y;\n    }\n    println!("Received {}.", sum);\n  =}\n}\n
    \n

    $end(Destination)$

    \n

    NOTE: if a reaction fails to test for the presence of an input and reads its value anyway, then the result it will get is target dependent.\nIn the C target, the value read will be the most recently seen input value, or, if no input event has occurred at an earlier logical time, then zero or NULL, depending on the data type of the input.\nIn the C++ target, a smart pointer is returned for present values and nullptr if the value is not present.\nIn the Python target, the value will be None if the input is not present.\nIn the TS target, the value will be undefined if the input is not present, a legitimate value in TypeScript.\nFIXME.

    \n

    Setting an Output Multiple Times

    \n

    If one or more reactions set an output multiple times at the same tag, then only the last value set will be seen by any downstream reactors.

    \n

    If a reaction wishes to test whether an output has been previously set at the current tag by some other reaction, it can test it in the same way it tests inputs for presence.

    \n

    Mutable Inputs

    \n

    Normally, a reaction does not modify the value of an input. An input is said to be immutable. The degree to which this is enforced varies by target language. Most of the target languages make it rather difficult to enforce, so the programmer needs to avoid modifying the input. Modifying an input value may lead to nondeterministic results.

    \n

    Occasionally, it is useful to modify an input. For example, the input may be a large data structure, and a reaction may wish to make a small modification and forward the result to an output. To accomplish this, the programmer should declare the input $mutable$ as follows:

    \n
    \n
      mutable input <name>:<type>\n
    \n
    \n
    \n
      mutable input <name>\n
    \n
    \n

    This is a directive to the code generator indicating that reactions that read this input may also modify the value of the input. The code generator will attempt to optimize the scheduling to avoid copying the input value, but this may not be possible, in which case it will automatically insert a copy operation, making it safe to modify the input. The target-specific reference documentation has more details about how this works.

    ","headings":[{"value":"Input and Output Declarations","depth":2},{"value":"Setting an Output Multiple Times","depth":2},{"value":"Mutable Inputs","depth":2}],"frontmatter":{"permalink":"/docs/handbook/inputs-and-outputs","title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"A First Reactor","oneline":"Writing your first Lingua Franca reactor.","permalink":"/docs/handbook/a-first-reactor"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","permalink":"/docs/handbook/parameters-and-state-variables"}}}},"pageContext":{"id":"1-inputs-and-outputs","slug":"/docs/handbook/inputs-and-outputs","repoPath":"/packages/documentation/copy/en/topics/Inputs and Outputs.md","previousID":"7ab4c09f-77ee-57d7-ac47-116061262590","nextID":"20781702-b6a5-5a16-b4d8-b4c45cd76fa3","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/intellij/page-data.json b/page-data/docs/handbook/intellij/page-data.json index d0c83e743..f1ed6d73b 100644 --- a/page-data/docs/handbook/intellij/page-data.json +++ b/page-data/docs/handbook/intellij/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/intellij","result":{"data":{"markdownRemark":{"id":"57e21136-002a-540c-84e6-b179b7e75eb2","excerpt":"Prerequisites Java 17 (download from Oracle) IntelliJ IDEA Community Edition (download from Jetbrains) Cloning lingua-franca repository If you have not done so…","html":"

    Prerequisites

    \n\n

    Cloning lingua-franca repository

    \n

    If you have not done so already, clone the lingua-franca repository into your working directory.

    \n
    $ git clone git@github.com:lf-lang/lingua-franca.git lingua-franca\n$ cd lingua-franca\n$ git submodule update --init --recursive\n
    \n

    Opening lingua-franca as IntelliJ Project

    \n

    To import the Lingua Franca repository as a project, simply run ./gradlew openIdea.\nThis will create some project files and then open the project in IntelliJ.

    \n

    When you open the project for the first time, you will see a small pop-up in the lower right corner.

    \n

    \n \n \n

    \n

    Click on Load Gradle Project to import the Gradle configurations.

    \n

    If you are prompted to a pop-up window asking if you trust the Gradle project, click Trust Project.

    \n

    \n \n \n

    \n

    Once the repository is imported as a Gradle project, you will see a Gradle tab on the right.

    \n

    Once the indexing finishes, you can expand the Gradle project and see the set of Tasks.

    \n

    \n \n \n

    \n

    You can run any Gradle command from IntelliJ simply by clicking on the Execute Gradle Task icon in the Gradle tab. You are then prompted for the precise command to run.

    \n

    Setting up run configurations

    \n

    You can set up a run configuration for running and debugging various Gradle tasks from the Gradle tab, including the code generation through lfc.\nTo set up a run configuration for the run task of lfc, expand the application task group under org.lflang > Tasks, right-click on ⚙️ run, and select Modify Run Configuration….\nThis will create a custom run/debug configuration for you.

    \n

    In the Create Run Configuration dialog, click on the text box next to Run, select cli:lfc:run from the drop-down menu, and append arguments to be passed to lfc using the --args flag. For instance, to invoke lfc on test/Cpp/src/HelloWorld.lf, enter cli:lfc:run --args 'test/Cpp/src/HelloWorld.lf' Then click OK.

    \n

    \n \n \n

    \n

    You will see a new run/debug config added to the top-level menu bar, as shown below.\nYou can always change the config, for example, changing the --args, by clicking Edit Configurations via a drop-down menu.

    \n

    \n \n \n

    \n

    Running and Debugging

    \n

    Using the newly added config, you can run and debug the code generator by clicking the play button and the debug button.

    \n

    \n \n \n

    \n

    Set up breakpoints before starting the debugger by clicking the space right next to the line numbers.\nWhile debugging, you can run code step-by-step by using the debugger tools.

    \n

    \n to the targetTest command or -DsingleTest=... to your singleTest command to specify the target (e.g., C) or the specific test that you would like to run.

    ","headings":[{"value":"Prerequisites","depth":2},{"value":"Cloning lingua-franca repository","depth":2},{"value":"Opening lingua-franca as IntelliJ Project","depth":2},{"value":"Setting up run configurations","depth":2},{"value":"Running and Debugging","depth":2},{"value":"Integration Tests","depth":2}],"frontmatter":{"permalink":"/docs/handbook/intellij","title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","permalink":"/docs/handbook/developer-setup"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","permalink":"/docs/handbook/regression-tests"}}}},"pageContext":{"id":"5-developer-intellij-setup","slug":"/docs/handbook/intellij","repoPath":"/packages/documentation/copy/en/developer/Developer IntelliJ Setup.md","previousID":"81b7347d-2806-53c1-91ad-c7b12d062d3c","nextID":"b004db16-2d0f-54e3-a2a8-d9d6f510eea1","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/intellij","result":{"data":{"markdownRemark":{"id":"57e21136-002a-540c-84e6-b179b7e75eb2","excerpt":"Prerequisites Java 17 (download from Oracle) IntelliJ IDEA Community Edition (download from Jetbrains) Cloning lingua-franca repository If you have not done so…","html":"

    Prerequisites

    \n\n

    Cloning lingua-franca repository

    \n

    If you have not done so already, clone the lingua-franca repository into your working directory.

    \n
    $ git clone git@github.com:lf-lang/lingua-franca.git lingua-franca\n$ cd lingua-franca\n$ git submodule update --init --recursive\n
    \n

    Opening lingua-franca as IntelliJ Project

    \n

    To import the Lingua Franca repository as a project, simply run ./gradlew openIdea.\nThis will create some project files and then open the project in IntelliJ.

    \n

    When you open the project for the first time, you will see a small pop-up in the lower right corner.

    \n

    \n \n \n

    \n

    Click on Load Gradle Project to import the Gradle configurations.

    \n

    If you are prompted to a pop-up window asking if you trust the Gradle project, click Trust Project.

    \n

    \n \n \n

    \n

    Once the repository is imported as a Gradle project, you will see a Gradle tab on the right.

    \n

    Once the indexing finishes, you can expand the Gradle project and see the set of Tasks.

    \n

    \n \n \n

    \n

    You can run any Gradle command from IntelliJ simply by clicking on the Execute Gradle Task icon in the Gradle tab. You are then prompted for the precise command to run.

    \n

    Setting up run configurations

    \n

    You can set up a run configuration for running and debugging various Gradle tasks from the Gradle tab, including the code generation through lfc.\nTo set up a run configuration for the run task of lfc, expand the application task group under org.lflang > Tasks, right-click on ⚙️ run, and select Modify Run Configuration….\nThis will create a custom run/debug configuration for you.

    \n

    In the Create Run Configuration dialog, click on the text box next to Run, select cli:lfc:run from the drop-down menu, and append arguments to be passed to lfc using the --args flag. For instance, to invoke lfc on test/Cpp/src/HelloWorld.lf, enter cli:lfc:run --args 'test/Cpp/src/HelloWorld.lf' Then click OK.

    \n

    \n \n \n

    \n

    You will see a new run/debug config added to the top-level menu bar, as shown below.\nYou can always change the config, for example, changing the --args, by clicking Edit Configurations via a drop-down menu.

    \n

    \n \n \n

    \n

    Running and Debugging

    \n

    Using the newly added config, you can run and debug the code generator by clicking the play button and the debug button.

    \n

    \n \n \n

    \n

    Set up breakpoints before starting the debugger by clicking the space right next to the line numbers.\nWhile debugging, you can run code step-by-step by using the debugger tools.

    \n

    \n to the targetTest command or -DsingleTest=... to your singleTest command to specify the target (e.g., C) or the specific test that you would like to run.

    ","headings":[{"value":"Prerequisites","depth":2},{"value":"Cloning lingua-franca repository","depth":2},{"value":"Opening lingua-franca as IntelliJ Project","depth":2},{"value":"Setting up run configurations","depth":2},{"value":"Running and Debugging","depth":2},{"value":"Integration Tests","depth":2}],"frontmatter":{"permalink":"/docs/handbook/intellij","title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Developer Setup","oneline":"Setting up Lingua Franca for developers.","permalink":"/docs/handbook/developer-setup"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","permalink":"/docs/handbook/regression-tests"}}}},"pageContext":{"id":"5-developer-intellij-setup","slug":"/docs/handbook/intellij","repoPath":"/packages/documentation/copy/en/developer/Developer IntelliJ Setup.md","previousID":"81b7347d-2806-53c1-91ad-c7b12d062d3c","nextID":"b004db16-2d0f-54e3-a2a8-d9d6f510eea1","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/language-specification/page-data.json b/page-data/docs/handbook/language-specification/page-data.json index 222d26cb2..2601ea7fd 100644 --- a/page-data/docs/handbook/language-specification/page-data.json +++ b/page-data/docs/handbook/language-specification/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/language-specification","result":{"data":{"markdownRemark":{"id":"71ea83b3-e77d-5dfb-a0d9-c5f2dfa023e6","excerpt":"A Lingua Franca file, which has a .lf extension, contains the following: One target specification. Zero or more import statements. One or more reactor blocks…","html":"

    A Lingua Franca file, which has a .lf extension, contains the following:

    \n\n

    If one of the reactors in the file is designated main or federated, then the file defines an executable application. Otherwise, it defines one or more library reactors that can be imported into other LF files. For example, an LF file might be structured like this:

    \n
    target C;\nmain reactor C {\n    a = new A();\n    b = new B();\n    a.y -> b.x;\n}\nreactor A {\n    output y;\n    ...\n}\nreactor B {\n    input x;\n    ...\n}\n
    \n

    The name of the main reactor (C above) is optional. If given, it must match the filename (C.lf in the above example).

    \n

    This example specifies and instantiates two reactors, one of which sends messages to the other. A minimal but complete Lingua Franca file with one reactor is this:

    \n
    target C;\nmain reactor HelloWorld {\n    reaction(startup) {=\n        printf("Hello World.\\n");\n    =}\n}\n
    \n

    See the C target documentation for details about this example.

    \n

    Target Language Specification

    \n

    Every Lingua Franca program begins with a target language specification that specifies the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca compiler.

    \n

    Import Statement

    \n

    An import statement has the form:

    \n
    \n

    import { reactor1, reactor2 as alias2, […] } frompath“;

    \n
    \n

    where path specifies another Lingua Franca file relative to the location of the current file.

    \n

    Reactor Block

    \n

    A reactor is a software component that reacts to input events, timer events, and internal events. It has private state variables that are not visible to any other reactor. Its reactions can consist of altering its own state, sending messages to other reactors, or affecting the environment through some kind of actuation or side effect.

    \n

    The general structure of a reactor block is as follows:

    \n
    \n

    reactor name (parameters) {
    >   state declarations
    >   method declarations
    >   input declarations
    >   output declarations
    >   timer declarations
    >   action declarations
    >   reaction declarations
    >   contained reactors
    >    …
    \n}

    \n
    \n

    Parameter, inputs, outputs, timers, actions, and contained reactors all have names, and the names are required to be distinct from one another.

    \n

    If the reactor keyword is preceded by main, then this reactor will be instantiated and run by the generated code. If an imported LF file contains a main reactor, that reactor is ignored. Only reactors that not designated main are imported. This makes it easy to create a library of reusable reactors that each come with a test case or demonstration in the form of a main reactor.

    \n

    Parameter Declaration

    \n

    A reactor class definition can define parameters as follows:

    \n
    \n

    reactor ClassName(paramName1:type(expr), paramName2:type(expr)) {
    >    …
    \n}

    \n
    \n

    Each parameter may have a type annotation, written :type, and must have a default value, written (expr).

    \n

    The type annotation specifies a type in the target language, which is necessary for some target languages. For instance in C you might write

    \n
    reactor Foo(size: int(100)) {\n    ...\n}\n
    \n
    \nIntroduction to basic LF types and expressions... click to expand\n

    One useful type predefined by LF is the time type, which represents time durations. Values of this type may be written with time expressions, like 100 msec or 1 second (see Basic expressions for a reference).

    \n

    For instance, you can write the following in any target language:

    \n
    reactor Foo(period: time(100 msec)) {\n    ...\n}\n
    \n

    Container types may also be written e.g. int[], which is translated to a target-specific array or list type. The acceptable expressions for these types vary across targets (see Complex expressions), for instance in C, you can initialize an array parameter as follows:

    \n
    reactor Foo(my_array:int[](1, 2, 3)) {\n   ...\n}\n
    \n

    If the type or expression uses syntax that Lingua Franca does not support, you can use {= ... =} delimiters to enclose them and escape them. For instance to have a 2-dimensional array as a parameter in C:

    \n
    reactor Foo(param:{= int[][] =}({= { {1}, {2} } =})) {\n    ...\n}\n
    \n

    Both int[][] and {% raw %}{{1}, {2}} {% endraw %} are C fragments here, not LF.

    \n
    \n

    Other forms for types and expressions are described in LF types and LF expressions.

    \n

    How parameters may be used in the body of a reaction depends on the target. For example, in the C target, a self struct is provided that contains the parameter values. The following example illustrates this:

    \n
    target C;\nreactor Gain(scale:int(2)) {\n    input x:int;\n    output y:int;\n    reaction(x) -> y {=\n        lf_set(y, x->value * self->scale);\n    =}\n}\n
    \n

    This reactor, given any input event x will produce an output y with value equal to the input scaled by the scale parameter. The default value of the scale parameter is 2, but this can be changed when the Gain reactor is instantiated. The lf_set() is the mechanism provided by the C target for setting the value of outputs. The parameter scale and input x are just referenced in the C code as shown above.

    \n

    State Declaration

    \n

    A state declaration has one of the forms:

    \n
    \n

    state name:type(initial_value);
    \nstate name(parameter);

    \n
    \n

    In the first form, the type annotation is only required in some targets. The initial value may be any expression, including a special initializer forms.

    \n

    In the second form, the state variable inherits its type from the specified parameter, which also provides the initial value for the state variable.

    \n

    How state variables may be used in the body of a reaction depends on the target. For example, in the C target, a self struct is provided that contains the state values. The following example illustrates this:

    \n
    reactor Count {\n\toutput c:int;\n\ttimer t(0, 1 sec);\n\tstate i:int(0);\n\treaction(t) -> c {=\n\t\t(self->i)++;\n\t\tlf_set(c, self->i);\n\t=}\n}\n
    \n

    Method Declaration

    \n

    A method declaration has one of the forms:

    \n
    \n

    method name();
    \nmethod name():type;
    \nmethod name(arg1_name:arg1type, _arg2_name:arg2type, …);\nmethod _name(arg1_name:arg1type, _arg2_name:arg2type, …):_type;

    \n
    \n

    The first form defines a method with no arguments and no return value. The second form defines a method with the return type type but no arguments. The third form defines a method with arguments given by their name and type, but without a return value. Finally, the fourth form is similar to the third, but adds a return type.

    \n

    The method keyword can optionally be prefixed with the const qualifier, which indicates that the method is “read-only”. This is relevant for some target languages such as C++.

    \n

    See the C++ documentation for a usage example.

    \n

    Input Declaration

    \n

    An input declaration has the form:

    \n
    \n

    input name:type;

    \n
    \n

    The Gain reactor given above provides an example. The type is just like parameter types.

    \n

    An input may have the modifier mutable, as follows:

    \n
    \n

    mutable input name:type

    \n
    \n

    This is a directive to the code generator indicating that reactions that read this input will also modify the value of the input. Without this modifier, inputs are immutable; modifying them is disallowed. The precise mechanism for making use of mutable inputs is target-language specific. See, for example, the C language target.

    \n

    An input port may have more than one channel. See multiports documentation.

    \n

    Output Declaration

    \n

    An output declaration has the form:

    \n
    \n

    output name:type;

    \n
    \n

    The Gain reactor given above provides an example. The type is just like parameter types.

    \n

    An output port may have more than one channel. See multiports documentation.

    \n

    Timer Declaration

    \n

    A timer, like an input and an action, causes reactions to be invoked. Unlike an action, it is triggered automatically by the scheduler. This declaration is used when you want to invoke reactions once at specific times or periodically. A timer declaration has the form:

    \n
    \n

    timer name(offset, period);

    \n
    \n

    For example,

    \n
    timer foo(10 msec, 100 msec);\n
    \n

    This specifies a timer named foo that will first trigger 10 milliseconds after the start of execution and then repeatedly trigger at intervals of 100 ms. The units are optional, and if they are not included, then the number will be interpreted in a target-dependent way. The units supported are the same as in parameter declarations described above.

    \n

    The times specified are logical times. Specifically, if two timers have the same offset and period, then they are logically simultaneous. No observer will be able to see that one timer has triggered and the other has not. Even though these are logical times, the runtime system will make an effort to align those times to physical times. Such alignment can never be perfect, and its accuracy will depend on the execution platform.

    \n

    Both arguments are optional, with both having default value zero. An offset of zero or greater specifies the minimum time delay between the time at the start of execution and when the action is triggered. The period is zero or greater, where a value of zero specifies that the reactions should be triggered exactly once,\nwhereas a value greater than zero specifies that they should be triggered repeatedly with the period given.

    \n

    To cause a reaction to be invoked at the start of execution, a special startup trigger is provided:

    \n
    reactor Foo {\n    reaction(startup) {=\n        ... perform initialization ...\n    =}\n}\n
    \n

    The startup trigger is equivalent to a timer with no offset or period.

    \n

    Action Declaration

    \n

    An action, like an input, can cause reactions to be invoked. Whereas inputs are provided by other reactors, actions are scheduled by this reactor itself, either in response to some observed external event or as a delayed response to some input event. The action can be scheduled by a reactor by invoking a schedule function in a reaction or in an asynchronous callback function.

    \n

    An action declaration is either physical or logical:

    \n
    \n

    physical action name(min_delay, min_spacing, policy):type;
    > logical action name(min_delay, min_spacing, policy):type;

    \n
    \n

    The min_delay, min_spacing, and policy are all optional. If only one argument is given in parentheses, then it is interpreted as an min_delay, if two are given, then they are interpreted as min_delay and min_spacing, etc. The min_delay and min_spacing have to be a time value. The policy argument is a string that can be one of the following: 'defer' (default), 'drop', or 'replace'.

    \n

    An action will trigger at a logical time that depends on the arguments given to the schedule function, the min_delay, min_spacing, and policy arguments above, and whether the action is physical or logical.

    \n

    If the logical keyword is given, then the tag assigned to the event resulting from a call to schedule function is computed as follows. First, let t be the current logical time. For a logical action, the schedule function must be invoked from within a reaction (synchronously), so t is just the logical time of that reaction.

    \n

    The (preliminary) tag of the action is then just t plus min_delay plus the offset argument to schedule function.

    \n

    If the physical keyword is given, then the physical clock on the local platform is used as the timestamp assigned to the action. Moreover, for a physical action, unlike a logical action, the schedule function can be invoked from outside of any reaction (asynchronously), e.g. from an interrupt service routine or callback function.

    \n

    If a min_spacing has been declared, then a minimum distance between the tags of two subsequently scheduled events on the same action is enforced. If the preliminary tag is closer to the tag of the previously scheduled event (if there is one), then policy determines how the given constraints is enforced.

    \n
      \n
    • 'drop': the new event is dropped and schedule returns without having modified the event queue.
    • \n
    • 'replace': the payload of the new event is assigned to the preceding event if it is still pending in the event queue; no new event is added to the event queue in this case. If the preceding event has already been pulled from the event queue, the default 'defer' policy is applied.
    • \n
    • 'defer': the event is added to the event queue with a tag that is equal to earliest time that satisfies the minimal spacing requirement. Assuming the tag of the preceding event is t_prev, then the tag of the new event simply becomes t_prev + min_spacing.
    • \n
    \n

    Note that while the 'defer' policy is conservative in the sense that it does not discard events, it could potentially cause an unbounded growth of the event queue.

    \n

    In all cases, the logical time of a new event will always be strictly greater than the logical time at which it is scheduled by at least one microstep (see the Time section).

    \n

    The default min_delay is zero. The default min_spacing is undefined (meaning that no minimum spacing constraint is enforced). If a min_spacing is defined, it has to be strictly greater than zero, and greater than or equal to the time precision of the target (for the C target, it is one nanosecond).

    \n

    The min_delay parameter in the action declaration is static (set at compile time), while the offset parameter given to the schedule function may be dynamically set at runtime. Hence, for static analysis and scheduling, the action’s’ min_delay parameter can be assumed to be a minimum delay for analysis purposes.

    \n

    Discussion

    \n

    Logical actions are used to schedule events at a future logical time relative to the current logical time. Physical time is ignored. They must be scheduled within reactions, and the timestamp of the scheduled event will be relative to the current logical time of the reaction that schedules them. It is an error to schedule a logical action asynchronously, outside of the context of a reaction. Asynchronous actions are required to be physical.

    \n

    Physical actions are typically used to assign timestamps to externally triggered events, such as the arrival of a network message or the acquisition of sensor data, where the time at which these external events occurs is of interest. There are (at least) three interesting use cases:

    \n
      \n
    1. An asynchronous event, such as a callback function or interrupt service routine (ISR), is invoked at a physical time t and schedules an action with timestamp T=t. To get this behavior, just set the physical action to have min_delay = 0 and call the schedule function with offset = 0. The min_spacing can be useful here to prevent these external events from overwhelming the software system.
    2. \n
    3. A periodic task that is occasionally modified by a sporadic sensor. In this case, you can set min_delay = period and call schedule with offset = 0. The resulting timestamp of the sporadic sensor event will always align with the periodic events. This is similar to periodic polling, but without the overhead of polling the sensor when nothing interesting is happening.
    4. \n
    5. You can impose a minimum physical time delay between an event’s occurrence, such as a push of a button, and system response by adjusting the offset.
    6. \n
    \n

    Actions With Values

    \n

    If an action is declared with a type, then it can carry a value, a data value passed to the schedule function. This value will be available to any reaction that is triggered by the action. The specific mechanism, however, is target-language dependent. See the C target for an example.

    \n

    Reaction Declaration

    \n

    A reaction is defined within a reactor using the following syntax:

    \n
    \n

    reaction(triggers) uses -> effects {=
    >    … target language code …
    \n=}

    \n
    \n

    The uses and effects fields are optional. A simple example appears in the “hello world” example given above:

    \n
        reaction(t) {=\n        printf("Hello World.\\n");\n    =}\n
    \n

    In this example, t is a trigger (a timer named t). When that timer fires, the reaction will be invoked. Triggers can be timers, inputs, outputs of contained reactors, or actions. A comma-separated list of triggers can be given, in which case any of the specified triggers can trigger the reaction. If, at any logical time instant, more than one of the triggers fires, the reaction will nevertheless be invoked only once.

    \n

    The uses field specifies inputs that the reaction observes but that do not trigger the reaction. This field can also be a comma-separated list of inputs. Since the input does not trigger the reaction, the body of the reaction will normally need to test for presence of the input before using it. How to do this is target specific. See how this is done in the C target.

    \n

    The effects field, occurring after the right arrow, declares which outputs and actions the target code may produce or schedule. The effects field may also specify inputs of contained reactors, provided that those inputs do not have any other sources of data. These declarations make it possible for the reaction to send outputs or enable future actions, but they do not require that the reaction code do that.

    \n

    Target Code

    \n

    The body of the reaction is code in the target language surrounded by {= and =}. This code is not parsed by the Lingua Franca compiler. It is used verbatim in the program that is generated.

    \n

    The target provides language-dependent mechanisms for referring to inputs, outputs, and actions in the target code. These mechanisms can be different in each target language, but all target languages provide the same basic set of mechanisms. These mechanisms include:

    \n
      \n
    • Obtaining the current logical time (logical time does not advance during the execution of a reaction, so the execution of a reaction is logically instantaneous).
    • \n
    • Determining whether inputs are present at the current logical time and reading their value if they are. If a reaction is triggered by exactly one input, then that input will always be present. But if there are multiple triggers, or if the input is specified in the uses field, then the input may not be present when the reaction is invoked.
    • \n
    • Setting output values. Reactions in a reactor may set an output value more than once at any instant of logical time, but only the last of the values set will be sent on the output port.
    • \n
    • Scheduling future actions.
    • \n
    \n

    In the C target, for example, the following reactor will add two inputs if they are present at the time of a reaction:

    \n
    reactor Add {\n    input in1:int;\n    input in2:int;\n    output out:int;\n    reaction(in1, in2) -> out {=\n        int result = 0;\n        if (in1->is_present) result += in1->value;\n        if (in2->is_present) result += in2->value;\n        lf_set(out, result);\n    =}\n}\n
    \n

    See the C target for an example of how these things are specified in C.

    \n

    NOTE: if a reaction fails to test for the presence of an input and reads its value anyway, then the result it will get is undefined and may be target dependent. In the C target, as of this writing, the value read will be the most recently seen input value, or, if no input event has occurred at an earlier logical time, then zero or NULL, depending on the data type of the input. In the TS target, the value will be undefined, a legitimate value in TypeScript.

    \n

    Scheduling Future Reactions

    \n

    Each target language provides some mechanism for scheduling future reactions. Typically, this takes the form of a schedule function that takes as an argument an action, a time interval, and (perhaps optionally), a payload. For example, in the C target, in the following program, each reaction to the timer t schedules another reaction to occur 100 msec later:

    \n
    target C;\nmain reactor Schedule {\n    timer t(0, 1 sec);\n    logical action a;\n    reaction(t) -> a {=\n        schedule(a, MSEC(100));\n    =}\n    reaction(a) {=\n        printf("Nanoseconds since start: %lld.\\n", lf_time_logical_elapsed());\n    =}\n}\n
    \n

    When executed, this will produce the following output:

    \n
    Start execution at time Sun Aug 11 04:11:57 2019\nplus 919310000 nanoseconds.\nNanoseconds since start: 100000000.\nNanoseconds since start: 1100000000.\nNanoseconds since start: 2100000000.\n...
    \n

    This action has no data type and carries no value, but, as explained below, an action can carry a value.

    \n

    Asynchronous Callbacks

    \n

    In targets that support multitasking, the schedule function, which schedules future reactions, may be safely invoked on a physical action in code that is not part of a reaction. For example, in the multithreaded version of the C target, schedule may be invoked in an interrupt service routine. The reaction(s) that are scheduled are guaranteed to occur at a time that is strictly larger than the current logical time of any reactions that are being interrupted.

    \n

    Superdense Time

    \n

    Lingua Franca uses a concept known as superdense time, where two time values that appear to be the same are not logically simultaneous. At every logical time value, for example midnight on January 1, 1970, there exist a logical sequence of microsteps that are not simultaneous. The Microsteps example illustrates this:

    \n
    target C;\nreactor Destination {\n    input x:int;\n    input y:int;\n    reaction(x, y) {=\n        printf("Time since start: %lld.\\n", lf_time_logical_elapsed());\n        if (x->is_present) {\n            printf("  x is present.\\n");\n        }\n        if (y->is_present) {\n            printf("  y is present.\\n");\n        }\n    =}\n}\nmain reactor Microsteps {\n    timer start;\n    logical action repeat;\n    d = new Destination();\n    reaction(start) -> d.x, repeat {=\n        lf_set(d.x, 1);\n        schedule(repeat, 0);\n    =}\n    reaction(repeat) -> d.y {=\n        lf_set(d.y, 1);\n    =}\n}\n
    \n

    The Destination reactor has two inputs, x and y, and it simply reports at each logical time where either is present what is the logical time and which is present. The Microsteps reactor initializes things with a reaction to the one-time timer event start by sending data to the x input of Destination. It then schedules a repeat action.

    \n

    Note that time delay in the call to schedule is zero. However, any reaction scheduled by schedule is required to occur strictly later than current logical time. In Lingua Franca, this is handled by scheduling the repeat reaction to occur one microstep later. The output printed, therefore, will look like this:

    \n
    Time since start: 0.\n  x is present.\nTime since start: 0.\n  y is present.
    \n

    Note that the numerical time reported by get_elapsed_logical_time() has not advanced in the second reaction, but the fact that x is not present in the second reaction proves that the first reaction and the second are not logically simultaneous. The second occurs one microstep later.

    \n

    Note that it is possible to write code that will prevent logical time from advancing except by microsteps. For example, we could replace the reaction to repeat in Main with this one:

    \n
        reaction(repeat) -> d.y, repeat {=\n        lf_set(d.y, 1);\n        schedule(repeat, 0);\n    =}\n
    \n

    This would create what is known as a stuttering Zeno condition, where logical time cannot advance. The output will be an unbounded sequence like this:

    \n
    Time since start: 0.\n  x is present.\nTime since start: 0.\n  y is present.\nTime since start: 0.\n  y is present.\nTime since start: 0.\n  y is present.\n...
    \n

    Startup and Shutdown Reactions

    \n

    Two special triggers are supported, startup and shutdown. A reaction that specifies the startup trigger will be invoked at the start of execution of the model. The following two syntaxes have exactly the same effect:

    \n
        reaction(startup) {= ... =}\n
    \n

    and

    \n
        timer t;\n    reaction(t) {= ... =}\n
    \n

    In other words, startup is a timer that triggers once at the first logical time of execution. As with any other reaction, the reaction can also be triggered by inputs and can produce outputs or schedule actions.

    \n

    The shutdown trigger is slightly different. A shutdown reaction is specified as follows:

    \n
       reaction(shutdown) {= ... =}\n
    \n

    This reaction will be invoked when the program terminates normally (there are no more events, some reaction has called a request_stop() utility provided in the target language, or the execution was specified to last a finite logical time). The reaction will be invoked at a logical time one microstep later than the last logical time of the execution. In other words, the presence of this reaction means that the program will execute one extra logical time cycle beyond what it would have otherwise, and that logical time is one microstep later than what would have otherwise been the last logical time.

    \n

    If the reaction produces outputs, then downstream reactors will also be invoked at that later logical time. If the reaction schedules future reactions, those will be ignored. After the completion of this final logical time cycle, one microstep later than the normal termination, the program will exit.

    \n

    Contained Reactors

    \n

    Reactors can contain instances of other reactors defined in the same file or in an imported file. Assuming the above Count reactor is stored in a file Count.lf, then CountTest is an example that imports and instantiates it to test the reactor:

    \n
    target C;\nimport Count.lf;\nreactor Test {\n    input c:int;\n    state i:int(0);\n    reaction(c) {=\n        printf("Received %d.\\n", c->value);\n        (self->i)++;\n        if (c->value != self->i) {\n            printf("ERROR: Expected %d but got %d\\n.", self->i, c->value);\n            exit(1);\n        }\n    =}\n    reaction(shutdown) {=\n        if (self->i != 4) {\n            printf("ERROR: Test should have reacted 4 times, but reacted %d times.\\n", self->i);\n            exit(2);\n        }\n    =}\n}\n\nmain reactor CountTest {\n    count = new Count();\n    test = new Test();\n    count.out -> test.c;\n}\n
    \n

    An instance is created with the syntax:

    \n
    \n

    instance_name = new class_name(parameters);

    \n
    \n

    A bank with several instances can be created in one such statement, as explained in the banks of reactors documentation.

    \n

    The parameters argument has the form:

    \n
    \n

    parameter1_name = parameter1_value, parameter2_name = parameter2_value, …

    \n
    \n

    Connections between ports are specified with the syntax:

    \n
    \n

    output_port -> input_port

    \n
    \n

    where the ports are either instance_name.port_name or just port_name, where the latter form denotes a port belonging to the reactor that contains the instances.

    \n

    Physical Connections

    \n

    A subtle and rarely used variant is a physical connection, denoted ~>. In such a connection, the logical time at the recipient is derived from the local physical clock rather than being equal to the logical time at the sender. The physical time will always exceed the logical time of the sender, so this type of connection incurs a nondeterministic positive logical time delay. Physical connections are useful sometimes in Distributed Execution in situations where the nondeterministic logical delay is tolerable. Such connections are more efficient because timestamps need not be transmitted and messages do not need to flow through through a centralized coordinator (if a centralized coordinator is being used).

    \n

    Connections with Delays

    \n

    Connections may include a logical delay using the after keyword, as follows:

    \n
    \n

    output_port -> input_port after 10 msec

    \n
    \n

    This means that the logical time of the message delivered to the input_port will be 10 milliseconds larger than the logical time of the reaction that wrote to output_port. If the time value is greater than zero, then the event will appear at microstep 0. If it is equal to zero, then it will appear at the current microstep plus one.

    \n

    When there are multiports or banks of reactors, several channels can be connected with a single connection statement. See Multiports and Banks of Reactors.

    \n

    The following example defines a reactor that adds a counting sequence to its input. It uses the above Count and Add reactors (see Hierarchy2):

    \n
    import Count.lf;\nimport Add.lf;\nreactor AddCount {\n    input in:int;\n    output out:int;\n    count = new Count();\n    add = new Add();\n    in -> add.in1;\n    count.out -> add.in2;\n    add.out -> out;\n}\n
    \n

    A reactor that contains other reactors may, within a reaction, send data to the contained reactor. The following example illustrates this (see SendingInside):

    \n
    target C;\nreactor Printer {\n\tinput x:int;\n\treaction(x) {=\n\t\tprintf("Inside reactor received: %d\\n", x->value);\n\t=}\n}\nmain reactor SendingInside {\n\tp = new Printer();\n\treaction(startup) -> p.x {=\n\t\tlf_set(p.x, 1);\n\t=}\n}\n
    \n

    Running this will print:

    \n
    Inside reactor received: 1
    \n

    Deadlines

    \n

    Lingua Franca includes a notion of a deadline, which is a relation between logical time and physical time. Specifically, a program may specify that the invocation of a reaction must occur within some physical-time interval of the logical timestamp of the message. If a reaction is invoked at logical time 12 noon, for example, and the reaction has a deadline of one hour, then the reaction is required to be invoked before the physical-time clock of the execution platform reaches 1 PM. If the deadline is violated, then the specified deadline handler is invoked instead of the reaction. For example (see Deadline):

    \n
    reactor Deadline() {\n    input x:int;\n    output d:int; // Produced if the deadline is violated.\n    reaction(x) -> d {=\n        printf("Normal reaction.\\n");\n    =} deadline(10 msec) {=\n        printf("Deadline violation detected.\\n");\n        lf_set(d, x->value);\n    =}\n
    \n

    This reactor specifies a deadline of 10 milliseconds (this can be a parameter of the reactor). If the reaction to x is triggered later in physical time than 10 msec past the timestamp of x, then the second body of code is executed instead of the first. That second body of code has access to anything the first body of code has access to, including the input x and the output d. The output can be used to notify the rest of the system that a deadline violation occurred.

    \n

    The amount of the deadline, of course, can be given by a parameter.

    \n

    A sometimes useful pattern is when a container reactor reacts to deadline violations in a contained reactor. The DeadlineHandledAbove example illustrates this:

    \n
    target C;\nreactor Deadline() {\n    input x:int;\n    output deadline_violation:bool;\n    reaction(x) -> deadline_violation {=\n        ... normal code to execute ...\n    =} deadline(100 msec) {=\n        printf("Deadline violation detected.\\n");\n        lf_set(deadline_violation, true);\n    =}\n}\nmain reactor DeadlineHandledAbove {\n    d = new Deadline();\n    ...\n    reaction(d.deadline_violation) {=\n        ... handle the deadline violation ...\n    =}\n}\n
    \n

    Comments

    \n

    Lingua Franca files can have C/C++/Java-style comments and/or Python-style comments. All of the following are valid comments:

    \n
        // Single-line C-style comment.\n    /*\n       Multi-line C-style comment.\n     */\n    # Single-line Python-style comment.\n    '''\n       Multi-line Python-style comment.\n    '''\n
    \n

    Appendix: LF types

    \n

    Type annotations may be written in many places in LF, including parameter declarations, state variable declarations, input and output declarations. In some targets, they are required, because the target language requires them too.

    \n

    Assigning meaning to type annotations is entirely offloaded to the target compiler, as LF does not feature a type system (yet?). However, LF’s syntax for types supports a few idioms that have target-specific meaning. Types may have the following forms:

    \n
      \n
    • the time type is reserved by LF, its values represent time durations. The time type accepts time expressions for values, e.g. 100 msec, or 0 (see Basic expressions for a reference).
    • \n
    • identifiers are valid types (e.g. int, size_t), and may be followed by type arguments (e.g. vector<int>).
    • \n
    • the syntactic forms type[] and type[integer] correspond to target-specific array types. The second form is available only in languages which support fixed-size array types (e.g. in C++, std::array<5>).
    • \n
    • the syntactic form {= some type =} allows writing an arbitrary type as target code. This is useful in target languages which have complex type grammar (e.g. in TypeScript, {= int | null =}).
    • \n
    \n

    Also note that to use strings conveniently in the C target, the “type” string is an alias for {=char*=}.

    \n

    (Types ending with a * are treated specially by the C target. See Sending and Receiving Arrays and Structs in the C target documentation.)

    \n

    Appendix: LF expressions

    \n

    A subset of LF syntax is used to write expressions, which represent target language values. Expressions are used in state variable initializers, default values for parameters, and parameter assignments.

    \n

    Expressions in LF support only simple forms, that are intended to be common across languages. Their precise meaning (e.g. the target language types they are compatible with) is target-specific and not specified here.

    \n

    Basic expressions

    \n

    The most basic expression forms, which are supported by all target languages, are the following:

    \n
      \n
    • \n

      Literals:

      \n
        \n
      • Numeric literals, e.g. 1, -120, 1.5. Note that the sign, if any, is part of the literal and must not be separated by whitespace.
      • \n
      • String literals, e.g. \"abcd\". String literals always use double-quotes, even in languages which support other forms (like Python).
      • \n
      • Character literals. e.g. 'a'. Single-quoted literals must be exactly one character long —even in Python.
      • \n
      • Boolean literals: true, false, True, False. The latter two are there for Python.
      • \n
      \n
    • \n
    • \n

      Parameter references, which are simple identifiers (e.g. foo). Any identifier in expression position must refer to a parameter of the enclosing reactor.

      \n
    • \n
    • \n

      Time values, e.g. 1 msec or 10 seconds. The syntax of time values is integer time_unit, where time_unit is one of the following

      \n
        \n
      • nsec: nanoseconds
      • \n
      • usec: microseconds
      • \n
      • msec: milliseconds
      • \n
      • sec or second: seconds
      • \n
      • minute: 60 seconds
      • \n
      • hour: 60 minutes
      • \n
      • day: 24 hours
      • \n
      • week: 7 days
      • \n
      \n

      Each of these units also support a pluralized version (e.g. nsecs, minutes, days), which means the same thing.

      \n

      The time value 0 may have no unit. Except in this specific case, the unit is always required.

      \n

      Time values are compatible with the time type.

      \n
    • \n
    • \n

      Escaped target-language expression, e.g. {= foo() =}. This syntax is used to write any expression which does not fall into one of the other forms described here. The contents are not parsed and are used verbatim in the generated file.

      \n

      The variables in scope are target-specific.

      \n
    • \n
    \n

    Complex expressions

    \n

    Some targets may make use of a few other syntactic forms for expressions. These syntactic forms may be ascribed a different meaning by different targets, to keep the source language close in meaning to the target language.

    \n

    We describe here these syntactic forms and what meaning they have in each target.

    \n
      \n
    • Bracket-list syntax, e.g. [1, 2, 3]. This syntax is used to create a list in Python. It is not supported by any other target at the moment.\n
      state x([1,2,3])\n
      \n
    • \n
    \n

    Initializer pseudo-expressions

    \n

    Some “expression” forms are only acceptable as the initializer of a state variable or parameter, but not in other places (like inside a list expression). These are

    \n
      \n
    • \n

      Tuple syntax, e.g. (1, 2, 3). This syntax is used:

      \n
        \n
      • \n

        in the Python target, to create a tuple value. Tuples are different from lists in that they are immutable.

        \n
      • \n
      • \n

        in C++, to pass arguments to a constructor:

        \n
        state x: int[](1,2);\n
        \n

        In that example, the initializer expression is translated to new std::vector(1,2). See also C++ target documentation.

        \n
      • \n
      • \n

        in C and all other targets, to create a target-specific array value. In the Python target, this is accomplished by the bracket-list syntax [1,2,3] instead. Note that to create a zero- or one-element array, fat braces are usually required. For instance in C:

        \n
      • \n
      \n
        state x: int[](1,2,3); // creates an int array, basically `int x[] = {1,2,3};`\n  state x: int[](1);     // `int x[] = 1;` - type error!\n  state x: int[]({= {1} =})  // one element array: `int x[] = {1};`\n
      \n
    • \n
    • \n

      Brace-list syntax, e.g. {1, 2, 3}. This syntax is at the moment only supported by the C++ target. It’s used to initialize a vector with the initializer list syntax instead of a constructor call.

      \n
    • \n
    ","headings":[{"value":"Target Language Specification","depth":2},{"value":"Import Statement","depth":2},{"value":"Reactor Block","depth":2},{"value":"Parameter Declaration","depth":3},{"value":"State Declaration","depth":3},{"value":"Method Declaration","depth":3},{"value":"Input Declaration","depth":3},{"value":"Output Declaration","depth":3},{"value":"Timer Declaration","depth":3},{"value":"Action Declaration","depth":3},{"value":"Discussion","depth":4},{"value":"Actions With Values","depth":3},{"value":"Reaction Declaration","depth":2},{"value":"Target Code","depth":3},{"value":"Scheduling Future Reactions","depth":3},{"value":"Asynchronous Callbacks","depth":3},{"value":"Superdense Time","depth":3},{"value":"Startup and Shutdown Reactions","depth":3},{"value":"Contained Reactors","depth":2},{"value":"Physical Connections","depth":3},{"value":"Connections with Delays","depth":3},{"value":"Deadlines","depth":2},{"value":"Comments","depth":2},{"value":"Appendix: LF types","depth":2},{"value":"Appendix: LF expressions","depth":2},{"value":"Basic expressions","depth":3},{"value":"Complex expressions","depth":3},{"value":"Initializer pseudo-expressions","depth":4}],"frontmatter":{"permalink":"/docs/handbook/language-specification","title":"Language Specification","oneline":"Language Specification for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/language-specification","repoPath":"/packages/documentation/copy/en/obsolete/Language Specification.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/language-specification","result":{"data":{"markdownRemark":{"id":"71ea83b3-e77d-5dfb-a0d9-c5f2dfa023e6","excerpt":"A Lingua Franca file, which has a .lf extension, contains the following: One target specification. Zero or more import statements. One or more reactor blocks…","html":"

    A Lingua Franca file, which has a .lf extension, contains the following:

    \n\n

    If one of the reactors in the file is designated main or federated, then the file defines an executable application. Otherwise, it defines one or more library reactors that can be imported into other LF files. For example, an LF file might be structured like this:

    \n
    target C;\nmain reactor C {\n    a = new A();\n    b = new B();\n    a.y -> b.x;\n}\nreactor A {\n    output y;\n    ...\n}\nreactor B {\n    input x;\n    ...\n}\n
    \n

    The name of the main reactor (C above) is optional. If given, it must match the filename (C.lf in the above example).

    \n

    This example specifies and instantiates two reactors, one of which sends messages to the other. A minimal but complete Lingua Franca file with one reactor is this:

    \n
    target C;\nmain reactor HelloWorld {\n    reaction(startup) {=\n        printf("Hello World.\\n");\n    =}\n}\n
    \n

    See the C target documentation for details about this example.

    \n

    Target Language Specification

    \n

    Every Lingua Franca program begins with a target language specification that specifies the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca compiler.

    \n

    Import Statement

    \n

    An import statement has the form:

    \n
    \n

    import { reactor1, reactor2 as alias2, […] } frompath“;

    \n
    \n

    where path specifies another Lingua Franca file relative to the location of the current file.

    \n

    Reactor Block

    \n

    A reactor is a software component that reacts to input events, timer events, and internal events. It has private state variables that are not visible to any other reactor. Its reactions can consist of altering its own state, sending messages to other reactors, or affecting the environment through some kind of actuation or side effect.

    \n

    The general structure of a reactor block is as follows:

    \n
    \n

    reactor name (parameters) {
    >   state declarations
    >   method declarations
    >   input declarations
    >   output declarations
    >   timer declarations
    >   action declarations
    >   reaction declarations
    >   contained reactors
    >    …
    \n}

    \n
    \n

    Parameter, inputs, outputs, timers, actions, and contained reactors all have names, and the names are required to be distinct from one another.

    \n

    If the reactor keyword is preceded by main, then this reactor will be instantiated and run by the generated code. If an imported LF file contains a main reactor, that reactor is ignored. Only reactors that not designated main are imported. This makes it easy to create a library of reusable reactors that each come with a test case or demonstration in the form of a main reactor.

    \n

    Parameter Declaration

    \n

    A reactor class definition can define parameters as follows:

    \n
    \n

    reactor ClassName(paramName1:type(expr), paramName2:type(expr)) {
    >    …
    \n}

    \n
    \n

    Each parameter may have a type annotation, written :type, and must have a default value, written (expr).

    \n

    The type annotation specifies a type in the target language, which is necessary for some target languages. For instance in C you might write

    \n
    reactor Foo(size: int(100)) {\n    ...\n}\n
    \n
    \nIntroduction to basic LF types and expressions... click to expand\n

    One useful type predefined by LF is the time type, which represents time durations. Values of this type may be written with time expressions, like 100 msec or 1 second (see Basic expressions for a reference).

    \n

    For instance, you can write the following in any target language:

    \n
    reactor Foo(period: time(100 msec)) {\n    ...\n}\n
    \n

    Container types may also be written e.g. int[], which is translated to a target-specific array or list type. The acceptable expressions for these types vary across targets (see Complex expressions), for instance in C, you can initialize an array parameter as follows:

    \n
    reactor Foo(my_array:int[](1, 2, 3)) {\n   ...\n}\n
    \n

    If the type or expression uses syntax that Lingua Franca does not support, you can use {= ... =} delimiters to enclose them and escape them. For instance to have a 2-dimensional array as a parameter in C:

    \n
    reactor Foo(param:{= int[][] =}({= { {1}, {2} } =})) {\n    ...\n}\n
    \n

    Both int[][] and {% raw %}{{1}, {2}} {% endraw %} are C fragments here, not LF.

    \n
    \n

    Other forms for types and expressions are described in LF types and LF expressions.

    \n

    How parameters may be used in the body of a reaction depends on the target. For example, in the C target, a self struct is provided that contains the parameter values. The following example illustrates this:

    \n
    target C;\nreactor Gain(scale:int(2)) {\n    input x:int;\n    output y:int;\n    reaction(x) -> y {=\n        lf_set(y, x->value * self->scale);\n    =}\n}\n
    \n

    This reactor, given any input event x will produce an output y with value equal to the input scaled by the scale parameter. The default value of the scale parameter is 2, but this can be changed when the Gain reactor is instantiated. The lf_set() is the mechanism provided by the C target for setting the value of outputs. The parameter scale and input x are just referenced in the C code as shown above.

    \n

    State Declaration

    \n

    A state declaration has one of the forms:

    \n
    \n

    state name:type(initial_value);
    \nstate name(parameter);

    \n
    \n

    In the first form, the type annotation is only required in some targets. The initial value may be any expression, including a special initializer forms.

    \n

    In the second form, the state variable inherits its type from the specified parameter, which also provides the initial value for the state variable.

    \n

    How state variables may be used in the body of a reaction depends on the target. For example, in the C target, a self struct is provided that contains the state values. The following example illustrates this:

    \n
    reactor Count {\n\toutput c:int;\n\ttimer t(0, 1 sec);\n\tstate i:int(0);\n\treaction(t) -> c {=\n\t\t(self->i)++;\n\t\tlf_set(c, self->i);\n\t=}\n}\n
    \n

    Method Declaration

    \n

    A method declaration has one of the forms:

    \n
    \n

    method name();
    \nmethod name():type;
    \nmethod name(arg1_name:arg1type, _arg2_name:arg2type, …);\nmethod _name(arg1_name:arg1type, _arg2_name:arg2type, …):_type;

    \n
    \n

    The first form defines a method with no arguments and no return value. The second form defines a method with the return type type but no arguments. The third form defines a method with arguments given by their name and type, but without a return value. Finally, the fourth form is similar to the third, but adds a return type.

    \n

    The method keyword can optionally be prefixed with the const qualifier, which indicates that the method is “read-only”. This is relevant for some target languages such as C++.

    \n

    See the C++ documentation for a usage example.

    \n

    Input Declaration

    \n

    An input declaration has the form:

    \n
    \n

    input name:type;

    \n
    \n

    The Gain reactor given above provides an example. The type is just like parameter types.

    \n

    An input may have the modifier mutable, as follows:

    \n
    \n

    mutable input name:type

    \n
    \n

    This is a directive to the code generator indicating that reactions that read this input will also modify the value of the input. Without this modifier, inputs are immutable; modifying them is disallowed. The precise mechanism for making use of mutable inputs is target-language specific. See, for example, the C language target.

    \n

    An input port may have more than one channel. See multiports documentation.

    \n

    Output Declaration

    \n

    An output declaration has the form:

    \n
    \n

    output name:type;

    \n
    \n

    The Gain reactor given above provides an example. The type is just like parameter types.

    \n

    An output port may have more than one channel. See multiports documentation.

    \n

    Timer Declaration

    \n

    A timer, like an input and an action, causes reactions to be invoked. Unlike an action, it is triggered automatically by the scheduler. This declaration is used when you want to invoke reactions once at specific times or periodically. A timer declaration has the form:

    \n
    \n

    timer name(offset, period);

    \n
    \n

    For example,

    \n
    timer foo(10 msec, 100 msec);\n
    \n

    This specifies a timer named foo that will first trigger 10 milliseconds after the start of execution and then repeatedly trigger at intervals of 100 ms. The units are optional, and if they are not included, then the number will be interpreted in a target-dependent way. The units supported are the same as in parameter declarations described above.

    \n

    The times specified are logical times. Specifically, if two timers have the same offset and period, then they are logically simultaneous. No observer will be able to see that one timer has triggered and the other has not. Even though these are logical times, the runtime system will make an effort to align those times to physical times. Such alignment can never be perfect, and its accuracy will depend on the execution platform.

    \n

    Both arguments are optional, with both having default value zero. An offset of zero or greater specifies the minimum time delay between the time at the start of execution and when the action is triggered. The period is zero or greater, where a value of zero specifies that the reactions should be triggered exactly once,\nwhereas a value greater than zero specifies that they should be triggered repeatedly with the period given.

    \n

    To cause a reaction to be invoked at the start of execution, a special startup trigger is provided:

    \n
    reactor Foo {\n    reaction(startup) {=\n        ... perform initialization ...\n    =}\n}\n
    \n

    The startup trigger is equivalent to a timer with no offset or period.

    \n

    Action Declaration

    \n

    An action, like an input, can cause reactions to be invoked. Whereas inputs are provided by other reactors, actions are scheduled by this reactor itself, either in response to some observed external event or as a delayed response to some input event. The action can be scheduled by a reactor by invoking a schedule function in a reaction or in an asynchronous callback function.

    \n

    An action declaration is either physical or logical:

    \n
    \n

    physical action name(min_delay, min_spacing, policy):type;
    > logical action name(min_delay, min_spacing, policy):type;

    \n
    \n

    The min_delay, min_spacing, and policy are all optional. If only one argument is given in parentheses, then it is interpreted as an min_delay, if two are given, then they are interpreted as min_delay and min_spacing, etc. The min_delay and min_spacing have to be a time value. The policy argument is a string that can be one of the following: 'defer' (default), 'drop', or 'replace'.

    \n

    An action will trigger at a logical time that depends on the arguments given to the schedule function, the min_delay, min_spacing, and policy arguments above, and whether the action is physical or logical.

    \n

    If the logical keyword is given, then the tag assigned to the event resulting from a call to schedule function is computed as follows. First, let t be the current logical time. For a logical action, the schedule function must be invoked from within a reaction (synchronously), so t is just the logical time of that reaction.

    \n

    The (preliminary) tag of the action is then just t plus min_delay plus the offset argument to schedule function.

    \n

    If the physical keyword is given, then the physical clock on the local platform is used as the timestamp assigned to the action. Moreover, for a physical action, unlike a logical action, the schedule function can be invoked from outside of any reaction (asynchronously), e.g. from an interrupt service routine or callback function.

    \n

    If a min_spacing has been declared, then a minimum distance between the tags of two subsequently scheduled events on the same action is enforced. If the preliminary tag is closer to the tag of the previously scheduled event (if there is one), then policy determines how the given constraints is enforced.

    \n
      \n
    • 'drop': the new event is dropped and schedule returns without having modified the event queue.
    • \n
    • 'replace': the payload of the new event is assigned to the preceding event if it is still pending in the event queue; no new event is added to the event queue in this case. If the preceding event has already been pulled from the event queue, the default 'defer' policy is applied.
    • \n
    • 'defer': the event is added to the event queue with a tag that is equal to earliest time that satisfies the minimal spacing requirement. Assuming the tag of the preceding event is t_prev, then the tag of the new event simply becomes t_prev + min_spacing.
    • \n
    \n

    Note that while the 'defer' policy is conservative in the sense that it does not discard events, it could potentially cause an unbounded growth of the event queue.

    \n

    In all cases, the logical time of a new event will always be strictly greater than the logical time at which it is scheduled by at least one microstep (see the Time section).

    \n

    The default min_delay is zero. The default min_spacing is undefined (meaning that no minimum spacing constraint is enforced). If a min_spacing is defined, it has to be strictly greater than zero, and greater than or equal to the time precision of the target (for the C target, it is one nanosecond).

    \n

    The min_delay parameter in the action declaration is static (set at compile time), while the offset parameter given to the schedule function may be dynamically set at runtime. Hence, for static analysis and scheduling, the action’s’ min_delay parameter can be assumed to be a minimum delay for analysis purposes.

    \n

    Discussion

    \n

    Logical actions are used to schedule events at a future logical time relative to the current logical time. Physical time is ignored. They must be scheduled within reactions, and the timestamp of the scheduled event will be relative to the current logical time of the reaction that schedules them. It is an error to schedule a logical action asynchronously, outside of the context of a reaction. Asynchronous actions are required to be physical.

    \n

    Physical actions are typically used to assign timestamps to externally triggered events, such as the arrival of a network message or the acquisition of sensor data, where the time at which these external events occurs is of interest. There are (at least) three interesting use cases:

    \n
      \n
    1. An asynchronous event, such as a callback function or interrupt service routine (ISR), is invoked at a physical time t and schedules an action with timestamp T=t. To get this behavior, just set the physical action to have min_delay = 0 and call the schedule function with offset = 0. The min_spacing can be useful here to prevent these external events from overwhelming the software system.
    2. \n
    3. A periodic task that is occasionally modified by a sporadic sensor. In this case, you can set min_delay = period and call schedule with offset = 0. The resulting timestamp of the sporadic sensor event will always align with the periodic events. This is similar to periodic polling, but without the overhead of polling the sensor when nothing interesting is happening.
    4. \n
    5. You can impose a minimum physical time delay between an event’s occurrence, such as a push of a button, and system response by adjusting the offset.
    6. \n
    \n

    Actions With Values

    \n

    If an action is declared with a type, then it can carry a value, a data value passed to the schedule function. This value will be available to any reaction that is triggered by the action. The specific mechanism, however, is target-language dependent. See the C target for an example.

    \n

    Reaction Declaration

    \n

    A reaction is defined within a reactor using the following syntax:

    \n
    \n

    reaction(triggers) uses -> effects {=
    >    … target language code …
    \n=}

    \n
    \n

    The uses and effects fields are optional. A simple example appears in the “hello world” example given above:

    \n
        reaction(t) {=\n        printf("Hello World.\\n");\n    =}\n
    \n

    In this example, t is a trigger (a timer named t). When that timer fires, the reaction will be invoked. Triggers can be timers, inputs, outputs of contained reactors, or actions. A comma-separated list of triggers can be given, in which case any of the specified triggers can trigger the reaction. If, at any logical time instant, more than one of the triggers fires, the reaction will nevertheless be invoked only once.

    \n

    The uses field specifies inputs that the reaction observes but that do not trigger the reaction. This field can also be a comma-separated list of inputs. Since the input does not trigger the reaction, the body of the reaction will normally need to test for presence of the input before using it. How to do this is target specific. See how this is done in the C target.

    \n

    The effects field, occurring after the right arrow, declares which outputs and actions the target code may produce or schedule. The effects field may also specify inputs of contained reactors, provided that those inputs do not have any other sources of data. These declarations make it possible for the reaction to send outputs or enable future actions, but they do not require that the reaction code do that.

    \n

    Target Code

    \n

    The body of the reaction is code in the target language surrounded by {= and =}. This code is not parsed by the Lingua Franca compiler. It is used verbatim in the program that is generated.

    \n

    The target provides language-dependent mechanisms for referring to inputs, outputs, and actions in the target code. These mechanisms can be different in each target language, but all target languages provide the same basic set of mechanisms. These mechanisms include:

    \n
      \n
    • Obtaining the current logical time (logical time does not advance during the execution of a reaction, so the execution of a reaction is logically instantaneous).
    • \n
    • Determining whether inputs are present at the current logical time and reading their value if they are. If a reaction is triggered by exactly one input, then that input will always be present. But if there are multiple triggers, or if the input is specified in the uses field, then the input may not be present when the reaction is invoked.
    • \n
    • Setting output values. Reactions in a reactor may set an output value more than once at any instant of logical time, but only the last of the values set will be sent on the output port.
    • \n
    • Scheduling future actions.
    • \n
    \n

    In the C target, for example, the following reactor will add two inputs if they are present at the time of a reaction:

    \n
    reactor Add {\n    input in1:int;\n    input in2:int;\n    output out:int;\n    reaction(in1, in2) -> out {=\n        int result = 0;\n        if (in1->is_present) result += in1->value;\n        if (in2->is_present) result += in2->value;\n        lf_set(out, result);\n    =}\n}\n
    \n

    See the C target for an example of how these things are specified in C.

    \n

    NOTE: if a reaction fails to test for the presence of an input and reads its value anyway, then the result it will get is undefined and may be target dependent. In the C target, as of this writing, the value read will be the most recently seen input value, or, if no input event has occurred at an earlier logical time, then zero or NULL, depending on the data type of the input. In the TS target, the value will be undefined, a legitimate value in TypeScript.

    \n

    Scheduling Future Reactions

    \n

    Each target language provides some mechanism for scheduling future reactions. Typically, this takes the form of a schedule function that takes as an argument an action, a time interval, and (perhaps optionally), a payload. For example, in the C target, in the following program, each reaction to the timer t schedules another reaction to occur 100 msec later:

    \n
    target C;\nmain reactor Schedule {\n    timer t(0, 1 sec);\n    logical action a;\n    reaction(t) -> a {=\n        schedule(a, MSEC(100));\n    =}\n    reaction(a) {=\n        printf("Nanoseconds since start: %lld.\\n", lf_time_logical_elapsed());\n    =}\n}\n
    \n

    When executed, this will produce the following output:

    \n
    Start execution at time Sun Aug 11 04:11:57 2019\nplus 919310000 nanoseconds.\nNanoseconds since start: 100000000.\nNanoseconds since start: 1100000000.\nNanoseconds since start: 2100000000.\n...
    \n

    This action has no data type and carries no value, but, as explained below, an action can carry a value.

    \n

    Asynchronous Callbacks

    \n

    In targets that support multitasking, the schedule function, which schedules future reactions, may be safely invoked on a physical action in code that is not part of a reaction. For example, in the multithreaded version of the C target, schedule may be invoked in an interrupt service routine. The reaction(s) that are scheduled are guaranteed to occur at a time that is strictly larger than the current logical time of any reactions that are being interrupted.

    \n

    Superdense Time

    \n

    Lingua Franca uses a concept known as superdense time, where two time values that appear to be the same are not logically simultaneous. At every logical time value, for example midnight on January 1, 1970, there exist a logical sequence of microsteps that are not simultaneous. The Microsteps example illustrates this:

    \n
    target C;\nreactor Destination {\n    input x:int;\n    input y:int;\n    reaction(x, y) {=\n        printf("Time since start: %lld.\\n", lf_time_logical_elapsed());\n        if (x->is_present) {\n            printf("  x is present.\\n");\n        }\n        if (y->is_present) {\n            printf("  y is present.\\n");\n        }\n    =}\n}\nmain reactor Microsteps {\n    timer start;\n    logical action repeat;\n    d = new Destination();\n    reaction(start) -> d.x, repeat {=\n        lf_set(d.x, 1);\n        schedule(repeat, 0);\n    =}\n    reaction(repeat) -> d.y {=\n        lf_set(d.y, 1);\n    =}\n}\n
    \n

    The Destination reactor has two inputs, x and y, and it simply reports at each logical time where either is present what is the logical time and which is present. The Microsteps reactor initializes things with a reaction to the one-time timer event start by sending data to the x input of Destination. It then schedules a repeat action.

    \n

    Note that time delay in the call to schedule is zero. However, any reaction scheduled by schedule is required to occur strictly later than current logical time. In Lingua Franca, this is handled by scheduling the repeat reaction to occur one microstep later. The output printed, therefore, will look like this:

    \n
    Time since start: 0.\n  x is present.\nTime since start: 0.\n  y is present.
    \n

    Note that the numerical time reported by get_elapsed_logical_time() has not advanced in the second reaction, but the fact that x is not present in the second reaction proves that the first reaction and the second are not logically simultaneous. The second occurs one microstep later.

    \n

    Note that it is possible to write code that will prevent logical time from advancing except by microsteps. For example, we could replace the reaction to repeat in Main with this one:

    \n
        reaction(repeat) -> d.y, repeat {=\n        lf_set(d.y, 1);\n        schedule(repeat, 0);\n    =}\n
    \n

    This would create what is known as a stuttering Zeno condition, where logical time cannot advance. The output will be an unbounded sequence like this:

    \n
    Time since start: 0.\n  x is present.\nTime since start: 0.\n  y is present.\nTime since start: 0.\n  y is present.\nTime since start: 0.\n  y is present.\n...
    \n

    Startup and Shutdown Reactions

    \n

    Two special triggers are supported, startup and shutdown. A reaction that specifies the startup trigger will be invoked at the start of execution of the model. The following two syntaxes have exactly the same effect:

    \n
        reaction(startup) {= ... =}\n
    \n

    and

    \n
        timer t;\n    reaction(t) {= ... =}\n
    \n

    In other words, startup is a timer that triggers once at the first logical time of execution. As with any other reaction, the reaction can also be triggered by inputs and can produce outputs or schedule actions.

    \n

    The shutdown trigger is slightly different. A shutdown reaction is specified as follows:

    \n
       reaction(shutdown) {= ... =}\n
    \n

    This reaction will be invoked when the program terminates normally (there are no more events, some reaction has called a request_stop() utility provided in the target language, or the execution was specified to last a finite logical time). The reaction will be invoked at a logical time one microstep later than the last logical time of the execution. In other words, the presence of this reaction means that the program will execute one extra logical time cycle beyond what it would have otherwise, and that logical time is one microstep later than what would have otherwise been the last logical time.

    \n

    If the reaction produces outputs, then downstream reactors will also be invoked at that later logical time. If the reaction schedules future reactions, those will be ignored. After the completion of this final logical time cycle, one microstep later than the normal termination, the program will exit.

    \n

    Contained Reactors

    \n

    Reactors can contain instances of other reactors defined in the same file or in an imported file. Assuming the above Count reactor is stored in a file Count.lf, then CountTest is an example that imports and instantiates it to test the reactor:

    \n
    target C;\nimport Count.lf;\nreactor Test {\n    input c:int;\n    state i:int(0);\n    reaction(c) {=\n        printf("Received %d.\\n", c->value);\n        (self->i)++;\n        if (c->value != self->i) {\n            printf("ERROR: Expected %d but got %d\\n.", self->i, c->value);\n            exit(1);\n        }\n    =}\n    reaction(shutdown) {=\n        if (self->i != 4) {\n            printf("ERROR: Test should have reacted 4 times, but reacted %d times.\\n", self->i);\n            exit(2);\n        }\n    =}\n}\n\nmain reactor CountTest {\n    count = new Count();\n    test = new Test();\n    count.out -> test.c;\n}\n
    \n

    An instance is created with the syntax:

    \n
    \n

    instance_name = new class_name(parameters);

    \n
    \n

    A bank with several instances can be created in one such statement, as explained in the banks of reactors documentation.

    \n

    The parameters argument has the form:

    \n
    \n

    parameter1_name = parameter1_value, parameter2_name = parameter2_value, …

    \n
    \n

    Connections between ports are specified with the syntax:

    \n
    \n

    output_port -> input_port

    \n
    \n

    where the ports are either instance_name.port_name or just port_name, where the latter form denotes a port belonging to the reactor that contains the instances.

    \n

    Physical Connections

    \n

    A subtle and rarely used variant is a physical connection, denoted ~>. In such a connection, the logical time at the recipient is derived from the local physical clock rather than being equal to the logical time at the sender. The physical time will always exceed the logical time of the sender, so this type of connection incurs a nondeterministic positive logical time delay. Physical connections are useful sometimes in Distributed Execution in situations where the nondeterministic logical delay is tolerable. Such connections are more efficient because timestamps need not be transmitted and messages do not need to flow through through a centralized coordinator (if a centralized coordinator is being used).

    \n

    Connections with Delays

    \n

    Connections may include a logical delay using the after keyword, as follows:

    \n
    \n

    output_port -> input_port after 10 msec

    \n
    \n

    This means that the logical time of the message delivered to the input_port will be 10 milliseconds larger than the logical time of the reaction that wrote to output_port. If the time value is greater than zero, then the event will appear at microstep 0. If it is equal to zero, then it will appear at the current microstep plus one.

    \n

    When there are multiports or banks of reactors, several channels can be connected with a single connection statement. See Multiports and Banks of Reactors.

    \n

    The following example defines a reactor that adds a counting sequence to its input. It uses the above Count and Add reactors (see Hierarchy2):

    \n
    import Count.lf;\nimport Add.lf;\nreactor AddCount {\n    input in:int;\n    output out:int;\n    count = new Count();\n    add = new Add();\n    in -> add.in1;\n    count.out -> add.in2;\n    add.out -> out;\n}\n
    \n

    A reactor that contains other reactors may, within a reaction, send data to the contained reactor. The following example illustrates this (see SendingInside):

    \n
    target C;\nreactor Printer {\n\tinput x:int;\n\treaction(x) {=\n\t\tprintf("Inside reactor received: %d\\n", x->value);\n\t=}\n}\nmain reactor SendingInside {\n\tp = new Printer();\n\treaction(startup) -> p.x {=\n\t\tlf_set(p.x, 1);\n\t=}\n}\n
    \n

    Running this will print:

    \n
    Inside reactor received: 1
    \n

    Deadlines

    \n

    Lingua Franca includes a notion of a deadline, which is a relation between logical time and physical time. Specifically, a program may specify that the invocation of a reaction must occur within some physical-time interval of the logical timestamp of the message. If a reaction is invoked at logical time 12 noon, for example, and the reaction has a deadline of one hour, then the reaction is required to be invoked before the physical-time clock of the execution platform reaches 1 PM. If the deadline is violated, then the specified deadline handler is invoked instead of the reaction. For example (see Deadline):

    \n
    reactor Deadline() {\n    input x:int;\n    output d:int; // Produced if the deadline is violated.\n    reaction(x) -> d {=\n        printf("Normal reaction.\\n");\n    =} deadline(10 msec) {=\n        printf("Deadline violation detected.\\n");\n        lf_set(d, x->value);\n    =}\n
    \n

    This reactor specifies a deadline of 10 milliseconds (this can be a parameter of the reactor). If the reaction to x is triggered later in physical time than 10 msec past the timestamp of x, then the second body of code is executed instead of the first. That second body of code has access to anything the first body of code has access to, including the input x and the output d. The output can be used to notify the rest of the system that a deadline violation occurred.

    \n

    The amount of the deadline, of course, can be given by a parameter.

    \n

    A sometimes useful pattern is when a container reactor reacts to deadline violations in a contained reactor. The DeadlineHandledAbove example illustrates this:

    \n
    target C;\nreactor Deadline() {\n    input x:int;\n    output deadline_violation:bool;\n    reaction(x) -> deadline_violation {=\n        ... normal code to execute ...\n    =} deadline(100 msec) {=\n        printf("Deadline violation detected.\\n");\n        lf_set(deadline_violation, true);\n    =}\n}\nmain reactor DeadlineHandledAbove {\n    d = new Deadline();\n    ...\n    reaction(d.deadline_violation) {=\n        ... handle the deadline violation ...\n    =}\n}\n
    \n

    Comments

    \n

    Lingua Franca files can have C/C++/Java-style comments and/or Python-style comments. All of the following are valid comments:

    \n
        // Single-line C-style comment.\n    /*\n       Multi-line C-style comment.\n     */\n    # Single-line Python-style comment.\n    '''\n       Multi-line Python-style comment.\n    '''\n
    \n

    Appendix: LF types

    \n

    Type annotations may be written in many places in LF, including parameter declarations, state variable declarations, input and output declarations. In some targets, they are required, because the target language requires them too.

    \n

    Assigning meaning to type annotations is entirely offloaded to the target compiler, as LF does not feature a type system (yet?). However, LF’s syntax for types supports a few idioms that have target-specific meaning. Types may have the following forms:

    \n
      \n
    • the time type is reserved by LF, its values represent time durations. The time type accepts time expressions for values, e.g. 100 msec, or 0 (see Basic expressions for a reference).
    • \n
    • identifiers are valid types (e.g. int, size_t), and may be followed by type arguments (e.g. vector<int>).
    • \n
    • the syntactic forms type[] and type[integer] correspond to target-specific array types. The second form is available only in languages which support fixed-size array types (e.g. in C++, std::array<5>).
    • \n
    • the syntactic form {= some type =} allows writing an arbitrary type as target code. This is useful in target languages which have complex type grammar (e.g. in TypeScript, {= int | null =}).
    • \n
    \n

    Also note that to use strings conveniently in the C target, the “type” string is an alias for {=char*=}.

    \n

    (Types ending with a * are treated specially by the C target. See Sending and Receiving Arrays and Structs in the C target documentation.)

    \n

    Appendix: LF expressions

    \n

    A subset of LF syntax is used to write expressions, which represent target language values. Expressions are used in state variable initializers, default values for parameters, and parameter assignments.

    \n

    Expressions in LF support only simple forms, that are intended to be common across languages. Their precise meaning (e.g. the target language types they are compatible with) is target-specific and not specified here.

    \n

    Basic expressions

    \n

    The most basic expression forms, which are supported by all target languages, are the following:

    \n
      \n
    • \n

      Literals:

      \n
        \n
      • Numeric literals, e.g. 1, -120, 1.5. Note that the sign, if any, is part of the literal and must not be separated by whitespace.
      • \n
      • String literals, e.g. \"abcd\". String literals always use double-quotes, even in languages which support other forms (like Python).
      • \n
      • Character literals. e.g. 'a'. Single-quoted literals must be exactly one character long —even in Python.
      • \n
      • Boolean literals: true, false, True, False. The latter two are there for Python.
      • \n
      \n
    • \n
    • \n

      Parameter references, which are simple identifiers (e.g. foo). Any identifier in expression position must refer to a parameter of the enclosing reactor.

      \n
    • \n
    • \n

      Time values, e.g. 1 msec or 10 seconds. The syntax of time values is integer time_unit, where time_unit is one of the following

      \n
        \n
      • nsec: nanoseconds
      • \n
      • usec: microseconds
      • \n
      • msec: milliseconds
      • \n
      • sec or second: seconds
      • \n
      • minute: 60 seconds
      • \n
      • hour: 60 minutes
      • \n
      • day: 24 hours
      • \n
      • week: 7 days
      • \n
      \n

      Each of these units also support a pluralized version (e.g. nsecs, minutes, days), which means the same thing.

      \n

      The time value 0 may have no unit. Except in this specific case, the unit is always required.

      \n

      Time values are compatible with the time type.

      \n
    • \n
    • \n

      Escaped target-language expression, e.g. {= foo() =}. This syntax is used to write any expression which does not fall into one of the other forms described here. The contents are not parsed and are used verbatim in the generated file.

      \n

      The variables in scope are target-specific.

      \n
    • \n
    \n

    Complex expressions

    \n

    Some targets may make use of a few other syntactic forms for expressions. These syntactic forms may be ascribed a different meaning by different targets, to keep the source language close in meaning to the target language.

    \n

    We describe here these syntactic forms and what meaning they have in each target.

    \n
      \n
    • Bracket-list syntax, e.g. [1, 2, 3]. This syntax is used to create a list in Python. It is not supported by any other target at the moment.\n
      state x([1,2,3])\n
      \n
    • \n
    \n

    Initializer pseudo-expressions

    \n

    Some “expression” forms are only acceptable as the initializer of a state variable or parameter, but not in other places (like inside a list expression). These are

    \n
      \n
    • \n

      Tuple syntax, e.g. (1, 2, 3). This syntax is used:

      \n
        \n
      • \n

        in the Python target, to create a tuple value. Tuples are different from lists in that they are immutable.

        \n
      • \n
      • \n

        in C++, to pass arguments to a constructor:

        \n
        state x: int[](1,2);\n
        \n

        In that example, the initializer expression is translated to new std::vector(1,2). See also C++ target documentation.

        \n
      • \n
      • \n

        in C and all other targets, to create a target-specific array value. In the Python target, this is accomplished by the bracket-list syntax [1,2,3] instead. Note that to create a zero- or one-element array, fat braces are usually required. For instance in C:

        \n
      • \n
      \n
        state x: int[](1,2,3); // creates an int array, basically `int x[] = {1,2,3};`\n  state x: int[](1);     // `int x[] = 1;` - type error!\n  state x: int[]({= {1} =})  // one element array: `int x[] = {1};`\n
      \n
    • \n
    • \n

      Brace-list syntax, e.g. {1, 2, 3}. This syntax is at the moment only supported by the C++ target. It’s used to initialize a vector with the initializer list syntax instead of a constructor call.

      \n
    • \n
    ","headings":[{"value":"Target Language Specification","depth":2},{"value":"Import Statement","depth":2},{"value":"Reactor Block","depth":2},{"value":"Parameter Declaration","depth":3},{"value":"State Declaration","depth":3},{"value":"Method Declaration","depth":3},{"value":"Input Declaration","depth":3},{"value":"Output Declaration","depth":3},{"value":"Timer Declaration","depth":3},{"value":"Action Declaration","depth":3},{"value":"Discussion","depth":4},{"value":"Actions With Values","depth":3},{"value":"Reaction Declaration","depth":2},{"value":"Target Code","depth":3},{"value":"Scheduling Future Reactions","depth":3},{"value":"Asynchronous Callbacks","depth":3},{"value":"Superdense Time","depth":3},{"value":"Startup and Shutdown Reactions","depth":3},{"value":"Contained Reactors","depth":2},{"value":"Physical Connections","depth":3},{"value":"Connections with Delays","depth":3},{"value":"Deadlines","depth":2},{"value":"Comments","depth":2},{"value":"Appendix: LF types","depth":2},{"value":"Appendix: LF expressions","depth":2},{"value":"Basic expressions","depth":3},{"value":"Complex expressions","depth":3},{"value":"Initializer pseudo-expressions","depth":4}],"frontmatter":{"permalink":"/docs/handbook/language-specification","title":"Language Specification","oneline":"Language Specification for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/language-specification","repoPath":"/packages/documentation/copy/en/obsolete/Language Specification.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/logical-execution-time/page-data.json b/page-data/docs/handbook/logical-execution-time/page-data.json index 2500b0852..4808ad58d 100644 --- a/page-data/docs/handbook/logical-execution-time/page-data.json +++ b/page-data/docs/handbook/logical-execution-time/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/logical-execution-time","result":{"data":{"markdownRemark":{"id":"6f6a2de3-be2a-5e00-ba14-2bb0974cdec1","excerpt":"FIXME","html":"

    FIXME

    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/logical-execution-time","title":"Logical Execution Time","oneline":"Reactions that take non-zero logical time to execute.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/logical-execution-time","repoPath":"/packages/documentation/copy/en/less-developed/Logical Execution Time.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/logical-execution-time","result":{"data":{"markdownRemark":{"id":"6f6a2de3-be2a-5e00-ba14-2bb0974cdec1","excerpt":"FIXME","html":"

    FIXME

    ","headings":[],"frontmatter":{"permalink":"/docs/handbook/logical-execution-time","title":"Logical Execution Time","oneline":"Reactions that take non-zero logical time to execute.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/logical-execution-time","repoPath":"/packages/documentation/copy/en/less-developed/Logical Execution Time.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/methods/page-data.json b/page-data/docs/handbook/methods/page-data.json index 237792a0d..119b7c519 100644 --- a/page-data/docs/handbook/methods/page-data.json +++ b/page-data/docs/handbook/methods/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/methods","result":{"data":{"markdownRemark":{"id":"dd7daec6-0b6f-5466-ad7a-8b8332d3f262","excerpt":"Method Declaration The $target-language$ target does not currently support methods. Sometimes logic needs to be shared between reactions. In this case, methods…","html":"

    Method Declaration

    \n
    \n

    The $target-language$ target does not currently support methods.

    \n
    \n
    \n
    \n

    Sometimes logic needs to be shared between reactions. In this case, methods can be used to implement the shared logic, and these methods can then be called from reaction bodies. A method declaration has one of the forms:

    \n
      method <name>() {= ... =}\n  method <name>():<type> {= ... =}\n  method <name>(<argument_name>:<type>, ...) {= ... =}\n  method <name>(<argument_name>:<type>, ...):<type> {= ... =}\n
    \n

    The first form defines a method with no arguments and no return value. The second form defines a method with the return type <type> but no arguments. The third form defines a method with a comma-separated list of arguments given by their name and type, but without a return value. Finally, the fourth form is similar to the third, but adds a return type.

    \n
    \n
    \n

    A method declaration has the forms:

    \n
      method <name>() {= ... =}\n
    \n
    \n
    \n

    The $method$ keyword can optionally be prefixed with the $const$ qualifier, which indicates that the method has only read access to the reactor’s state.

    \n
    \n

    Methods are particularly useful in reactors that need to perform certain operations on state variables and/or parameters that are shared between reactions or that are too complex to be implemented in a single reaction. Analogous to class methods, methods in LF can access all state variables and parameters, and can be invoked from all reaction bodies or from other methods. Methods may also recursively invoke themselves. Consider the following example:

    \n

    $start(Methods)$

    \n
    target C\nmain reactor Methods {\n  state foo: int = 2\n  method getFoo(): int {=\n    return self->foo;\n  =}\n  method add(x: int) {=\n    self->foo += x;\n  =}\n  reaction(startup) {=\n    lf_print("Foo is initialized to %d", getFoo());\n    add(40);\n    lf_print("2 + 40 = %d", getFoo());\n  =}\n}\n
    \n
    target Cpp\nmain reactor Methods {\n  state foo: int(2)\n  const method getFoo(): int {=\n    return foo;\n  =}\n  method add(x: int) {=\n    foo += x;\n  =}\n  reaction(startup) {=\n    std::cout << "Foo is initialized to " << getFoo() << '\\n';\n    add(40);\n    std::cout << "2 + 40 = " << getFoo() << '\\n';\n  =}\n}\n
    \n
    target Python\nmain reactor Methods {\n  state foo = 2\n  method getFoo() {=\n    return self.foo\n  =}\n  method add(x) {=\n    self.foo += x\n  =}\n  reaction(startup) {=\n    print(f"Foo is initialized to {self.getFoo()}.")\n    self.add(40)\n    print(f"2 + 40 = {self.getFoo()}.")\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/Methods.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/Methods.lf\n
    \n

    $end(Methods)$

    \n

    This reactor defines two methods getFoo and add.\n\ngetFoo is qualified as a const method, which indicates that it has read-only\naccess to the state variables. This is directly translated to a C++ const method\nin the code generation process.\n\nThe getFoo method receives no arguments and returns an integer (int)\nindicating the current value of the foo state variable. The add method\nreturns nothing\n\n(void)\n\nand receives one integer argument, which it uses to increment foo. Both\nmethods are visible in all reactions of the reactor. In this example, the\nreaction to startup calls both methods in order to read and modify its state.

    \n
    ","headings":[{"value":"Method Declaration","depth":2}],"frontmatter":{"permalink":"/docs/handbook/methods","title":"Methods","oneline":"Methods in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Reactions","oneline":"Reactions in Lingua Franca.","permalink":"/docs/handbook/reactions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","permalink":"/docs/handbook/causality-loops"}}}},"pageContext":{"id":"1-methods","slug":"/docs/handbook/methods","repoPath":"/packages/documentation/copy/en/topics/Methods.md","previousID":"25ac2513-8979-52dd-9176-b0db61f55dc9","nextID":"de5af6b0-de72-5890-9668-c4f000ffdb2c","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/methods","result":{"data":{"markdownRemark":{"id":"dd7daec6-0b6f-5466-ad7a-8b8332d3f262","excerpt":"Method Declaration The $target-language$ target does not currently support methods. Sometimes logic needs to be shared between reactions. In this case, methods…","html":"

    Method Declaration

    \n
    \n

    The $target-language$ target does not currently support methods.

    \n
    \n
    \n
    \n

    Sometimes logic needs to be shared between reactions. In this case, methods can be used to implement the shared logic, and these methods can then be called from reaction bodies. A method declaration has one of the forms:

    \n
      method <name>() {= ... =}\n  method <name>():<type> {= ... =}\n  method <name>(<argument_name>:<type>, ...) {= ... =}\n  method <name>(<argument_name>:<type>, ...):<type> {= ... =}\n
    \n

    The first form defines a method with no arguments and no return value. The second form defines a method with the return type <type> but no arguments. The third form defines a method with a comma-separated list of arguments given by their name and type, but without a return value. Finally, the fourth form is similar to the third, but adds a return type.

    \n
    \n
    \n

    A method declaration has the forms:

    \n
      method <name>() {= ... =}\n
    \n
    \n
    \n

    The $method$ keyword can optionally be prefixed with the $const$ qualifier, which indicates that the method has only read access to the reactor’s state.

    \n
    \n

    Methods are particularly useful in reactors that need to perform certain operations on state variables and/or parameters that are shared between reactions or that are too complex to be implemented in a single reaction. Analogous to class methods, methods in LF can access all state variables and parameters, and can be invoked from all reaction bodies or from other methods. Methods may also recursively invoke themselves. Consider the following example:

    \n

    $start(Methods)$

    \n
    target C\nmain reactor Methods {\n  state foo: int = 2\n  method getFoo(): int {=\n    return self->foo;\n  =}\n  method add(x: int) {=\n    self->foo += x;\n  =}\n  reaction(startup) {=\n    lf_print("Foo is initialized to %d", getFoo());\n    add(40);\n    lf_print("2 + 40 = %d", getFoo());\n  =}\n}\n
    \n
    target Cpp\nmain reactor Methods {\n  state foo: int(2)\n  const method getFoo(): int {=\n    return foo;\n  =}\n  method add(x: int) {=\n    foo += x;\n  =}\n  reaction(startup) {=\n    std::cout << "Foo is initialized to " << getFoo() << '\\n';\n    add(40);\n    std::cout << "2 + 40 = " << getFoo() << '\\n';\n  =}\n}\n
    \n
    target Python\nmain reactor Methods {\n  state foo = 2\n  method getFoo() {=\n    return self.foo\n  =}\n  method add(x) {=\n    self.foo += x\n  =}\n  reaction(startup) {=\n    print(f"Foo is initialized to {self.getFoo()}.")\n    self.add(40)\n    print(f"2 + 40 = {self.getFoo()}.")\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/Methods.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/Methods.lf\n
    \n

    $end(Methods)$

    \n

    This reactor defines two methods getFoo and add.\n\ngetFoo is qualified as a const method, which indicates that it has read-only\naccess to the state variables. This is directly translated to a C++ const method\nin the code generation process.\n\nThe getFoo method receives no arguments and returns an integer (int)\nindicating the current value of the foo state variable. The add method\nreturns nothing\n\n(void)\n\nand receives one integer argument, which it uses to increment foo. Both\nmethods are visible in all reactions of the reactor. In this example, the\nreaction to startup calls both methods in order to read and modify its state.

    \n
    ","headings":[{"value":"Method Declaration","depth":2}],"frontmatter":{"permalink":"/docs/handbook/methods","title":"Methods","oneline":"Methods in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Reactions","oneline":"Reactions in Lingua Franca.","permalink":"/docs/handbook/reactions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Causality Loops","oneline":"Causality loops in Lingua Franca.","permalink":"/docs/handbook/causality-loops"}}}},"pageContext":{"id":"1-methods","slug":"/docs/handbook/methods","repoPath":"/packages/documentation/copy/en/topics/Methods.md","previousID":"25ac2513-8979-52dd-9176-b0db61f55dc9","nextID":"de5af6b0-de72-5890-9668-c4f000ffdb2c","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/modal-models/page-data.json b/page-data/docs/handbook/modal-models/page-data.json index ace57db53..e1273107c 100644 --- a/page-data/docs/handbook/modal-models/page-data.json +++ b/page-data/docs/handbook/modal-models/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/modal-models","result":{"data":{"markdownRemark":{"id":"3ab36bba-9f0e-54a5-bfd0-55f5e5259ca5","excerpt":"Modal Reactors are currently not supported in $target-language$. $page-showing-target$ The basic idea of modal reactors is to partition a reactor into disjoint…","html":"
    \n

    Modal Reactors are currently not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    The basic idea of modal reactors is to partition a reactor into disjoint subsets of reactions (or other components) that are associated with mutually exclusive modes.\nIn a modal reactor, only a single mode can be active at a particular logical time instant, meaning that activity in other modes is automatically suspended.\nTransitioning between modes switches the reactor’s behavior and provides control over continuing or resetting the previous history of the entered mode.

    \n

    Syntax

    \n

    Modes can be defined in any reactor, except federated ones.\nEach mode requires a unique (per reactor) name and can declare contents that are local to this mode.\nThere must be exactly one mode marked as $initial$.

    \n
    reactor TwoModes {\n  ...\n  initial mode One {\n    ...\n  }\n  mode Two {\n    ...\n  }\n}\n
    \n

    A mode can contain local state variables, timers, actions, reactions, reactor instantiations, and connections.\nWhile modes cannot be nested in modes directly, hierarchical composition is possible through the instantiation of modal reactors.\nThe main exception in allowed contents in modes are port declarations, as these are only possible on reactor level.\nYet, modes share the scope with their reactor and, hence, can access ports, state variables, and parameters of the reactor.\nOnly the contents of other modes are excluded.\nA modal reactor can still have reactions, reactor instantiations, etc., that are not located in modes and will consequently be executed independently from mode activity.

    \n

    Mode transitions are declared within reactions.\nIf a reactor has modes, reactions inside modes are allowed to list them as effects.\nSuch an effect enables the use of the target language API to set the next mode.

    \n
    reaction(trig) -> Two {=\n  if (trig->value) {\n    lf_set_mode(Two)\n  }\n=}\n
    \n
    reaction(trig) -> Two {=\n  if trig.value:\n    Two.set()\n=}\n
    \n

    You can also specify the type of the transition by adding the modifier reset(<mode>) or history(<mode>) in the effects of the\nreaction signature (i.e., after the ->). For example, a history transition to the state Two is enabled by listing history(Two)\namong the effects of the reaction. The reset variant is implicitly assumed when the mode is listed without modifier.

    \n

    Execution Semantics

    \n

    The basic effect of modes is that only parts that are contained in the currently active mode (or outside any mode) are executed at any point in time.\nThis also holds for parts that are nested in multiple ancestor modes due to hierarchy; consequently, all those ancestors must be active in order to execute.\nReactions in inactive modes are simply not executed, while all components that model timing behavior, namely timers, scheduled actions, and delayed connections, are subject to a concept of local time.\nThat means while a mode is inactive, the progress of time is suspended locally.\nHow the timing components behave when a mode becomes active depends on the transition type.\nA mode can be reset upon entry, returning it to its initial state.\nAlternatively, it may be entered preserving its history, which only has an actual effect if the mode was active before.\nIn the latter case all timing components will continue their delays or period as if no time had passed during inactivity of the mode.\nThe following section will provide a more detailed explanation of this effect.

    \n

    Upon reactor startup, the initial mode of each modal reactor is active, others are inactive.\nIf at a tag (t, m), all reactions of this reactor and all its contents have finished executing, and a new mode was set in a reaction, the current mode will be deactivated and the new one will be activated for future execution.\nThis means no reaction of the newly active mode will execute at tag (t, m); the earliest possible reaction in the new mode occurs one microstep later, at (t, m+1).\nHence, if the newly active mode has for example a timer that will elapse with an offset of zero, it will trigger at (t, m+1).\nIn case the mode itself does not require an immediate execution in the next microstep, it depends on future events, just as in the normal behavior of LF.\nHence, modes in the same reactor are always mutually exclusive w.r.t. superdense time.

    \n

    A transition is triggered if a new mode is set in a reaction body.\nAs with setting output ports in reaction, a new mode can be set multiple times in the same or different reactions.\nIn the end, the fixed ordering of reactions determines the last effective value that will be used.\nThe new mode does not have to be a different one; it is possible for a mode to reset itself via a reset transition.

    \n

    In case a mode is entered with the reset behavior:

    \n
      \n
    • all contained modal reactors are reset to their initial mode (recursively),
    • \n
    • all contained timers are reset and start again awaiting their initial offset,
    • \n
    • all contained state variables that are marked for automatic reset are reset to their initial value,
    • \n
    • any contained reactions triggered by reset are executed, and
    • \n
    • all events (actions, timers, delayed connections) that were previously scheduled from within this mode are discarded.
    • \n
    \n

    Note that contained refers to all contents defined locally in the mode and in local reactor instances (recursively) that are not otherwise enclosed in modes of lower levels.

    \n

    Whenever a mode is entered with a reset transition, the subsequent timing behavior is as if the mode was never executed before.\nIf there are state variables that need to be reset or reinitialized, then this can be done in a reaction triggered by reset or by marking the state variable for automatic reset (e.g.,\nreset state x:int(0)\nFIXME\nreset state x(0)\nFIXME\nFIXME\n).\nState variables are not reset automatically to their initial conditions because it is idiomatic for reactors to allocate resources or initialize subsystems (e.g., allocate memory or sockets, register an interrupt, or start a server) in reactions triggered by the startup, and to store references to these resources in state variables.\nIf these were to be automatically reset, those references would be lost.

    \n

    On the other hand, if a mode has been active prior and is then re-entered via a history transition, no reset is performed.\nEvents originating from timers, scheduled actions, and delayed connections are adjusted to reflect a remaining delay equal to the remaining delay recorded at the instant the mode was previously deactivated.\nAs a consequence, a mode has a notion of local time that elapses only when the mode is active.

    \n

    Local Time

    \n

    From the perspective of timers and actions, time is suspended when a mode is inactive.\nThis also applies to indirectly nested reactors within modes and connections with logical delays, if their source lies within a mode.

    \n\"Illustration\n

    The above LF model illustrates the different characteristics of local time affecting timers and actions in the presence of the two transition types.

    \n

    It consists of two modes One (the initial mode) and Two, both in the Modal reactor.\nThe next input toggles between these modes and is controlled by a reaction at the top level that is triggered by the timer T.\nAfter one second, a mode switch is triggered periodically with a period one second.\nEach mode has a timer T1/T2 that triggers a reaction after an initial offset of 100 msec and then periodically after 750 msec.\nThis reaction then schedules a logical action with a delay of 500 msec (the actual target code does not add an additional delay over the minimum specified).\nThis action triggers the second reaction, which writes to the output out.\nThe main difference between the modes is that One is entered via a history transition, continuing its behavior, while Two is reset.\n(History behavior is indicated by an “H” on the transition edge because it enters into the entire history of the mode.)

    \n\"Illustration\n

    Above is the execution trace of the first 4 seconds of this program.\nBelow the timeline is the currently active mode and above the timeline are the model elements that are executed at certain points in time, together with indicating triggering and their relation through time.\nFor example, at 100 msec, the initial offset of timer T1 elapses, which leads to the scheduling of the logical action in this mode.\nThe action triggers the reaction 500 msec later, at 600 msec, and thus causes an output.\nThe timing diagram illustrates the different handling of time between history transitions and reset transitions.\nSpecifically, when mode One is re-entered via a history transition, at time 2000 msec, the action triggered by T1 before, at time 850 msec, resumes.\nIn contrast, when mode Two is re-entered via a reset transition, at time 3000 msec, the action triggered by T2 before, at time 1850 msec, gets discarded.

    \n\"Illustration\n

    The above plot illustrates the relation between global time in the environment and the localized time for each timer in the model.\nSince the top-level reactor TimingExample is not enclosed by any mode, its time always corresponds to the global time.\nMode One is the initial mode and hence progresses in sync with TimingExample for the first second.\nDuring inactivity of mode One the timer is suspended and does not advance in time.\nAt 2000 msec it continues relative to this time.\nT2 only starts advancing when the mode becomes active at 1000 msec.\nThe reentry via reset at 3000 msec causes the local time to be reset to zero.

    \n

    Startup and Shutdown

    \n

    A challenge for modal execution is the handling startup and shutdown behavior.\nThese are commonly used for managing memory for state variables, handling connections to sensors or actuators, or starting/joining external threads.\nIf reactions to these triggers are located inside modes they are subject to a special execution regime.

    \n

    First, startup reactions are invoked at most once at the first activation of a mode.\nSecond, shutdown reactions are executed when the reactor shuts down, irrespective of mode activity, but only if the enclosing modes have been activated at least once.\nHence, every startup has a corresponding shutdown.\nThird, as mentioned before, the new reset trigger for reactions can be used, if a startup behavior should be re-executed if a mode is entered with a reset transition.

    \n

    Note that this may have unexpected implications:

    \n
      \n
    • Startup behavior inside modes may occur during execution and not only at program start.
    • \n
    • Multiple shutdown reactions may be executed, bypassing mutual exclusion of modes.
    • \n
    • Reactors that are designed without consideration of modes and use only startup (not reset) to trigger an execution chain, may not work in modes and cease to function if re-entered with a reset.
    • \n
    \n
    ","headings":[{"value":"Syntax","depth":2},{"value":"Execution Semantics","depth":2},{"value":"Local Time","depth":3},{"value":"Startup and Shutdown","depth":3}],"frontmatter":{"permalink":"/docs/handbook/modal-models","title":"Modal Reactors","oneline":"Modal Reactors","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","permalink":"/docs/handbook/superdense-time"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Deadlines","oneline":"Deadlines in Lingua Franca.","permalink":"/docs/handbook/deadlines"}}}},"pageContext":{"id":"1-modal-reactors","slug":"/docs/handbook/modal-models","repoPath":"/packages/documentation/copy/en/topics/Modal Models.md","previousID":"1986cb1b-8feb-57bc-8088-50548df2f061","nextID":"cb17ff3f-7a86-5f3b-95f7-e7d1f4e920c0","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/modal-models","result":{"data":{"markdownRemark":{"id":"3ab36bba-9f0e-54a5-bfd0-55f5e5259ca5","excerpt":"Modal Reactors are currently not supported in $target-language$. $page-showing-target$ The basic idea of modal reactors is to partition a reactor into disjoint…","html":"
    \n

    Modal Reactors are currently not supported in $target-language$.

    \n
    \n
    \n

    $page-showing-target$

    \n

    The basic idea of modal reactors is to partition a reactor into disjoint subsets of reactions (or other components) that are associated with mutually exclusive modes.\nIn a modal reactor, only a single mode can be active at a particular logical time instant, meaning that activity in other modes is automatically suspended.\nTransitioning between modes switches the reactor’s behavior and provides control over continuing or resetting the previous history of the entered mode.

    \n

    Syntax

    \n

    Modes can be defined in any reactor, except federated ones.\nEach mode requires a unique (per reactor) name and can declare contents that are local to this mode.\nThere must be exactly one mode marked as $initial$.

    \n
    reactor TwoModes {\n  ...\n  initial mode One {\n    ...\n  }\n  mode Two {\n    ...\n  }\n}\n
    \n

    A mode can contain local state variables, timers, actions, reactions, reactor instantiations, and connections.\nWhile modes cannot be nested in modes directly, hierarchical composition is possible through the instantiation of modal reactors.\nThe main exception in allowed contents in modes are port declarations, as these are only possible on reactor level.\nYet, modes share the scope with their reactor and, hence, can access ports, state variables, and parameters of the reactor.\nOnly the contents of other modes are excluded.\nA modal reactor can still have reactions, reactor instantiations, etc., that are not located in modes and will consequently be executed independently from mode activity.

    \n

    Mode transitions are declared within reactions.\nIf a reactor has modes, reactions inside modes are allowed to list them as effects.\nSuch an effect enables the use of the target language API to set the next mode.

    \n
    reaction(trig) -> Two {=\n  if (trig->value) {\n    lf_set_mode(Two)\n  }\n=}\n
    \n
    reaction(trig) -> Two {=\n  if trig.value:\n    Two.set()\n=}\n
    \n

    You can also specify the type of the transition by adding the modifier reset(<mode>) or history(<mode>) in the effects of the\nreaction signature (i.e., after the ->). For example, a history transition to the state Two is enabled by listing history(Two)\namong the effects of the reaction. The reset variant is implicitly assumed when the mode is listed without modifier.

    \n

    Execution Semantics

    \n

    The basic effect of modes is that only parts that are contained in the currently active mode (or outside any mode) are executed at any point in time.\nThis also holds for parts that are nested in multiple ancestor modes due to hierarchy; consequently, all those ancestors must be active in order to execute.\nReactions in inactive modes are simply not executed, while all components that model timing behavior, namely timers, scheduled actions, and delayed connections, are subject to a concept of local time.\nThat means while a mode is inactive, the progress of time is suspended locally.\nHow the timing components behave when a mode becomes active depends on the transition type.\nA mode can be reset upon entry, returning it to its initial state.\nAlternatively, it may be entered preserving its history, which only has an actual effect if the mode was active before.\nIn the latter case all timing components will continue their delays or period as if no time had passed during inactivity of the mode.\nThe following section will provide a more detailed explanation of this effect.

    \n

    Upon reactor startup, the initial mode of each modal reactor is active, others are inactive.\nIf at a tag (t, m), all reactions of this reactor and all its contents have finished executing, and a new mode was set in a reaction, the current mode will be deactivated and the new one will be activated for future execution.\nThis means no reaction of the newly active mode will execute at tag (t, m); the earliest possible reaction in the new mode occurs one microstep later, at (t, m+1).\nHence, if the newly active mode has for example a timer that will elapse with an offset of zero, it will trigger at (t, m+1).\nIn case the mode itself does not require an immediate execution in the next microstep, it depends on future events, just as in the normal behavior of LF.\nHence, modes in the same reactor are always mutually exclusive w.r.t. superdense time.

    \n

    A transition is triggered if a new mode is set in a reaction body.\nAs with setting output ports in reaction, a new mode can be set multiple times in the same or different reactions.\nIn the end, the fixed ordering of reactions determines the last effective value that will be used.\nThe new mode does not have to be a different one; it is possible for a mode to reset itself via a reset transition.

    \n

    In case a mode is entered with the reset behavior:

    \n
      \n
    • all contained modal reactors are reset to their initial mode (recursively),
    • \n
    • all contained timers are reset and start again awaiting their initial offset,
    • \n
    • all contained state variables that are marked for automatic reset are reset to their initial value,
    • \n
    • any contained reactions triggered by reset are executed, and
    • \n
    • all events (actions, timers, delayed connections) that were previously scheduled from within this mode are discarded.
    • \n
    \n

    Note that contained refers to all contents defined locally in the mode and in local reactor instances (recursively) that are not otherwise enclosed in modes of lower levels.

    \n

    Whenever a mode is entered with a reset transition, the subsequent timing behavior is as if the mode was never executed before.\nIf there are state variables that need to be reset or reinitialized, then this can be done in a reaction triggered by reset or by marking the state variable for automatic reset (e.g.,\nreset state x:int(0)\nFIXME\nreset state x(0)\nFIXME\nFIXME\n).\nState variables are not reset automatically to their initial conditions because it is idiomatic for reactors to allocate resources or initialize subsystems (e.g., allocate memory or sockets, register an interrupt, or start a server) in reactions triggered by the startup, and to store references to these resources in state variables.\nIf these were to be automatically reset, those references would be lost.

    \n

    On the other hand, if a mode has been active prior and is then re-entered via a history transition, no reset is performed.\nEvents originating from timers, scheduled actions, and delayed connections are adjusted to reflect a remaining delay equal to the remaining delay recorded at the instant the mode was previously deactivated.\nAs a consequence, a mode has a notion of local time that elapses only when the mode is active.

    \n

    Local Time

    \n

    From the perspective of timers and actions, time is suspended when a mode is inactive.\nThis also applies to indirectly nested reactors within modes and connections with logical delays, if their source lies within a mode.

    \n\"Illustration\n

    The above LF model illustrates the different characteristics of local time affecting timers and actions in the presence of the two transition types.

    \n

    It consists of two modes One (the initial mode) and Two, both in the Modal reactor.\nThe next input toggles between these modes and is controlled by a reaction at the top level that is triggered by the timer T.\nAfter one second, a mode switch is triggered periodically with a period one second.\nEach mode has a timer T1/T2 that triggers a reaction after an initial offset of 100 msec and then periodically after 750 msec.\nThis reaction then schedules a logical action with a delay of 500 msec (the actual target code does not add an additional delay over the minimum specified).\nThis action triggers the second reaction, which writes to the output out.\nThe main difference between the modes is that One is entered via a history transition, continuing its behavior, while Two is reset.\n(History behavior is indicated by an “H” on the transition edge because it enters into the entire history of the mode.)

    \n\"Illustration\n

    Above is the execution trace of the first 4 seconds of this program.\nBelow the timeline is the currently active mode and above the timeline are the model elements that are executed at certain points in time, together with indicating triggering and their relation through time.\nFor example, at 100 msec, the initial offset of timer T1 elapses, which leads to the scheduling of the logical action in this mode.\nThe action triggers the reaction 500 msec later, at 600 msec, and thus causes an output.\nThe timing diagram illustrates the different handling of time between history transitions and reset transitions.\nSpecifically, when mode One is re-entered via a history transition, at time 2000 msec, the action triggered by T1 before, at time 850 msec, resumes.\nIn contrast, when mode Two is re-entered via a reset transition, at time 3000 msec, the action triggered by T2 before, at time 1850 msec, gets discarded.

    \n\"Illustration\n

    The above plot illustrates the relation between global time in the environment and the localized time for each timer in the model.\nSince the top-level reactor TimingExample is not enclosed by any mode, its time always corresponds to the global time.\nMode One is the initial mode and hence progresses in sync with TimingExample for the first second.\nDuring inactivity of mode One the timer is suspended and does not advance in time.\nAt 2000 msec it continues relative to this time.\nT2 only starts advancing when the mode becomes active at 1000 msec.\nThe reentry via reset at 3000 msec causes the local time to be reset to zero.

    \n

    Startup and Shutdown

    \n

    A challenge for modal execution is the handling startup and shutdown behavior.\nThese are commonly used for managing memory for state variables, handling connections to sensors or actuators, or starting/joining external threads.\nIf reactions to these triggers are located inside modes they are subject to a special execution regime.

    \n

    First, startup reactions are invoked at most once at the first activation of a mode.\nSecond, shutdown reactions are executed when the reactor shuts down, irrespective of mode activity, but only if the enclosing modes have been activated at least once.\nHence, every startup has a corresponding shutdown.\nThird, as mentioned before, the new reset trigger for reactions can be used, if a startup behavior should be re-executed if a mode is entered with a reset transition.

    \n

    Note that this may have unexpected implications:

    \n
      \n
    • Startup behavior inside modes may occur during execution and not only at program start.
    • \n
    • Multiple shutdown reactions may be executed, bypassing mutual exclusion of modes.
    • \n
    • Reactors that are designed without consideration of modes and use only startup (not reset) to trigger an execution chain, may not work in modes and cease to function if re-entered with a reset.
    • \n
    \n
    ","headings":[{"value":"Syntax","depth":2},{"value":"Execution Semantics","depth":2},{"value":"Local Time","depth":3},{"value":"Startup and Shutdown","depth":3}],"frontmatter":{"permalink":"/docs/handbook/modal-models","title":"Modal Reactors","oneline":"Modal Reactors","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","permalink":"/docs/handbook/superdense-time"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Deadlines","oneline":"Deadlines in Lingua Franca.","permalink":"/docs/handbook/deadlines"}}}},"pageContext":{"id":"1-modal-reactors","slug":"/docs/handbook/modal-models","repoPath":"/packages/documentation/copy/en/topics/Modal Models.md","previousID":"1986cb1b-8feb-57bc-8088-50548df2f061","nextID":"cb17ff3f-7a86-5f3b-95f7-e7d1f4e920c0","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/multiports-and-banks/page-data.json b/page-data/docs/handbook/multiports-and-banks/page-data.json index ab836d45f..9a8b408bc 100644 --- a/page-data/docs/handbook/multiports-and-banks/page-data.json +++ b/page-data/docs/handbook/multiports-and-banks/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/multiports-and-banks","result":{"data":{"markdownRemark":{"id":"9347e103-d59f-5190-ba87-89f54110e437","excerpt":"$page-showing-target$ Lingua Franca provides a compact syntax for ports that can send or receive over multiple channels and another syntax for multiple…","html":"

    $page-showing-target$

    \n

    Lingua Franca provides a compact syntax for ports that can send or receive over multiple channels and another syntax for multiple instances of a reactor class. These are respectively called multiports and banks of reactors.

    \n

    Multiports

    \n

    To declare an input or output port to be a multiport, use the following syntax:

    \n
    \n
      input[<width>] <name>:<type>;\n  output[<width>] <name>:<type>;\n
    \n
    \n
    \n
      input[<width>] <name>\n  output[<width>] <name>\n
    \n
    \n

    where <width> is a positive integer. This can be given either as an integer literal or a parameter name. The width can also be given by target code enclosed in {=...=}. Consider the following example:

    \n

    $start(Multiport)$

    \n
    target C;\nreactor Source {\n  output[4] out:int;\n  reaction(startup) -> out {=\n    for(int i = 0; i < out_width; i++) {\n      lf_set(out[i], i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] in:int;\n  reaction(in) {=\n    int sum = 0;\n    for (int i = 0; i < in_width; i++) {\n      if (in[i]->is_present) sum += in[i]->value;\n    }\n    printf("Sum of received: %d.\\n", sum);\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.in;\n}\n
    \n
    target Cpp;\nreactor Source {\n  output[4] out:int;\n  reaction(startup) -> out {=\n    for(auto i = 0ul; i < out.size(); i++) {\n      out[i].set(i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] in:int;\n  reaction(in) {=\n    int sum = 0;\n    for (auto i = 0ul; i < in.size(); i++) {\n      if (in[i].is_present()){\n        sum += *in[i].get();\n      }\n    }\n    std::cout << "Sum of received: " << sum << std::endl;\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.in;\n}\n
    \n
    target Python;\nreactor Source {\n  output[4] out;\n  reaction(startup) -> out {=\n    for i, port in enumerate(out):\n      port.set(i)\n  =}\n}\nreactor Destination {\n  input[4] inp;\n  reaction(inp) {=\n    sum = 0\n    for port in inp:\n      if port.is_present: sum += port.value\n    print(f"Sum of received: {sum}.")\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.inp;\n}\n
    \n
    target TypeScript\nreactor Source {\n  output[4] out:number\n  reaction(startup) -> out {=\n    for (let i = 0 ; i < out.length; i++) {\n      out[i] = i\n    }\n  =}\n}\nreactor Destination {\n  input[4] inp:number\n  reaction(inp) {=\n    let sum = 0\n    for (let i = 0 ; i < inp.length; i++) {\n      const val = inp[i]\n      if (val) sum += val\n    }\n    console.log(`Sum of received: ${sum}`)\n  =}\n}\nmain reactor {\n  a = new Source()\n  b = new Destination()\n  a.out -> b.inp\n}\n
    \n
    target Rust;\nreactor Source {\n  output[4] out:usize;\n  reaction(startup) -> out {=\n    for (i, o) in out.into_iter().enumerate() {\n      ctx.set(o, i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] inp:usize;\n  reaction(inp) {=\n    let mut sum = 0;\n    for i in inp {\n      if let Some(v) = ctx.get(&i) {\n        sum += v;\n      }\n    }\n    println!("Sum of received: {}.", sum);\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.inp;\n}\n
    \n

    $end(Multiport)$

    \n\"Lingua\n

    Executing this program will yield:

    \n
    Sum of received: 6.
    \n

    The Source reactor has a four-way multiport output and the Destination reactor has a four-way multiport input. These channels are connected all at once on one line, the second line from the last. Notice that the generated diagram shows multiports with hollow triangles. Whether it shows the widths is controlled by an option in the diagram generator.

    \n

    The Source reactor specifies out as an effect of its reaction using the syntax -> out. This brings into scope of the reaction body a way to access the width of the port and a way to write to each channel of the port.

    \n

    NOTE: In Destination, the reaction is triggered by in, not by some individual channel of the multiport input. Hence, it is important when using multiport inputs to test for presence of the input on each channel, as done above with the syntax:

    \n
        if (in[i]->is_present) ...\n
    \n
        if (in[i]->is_present()) ...\n
    \n
        if port.is_present: ...\n
    \n
        if (val) ...\n
    \n
        if let Some(v) = ctx.get(&i) ...\n
    \n

    An event on any one of the channels is sufficient to trigger the reaction.

    \n
    \n

    In the Python target, multiports can be iterated on in a for loop (e.g., for p in out) or enumerated (e.g., for i, p in enumerate(out)) and the length of the multiport can be obtained by using the len() (e.g., len(out)) expression.

    \n
    \n
    \n

    Sparse Inputs

    \n

    Sometimes, a program needs a wide multiport input, but when reactions are triggered by this input, few of the channels are present.\nIn this case, it can be inefficient to iterate over all the channels to determine which are present.\nIf you know that a multiport input will be sparse in this way, then you can provide a hint to the compiler and use a more efficient iterator to access the port. For example:

    \n

    $start(Sparse)$

    \n
    target C;\nreactor Sparse {\n  @sparse\n  input[100] in:int;\n  reaction(in) {=\n    // Create an iterator over the input channels.\n    struct lf_multiport_iterator_t i = lf_multiport_iterator(in);\n    // Get the least index of a channel with present inputs.\n    int channel = lf_multiport_next(&i);\n    // Iterate until no more channels have present inputs.\n    while(channel >= 0) {\n      printf("Received %d on channel %d\\n", in[channel]->value, channel);\n      // Get the next channel with a present input.\n      channel = lf_multiport_next(&i);\n    }\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/py/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/Sparse.lf\n
    \n

    $end(Sparse)$

    \n

    Notice the @sparse annotation on the input declaration.\nThis provides a hint to the compiler to optimize for sparse inputs.\nThen, instead of iterating over all input channels, this code uses the built-in function lf_multiport_iterator() to construct an iterator. The function lf_multiport_next() returns the first (and later, the next) channel index that is present. It returns -1 when no more channels have present inputs.

    \n

    The multiport iterator can be used for any input multiport, even if it is not marked sparse.\nBut if it is not marked sparse, then the lf_multiport_next() function will not optimize for sparse inputs and will simply iterate over the channels until it finds one that is present.

    \n
    \n

    Parameterized Widths

    \n

    The width of a port may be given by a parameter. For example, the above Source reactor can be rewritten

    \n
    reactor Source(width:int = 4) {\n  output[width] out:int;\n  reaction(startup) -> out {=\n    ...\n  =}\n}\n
    \n
    \n

    Parameters to the main reactor can be overwritten on the command line interface when running the generated program. As a consequence, the scale of the application can be determined at run time rather than at compile time.

    \n
    \n

    Connecting Reactors with Different Widths

    \n

    Assume that the Source and Destination reactors above both use a parameter width to specify the width of their ports. Then the following connection is valid:

    \n
    main reactor {\n  a1 = new Source(width = 3);\n  a2 = new Source(width = 2);\n  b = new Destination(width = 5);\n  a1.out, a2.out -> b.in;\n}\n
    \n

    The first three ports of b will received input from a1, and the last two ports will receive input from a2. Parallel composition can appear on either side of a connection. For example:

    \n
      a1.out, a2.out -> b1.out, b2.out, b3.out;\n
    \n

    If the total width on the left does not match the total width on the right, then a warning is issued. If the left side is wider than the right, then output data will be discarded. If the right side is wider than the left, then input channels will be absent.

    \n

    Any given port can appear only once on the right side of the -> connection operator, so all connections to a multiport destination must be made in one single connection statement.

    \n

    Banks of Reactors

    \n

    Using a similar notation, it is possible to create a bank of reactors. For example, we can create a bank of four instances of Source and four instances of Destination and connect them as follows:

    \n
    main reactor {\n  a = new[4] Source();\n  b = new[4] Destination();\n  a.out -> b.in;\n}\n
    \n\"Lingua\n

    If the Source and Destination reactors have multiport inputs and outputs, as in the examples above, then a warning will be issued if the total width on the left does not match the total width on the right. For example, the following is balanced:

    \n
    main reactor {\n  a = new[3] Source(width = 4);\n  b = new[4] Destination(width = 3);\n  a.out -> b.in;\n}\n
    \n

    There will be three instances of Source, each with an output of width four, and four instances of Destination, each with an input of width 3, for a total of 12 connections.

    \n

    To distinguish the instances in a bank of reactors, the reactor can define a parameter called bank_index with any type that can be assigned a non-negative integer value (for example, int, size_t, or uint32_t). If such a parameter is defined for the reactor, then when the reactor is instantiated in a bank, each instance will be assigned a number between 0 and n-1, where n is the number of reactor instances in the bank. For example, the following source reactor increments the output it produces by the value of bank_index on each reaction to the timer:

    \n

    $start(MultiportSource)$

    \n
    target C\nreactor MultiportSource(bank_index: int = 0) {\n  timer t(0, 200 msec)\n  output out: int\n  state s: int = 0\n  reaction(t) -> out {=\n    lf_set(out, self->s);\n    self->s += self->bank_index;\n  =}\n}\n
    \n
    target Cpp\nreactor MultiportSource(bank_index: int(0)) {\n  timer t(0, 200 ms)\n  output out: int\n  state s: int(0)\n  reaction(t) -> out {=\n    out.set(s);\n    s += bank_index;\n  =}\n}\n
    \n
    target Python\nreactor MultiportSource(bank_index=0) {\n  timer t(0, 200 msec)\n  output out\n  state s = 0\n  reaction(t) -> out {=\n    out.set(self.s)\n    self.s += self.bank_index\n  =}\n}\n
    \n
    target TypeScript\nreactor MultiportSource {\n  timer t(0, 200 msec)\n  output out: number\n  state s: number = 0\n  reaction(t) -> out {=\n    out = s\n    s += this.getBankIndex()\n  =}\n}\n
    \n
    target Rust\nreactor MultiportSource(bank_index: u32 = 0) {\n  state bank_index = bank_index\n  timer t(0, 200 msec)\n  output out: u32\n  state s: u32 = 0\n  reaction(t) -> out {=\n    ctx.set(out, self.s);\n    self.s += self.bank_index;\n  =}\n}\n
    \n

    $end(MultiportSource)$

    \n

    The width of a bank may also be given by a parameter, as in

    \n
    main reactor(\n  source_bank_width:int = 3,\n  destination_bank_width:int = 4\n) {\n  a = new[source_bank_width] Source(width = 4);\n  b = new[destination_bank_width] Destination(width = 3);\n  a.out -> b.in;\n}\n
    \n
    \n

    Initializing Bank Members from a Table

    \n

    It is often convenient to initialize parameters of bank members from a table.\nHere is an example:

    \n

    $start(BankIndex)$

    \n
    target C;\npreamble {=\n  int table[] = {4, 3, 2, 1};\n=}\nreactor A(bank_index:int = 0, value:int = 0) {\n  reaction (startup) {=\n    printf("bank_index: %d, value: %d\\n", self->bank_index, self->value);\n  =}\n}\nmain reactor {\n  a = new[4] A(value = {= table[bank_index] =});\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/BankIndex.lf\n
    \n
    target Python;\npreamble {=\n  table = [4, 3, 2, 1]\n=}\nreactor A(bank_index = 0, value = 0) {\n  reaction (startup) {=\n    print("bank_index: {:d}, value: {:d}".format(self.bank_index, self.value))\n  =}\n}\nmain reactor {\n  a = new[4] A(value = {= table[bank_index] =})\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/BankIndex.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/BankIndex.lf\n
    \n

    $end(BankIndex)$

    \n

    The global table defined in the $preamble$ is used to initialize the value parameter of each bank member. The result of running this is something like:

    \n
    bank_index: 0, value: 4\nbank_index: 1, value: 3\nbank_index: 2, value: 2\nbank_index: 3, value: 1
    \n
    \n

    Contained Banks

    \n

    Banks of reactors can be nested. For example, note the following program:

    \n

    $start(ChildBank)$

    \n
    target C;\nreactor Child (\n  bank_index:int = 0\n) {\n  reaction(startup) {=\n    printf("My bank index: %d.\\n", self->bank_index);\n  =}\n}\nreactor Parent (\n  bank_index:int = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target Cpp;\nreactor Child (\n  bank_index:int = 0\n) {\n  reaction(startup) {=\n    std::cout << "My bank index:" << bank_index << std::endl;\n  =}\n}\nreactor Parent (\n  bank_index:int = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target Python;\nreactor Child (\n  bank_index = 0\n) {\n  reaction(startup) {=\n    print(f"My bank index: {self.bank_index}.")\n  =}\n}\nreactor Parent (\n  bank_index = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target TypeScript\nreactor Child {\n  reaction(startup) {=\n    console.log(`My bank index ${this.getBankIndex()}`)\n  =}\n}\nreactor Parent {\n  c = new[2] Child()\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Rust;\nreactor Child (\n  bank_index:usize = 0\n) {\n  state bank_index = bank_index;\n  reaction(startup) {=\n    println!("My bank index: {}.", self.bank_index);\n  =}\n}\nreactor Parent (\n  bank_index:usize = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n

    $end(ChildBank)$

    \n\"Lingua\n

    In this program, the Parent reactor contains a bank of Child reactor instances\nwith a width of 2. In the main reactor, a bank of Parent reactors is\ninstantiated with a width of 2, therefore, creating 4 Child instances in the program in total.\nThe output of this program will be:

    \n
    My bank index: 0.\nMy bank index: 1.\nMy bank index: 0.\nMy bank index: 1.
    \n

    The order of these outputs will be nondeterministic if the execution is multithreaded (which it will be by default) because there is no dependence between the reactions, and, hence, they can execute in parallel.

    \n

    The bank index of a container (parent) reactor can be passed down to\ncontained (child) reactors. For example, note the following program:

    \n

    $start(ChildParentBank)$

    \n
    target C\nreactor Child(bank_index: int = 0, parent_bank_index: int = 0) {\n  reaction(startup) {=\n    printf(\n        "My bank index: %d. My parent's bank index: %d.\\n",\n        self->bank_index, self->parent_bank_index\n    );\n  =}\n}\nreactor Parent(bank_index: int = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Cpp\nreactor Child(bank_index: int(0), parent_bank_index: int(0)) {\n  reaction(startup) {=\n    std::cout <<"My bank index: " << bank_index << " My parent's bank index: " << parent_bank_index << std::endl;\n  =}\n}\nreactor Parent(bank_index: int(0)) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Python\nreactor Child(bank_index=0, parent_bank_index=0) {\n  reaction(startup) {=\n    print(\n        f"My bank index: {self.bank_index}. "\n        f"My parent's bank index: {self.parent_bank_index}."\n    )\n  =}\n}\nreactor Parent(bank_index=0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target TypeScript\nreactor Child(parentBankIndex: number = 0) {\n  reaction(startup) {=\n    console.log(`My bank index: ${this.getBankIndex()} My parent's bank index: ${parentBankIndex}`)\n  =}\n}\nreactor Parent {\n  c = new[2] Child(parentBankIndex = {= this.getBankIndex() =})\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Rust\nreactor Child(bank_index: usize = 0, parent_bank_index: usize = 0) {\n  state bank_index = bank_index\n  state parent_bank_index = parent_bank_index\n  reaction(startup) {=\n    println!(\n        "My bank index: {}. My parent's bank index: {}.",\n        self.bank_index,\n        self.parent_bank_index,\n    );\n  =}\n}\nreactor Parent(bank_index: usize = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n

    $end(ChildParentBank)$

    \n

    In this example, the bank index of the Parent reactor is passed to the\nparent_bank_index parameter of the Child reactor instances.\nThe output from this program will be:

    \n
    My bank index: 1. My parent's bank index: 1.\nMy bank index: 0. My parent's bank index: 0.\nMy bank index: 0. My parent's bank index: 1.\nMy bank index: 1. My parent's bank index: 0.
    \n

    Again, note that the order of these outputs is nondeterministic.

    \n

    Finally, members of contained banks of reactors can be individually addressed in\nthe body of reactions of the parent reactor if their input/output port appears\nin the reaction signature. For example, note the following program:

    \n

    $start(ChildParentBank2)$

    \n
    target C\nreactor Child(bank_index: int = 0, parent_bank_index: int = 0) {\n  output out: int\n  reaction(startup) -> out {=\n    lf_set(out, self->parent_bank_index * 2 + self->bank_index);\n  =}\n}\nreactor Parent(bank_index: int = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for (int i=0; i < c_width; i++) {\n        printf("Received %d from child %d.\\n", c[i].out->value, i);\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Cpp\nreactor Child(bank_index: int(0), parent_bank_index: int(0)) {\n  output out: int\n  reaction(startup) -> out {=\n    out.set(parent_bank_index * 2 + bank_index);\n  =}\n}\nreactor Parent(bank_index: int(0)) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for (auto i = 0ul; i < c.size(); i++) {\n        std::cout << "Received " << *c[i].out.get() <<" from child " << i << std::endl;\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Python\nreactor Child(bank_index=0, parent_bank_index=0) {\n  output out\n  reaction(startup) -> out {=\n    out.set(self.parent_bank_index * 2 + self.bank_index)\n  =}\n}\nreactor Parent(bank_index=0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for i, child in enumerate(c):\n        print(f"Received {child.out.value} from child {i}.")\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target TypeScript\nreactor Child(parentBankIndex: number = 0) {\n  output out: number\n  reaction(startup) -> out {=\n    out = parentBankIndex * 2 + this.getBankIndex()\n  =}\n}\nreactor Parent {\n  c = new[2] Child(parentBankIndex = {= this.getBankIndex() =})\n  reaction(c.out) {=\n    for (let i = 0; i < c.length; i++) {\n        console.log(`Received ${c[i].out} from child ${i}`)\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/ChildParentBank2.lf\n
    \n

    $end(ChildParentBank2)$

    \n\"Lingua\n

    Running this program will give something like the following:

    \n
    Received 0 from child 0.\nReceived 1 from child 1.\nReceived 2 from child 0.\nReceived 3 from child 1.
    \n
    \n

    Note the usage of c_width, which holds the width of the c bank of reactors.

    \n
    \n
    \n

    Note that len(c) can be used to get the width of the bank, and for p in c or for (i, p) in enumerate(c) can be used to iterate over the bank members.

    \n
    \n
    \n

    Note that c.size() can be used to get the width of the bank c.

    \n
    \n
    \n

    Note that that bank instance c in TypeScript is an array, so c.length is the width of the bank, and the bank members are referenced by indexing the array, as in c[i].

    \n
    \n
    \n

    FIXME: How to get the width of the bank in target code?

    \n
    \n

    Combining Banks and Multiports

    \n

    Banks of reactors may be combined with multiports, as in the following example:

    \n

    $start(MultiportToBank)$

    \n
    target C\nreactor Source {\n  output[3] out: int\n  reaction(startup) -> out {=\n    for(int i = 0; i < out_width; i++) {\n      lf_set(out[i], i);\n    }\n  =}\n}\nreactor Destination(bank_index: int = 0) {\n  input in: int\n  reaction(in) {=\n    printf("Destination %d received %d.\\n", self->bank_index, in->value);\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.in\n}\n
    \n
    target Cpp\nreactor Source {\n  output[3] out: int\n  reaction(startup) -> out {=\n    for(int i = 0; i < out.size(); i++) {\n      out[i].set(i);\n    }\n  =}\n}\nreactor Destination(bank_index: int(0)) {\n  input in: int\n  reaction(in) {=\n    std::cout << "Destination " << bank_index << " received " << *in.get() << std::endl;\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.in\n}\n
    \n
    target Python\nreactor Source {\n  output[3] out\n  reaction(startup) -> out {=\n    for i, port in enumerate(out):\n      port.set(i)\n  =}\n}\nreactor Destination(bank_index=0) {\n  input inp\n  reaction(inp) {=\n    print(f"Destination {self.bank_index} received {inp.value}.")\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n
    target TypeScript\nreactor Source {\n  output[3] out: number\n  reaction(startup) -> out {=\n     for (let i = 0 ; i < out.length; i++) {\n        out[i] = i\n    }\n  =}\n}\nreactor Destination {\n  input inp: number\n  reaction(inp) {=\n    console.log(`Destination ${this.getBankIndex()} received ${inp}`)\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n
    target Rust\nreactor Source {\n  output[3] out: usize\n  reaction(startup) -> out {=\n    for (i, o) in out.into_iter().enumerate() {\n      ctx.set(o, i);\n    }\n  =}\n}\nreactor Destination(bank_index: usize = 0) {\n  state bank_index = bank_index\n  input inp: usize\n  reaction(inp) {=\n    println!(\n        "Destination {} received {}.",\n        self.bank_index,\n        ctx.get(inp).unwrap(),\n    );\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n

    $end(MultiportToBank)$

    \n\"Lingua\n

    The three outputs from the Source instance a will be sent, respectively, to each of three instances of Destination, b[0], b[1], and b[2]. The result of the program will be something like the following:

    \n
    Destination 0 received 0.\nDestination 1 received 1.\nDestination 2 received 2.
    \n

    Again, the order is nondeterministic in a multithreaded context.

    \n

    The reactors in a bank may themselves have multiports. In all cases, the number of ports on the left of a connection must match the number on the right, unless the ones on the left are iterated, as explained next.

    \n

    Broadcast Connections

    \n

    Occasionally, you will want to have fewer ports on the left of a connection and have their outputs used repeatedly to broadcast to the ports on the right. In the following example, the outputs from an ordinary port are broadcast to the inputs of all instances of a bank of reactors:

    \n
    reactor Source {\n  output out:int;\n  reaction(startup) -> out {=\n    ... write to out ...\n  =}\n}\nreactor Destination {\n  input in:int;\n  reaction(in) {=\n    ... read from in ...\n  =}\n}\nmain reactor ThreadedThreaded(width:int(4)) {\n  a = new Source();\n  d = new[width] Destination();\n  (a.out)+ -> d.in;\n}\n
    \n

    The syntax (a.out)+ means “repeat the output port a.out one or more times as needed to supply all the input ports of d.in.” The content inside the parentheses can be a comma-separated list of ports, the ports inside can be ordinary ports or multiports, and the reactors inside can be ordinary reactors or banks of reactors. In all cases, the number of ports inside the parentheses on the left must divide the number of ports on the right.

    \n

    Interleaved Connections

    \n
    \n

    Sometimes, we don’t want to broadcast messages to all reactors, but need more fine-grained control as to which reactor within a bank receives a message. If we have separate source and destination reactors, this can be done by combining multiports and banks as was shown in Combining Banks and Multiports. Setting a value on the index n of the output multiport, will result in a message to the n-th reactor instance within the destination bank. However, this pattern gets slightly more complicated, if we want to exchange addressable messages between instances of the same bank. This pattern is shown in the following example:

    \n

    $start(Interleaved)$

    \n
    target C\nreactor Node(num_nodes: size_t = 4, bank_index: int = 0) {\n  input[num_nodes] in: int\n  output[num_nodes] out: int\n  reaction(startup) -> out {=\n    lf_set(out[1], 42);\n    printf("Bank index %d sent 42 on channel 1.\\n", self->bank_index);\n  =}\n  reaction(in) {=\n    for (int i = 0; i < in_width; i++) {\n      if (in[i]->is_present) {\n        printf("Bank index %d received %d on channel %d.\\n",\n          self->bank_index, in[i]->value, i\n        );\n      }\n    }\n  =}\n}\nmain reactor(num_nodes: size_t = 4) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.in)\n}\n
    \n
    target Cpp\nreactor Node(num_nodes: size_t(4), bank_index: int(0)) {\n  input[num_nodes] in: int\n  output[num_nodes] out: int\n  reaction(startup) -> out {=\n    out[1].set(42);\n    std::cout << "Bank index " << bank_index << " sent 42 on channel 1." << std::endl;\n  =}\n  reaction(in) {=\n    for (auto i = 0ul; i < in.size(); i++) {\n      if (in[i].is_present()) {\n        std::cout << "Bank index " << bank_index\n          << " received " << *in[i].get() << " on channel" << std::endl;\n      }\n    }\n  =}\n}\nmain reactor(num_nodes: size_t(4)) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.in)\n}\n
    \n
    target Python\nreactor Node(num_nodes=4, bank_index=0) {\n  input[num_nodes] inp\n  output[num_nodes] out\n  reaction(startup) -> out {=\n    out[1].set(42)\n    print(f"Bank index {self.bank_index} sent 42 on channel 1.")\n  =}\n  reaction(inp) {=\n    for i, port in enumerate(inp):\n      if port.is_present:\n        print(\n          f"Bank index {self.bank_index} received {port.value} on channel {i}.",\n        )\n  =}\n}\nmain reactor(num_nodes=4) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.inp)\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/Interleaved.lf\ntarget TypeScript\nreactor Node(numNodes: number(4)) {\n    input[numNodes] inp: number\n    output[numNodes] out: number\n    reaction (startup) -> out {=\n        out[1] = 42\n        console.log(`Bank index ${this.getBankIndex()} sent 42 on channel 1.`)\n    =}\n    reaction (inp) {=\n        for (let i = 0; i < in.length; i++) {\n            if (in[i] !== undefined) {\n                console.log(`Bank index ${this.getBankIndex()} received ${in[i]} on channel ${i}`)\n            }\n        }\n    =}\n}\nmain reactor(numNodes: number(4)) {\n    nodes = new[numNodes] Node(numNodes=numNodes);\n    nodes.out -> interleaved(nodes.inp)\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Interleaved.lf\n
    \n

    $end(Interleaved)$

    \n\"Lingua\n

    In the above program, four instance of Node are created, and, at startup, each instance sends 42 to its second (index 1) output channel. The result is that the second bank member (bank_index 1) will receive the number 42 on each input channel of its multiport input. Running this program gives something like the following:

    \n
    Bank index 0 sent 42 on channel 1.\nBank index 1 sent 42 on channel 1.\nBank index 2 sent 42 on channel 1.\nBank index 3 sent 42 on channel 1.\nBank index 1 received 42 on channel 0.\nBank index 1 received 42 on channel 1.\nBank index 1 received 42 on channel 2.\nBank index 1 received 42 on channel 3.
    \n

    In bank index 1, the 0-th channel receives from bank_index 0, the 1-th channel from bank_index 1, etc. In effect, the choice of output channel specifies the destination reactor in the bank, and the input channel specifies the source reactor from which the input comes.

    \n

    This style of connection is accomplished using the new keyword $interleaved$ in the connection. Normally, a port reference such as nodes.out where nodes is a bank and out is a multiport, would list all the individual ports by first iterating over the banks and then, for each bank index, iterating over the ports. If we consider the tuple (b,p) to denote the index b within the bank and the index p within the multiport, then the following list is created: (0,0), (0,1), (0,2), (0,3), (1,0), (1,1), (1,2), (1,3), (2,0), (2,1), (2,2), (2,3), (3,0), (3,1), (3,2), (3,3). However, if we use $interleaved$(nodes.out) instead, the connection logic will iterate over the ports first and then the banks, creating the following list: (0,0), (1,0), (2,0), (3,0), (0,1), (1,1), (2,1), (3,1), (0,2), (1,2), (2,2), (3,2), (0,3), (1,3), (2,3), (3,3). By combining a normal port reference with a interleaved reference, we can construct a fully connected network. The figure below visualizes this how this pattern would look without banks or multiports:

    \n\n \n \"Lingua\n \n

    If we were to use a normal connection nodes.out -> nodes.in; instead of the $interleaved$ connection, then the following pattern would be created:

    \n\n \n \"Lingua\n \n

    Effectively, this connects each reactor instance to itself, which isn’t very useful.

    \n
    \n
    \n

    The $interleaved$ keyword is not supported by $target-language$.

    \n
    ","headings":[{"value":"Multiports","depth":2},{"value":"Sparse Inputs","depth":2},{"value":"Parameterized Widths","depth":2},{"value":"Connecting Reactors with Different Widths","depth":2},{"value":"Banks of Reactors","depth":2},{"value":"Initializing Bank Members from a Table","depth":2},{"value":"Contained Banks","depth":2},{"value":"Combining Banks and Multiports","depth":2},{"value":"Broadcast Connections","depth":2},{"value":"Interleaved Connections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/multiports-and-banks","title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Deadlines","oneline":"Deadlines in Lingua Franca.","permalink":"/docs/handbook/deadlines"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","permalink":"/docs/handbook/generics"}}}},"pageContext":{"id":"1-multiports-and-banks","slug":"/docs/handbook/multiports-and-banks","repoPath":"/packages/documentation/copy/en/topics/Multiports and Banks.md","previousID":"cb17ff3f-7a86-5f3b-95f7-e7d1f4e920c0","nextID":"2a0b9619-72b6-5ee7-8a38-83ff8d48005a","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/multiports-and-banks","result":{"data":{"markdownRemark":{"id":"9347e103-d59f-5190-ba87-89f54110e437","excerpt":"$page-showing-target$ Lingua Franca provides a compact syntax for ports that can send or receive over multiple channels and another syntax for multiple…","html":"

    $page-showing-target$

    \n

    Lingua Franca provides a compact syntax for ports that can send or receive over multiple channels and another syntax for multiple instances of a reactor class. These are respectively called multiports and banks of reactors.

    \n

    Multiports

    \n

    To declare an input or output port to be a multiport, use the following syntax:

    \n
    \n
      input[<width>] <name>:<type>;\n  output[<width>] <name>:<type>;\n
    \n
    \n
    \n
      input[<width>] <name>\n  output[<width>] <name>\n
    \n
    \n

    where <width> is a positive integer. This can be given either as an integer literal or a parameter name. The width can also be given by target code enclosed in {=...=}. Consider the following example:

    \n

    $start(Multiport)$

    \n
    target C;\nreactor Source {\n  output[4] out:int;\n  reaction(startup) -> out {=\n    for(int i = 0; i < out_width; i++) {\n      lf_set(out[i], i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] in:int;\n  reaction(in) {=\n    int sum = 0;\n    for (int i = 0; i < in_width; i++) {\n      if (in[i]->is_present) sum += in[i]->value;\n    }\n    printf("Sum of received: %d.\\n", sum);\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.in;\n}\n
    \n
    target Cpp;\nreactor Source {\n  output[4] out:int;\n  reaction(startup) -> out {=\n    for(auto i = 0ul; i < out.size(); i++) {\n      out[i].set(i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] in:int;\n  reaction(in) {=\n    int sum = 0;\n    for (auto i = 0ul; i < in.size(); i++) {\n      if (in[i].is_present()){\n        sum += *in[i].get();\n      }\n    }\n    std::cout << "Sum of received: " << sum << std::endl;\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.in;\n}\n
    \n
    target Python;\nreactor Source {\n  output[4] out;\n  reaction(startup) -> out {=\n    for i, port in enumerate(out):\n      port.set(i)\n  =}\n}\nreactor Destination {\n  input[4] inp;\n  reaction(inp) {=\n    sum = 0\n    for port in inp:\n      if port.is_present: sum += port.value\n    print(f"Sum of received: {sum}.")\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.inp;\n}\n
    \n
    target TypeScript\nreactor Source {\n  output[4] out:number\n  reaction(startup) -> out {=\n    for (let i = 0 ; i < out.length; i++) {\n      out[i] = i\n    }\n  =}\n}\nreactor Destination {\n  input[4] inp:number\n  reaction(inp) {=\n    let sum = 0\n    for (let i = 0 ; i < inp.length; i++) {\n      const val = inp[i]\n      if (val) sum += val\n    }\n    console.log(`Sum of received: ${sum}`)\n  =}\n}\nmain reactor {\n  a = new Source()\n  b = new Destination()\n  a.out -> b.inp\n}\n
    \n
    target Rust;\nreactor Source {\n  output[4] out:usize;\n  reaction(startup) -> out {=\n    for (i, o) in out.into_iter().enumerate() {\n      ctx.set(o, i);\n    }\n  =}\n}\nreactor Destination {\n  input[4] inp:usize;\n  reaction(inp) {=\n    let mut sum = 0;\n    for i in inp {\n      if let Some(v) = ctx.get(&i) {\n        sum += v;\n      }\n    }\n    println!("Sum of received: {}.", sum);\n  =}\n}\nmain reactor {\n  a = new Source();\n  b = new Destination();\n  a.out -> b.inp;\n}\n
    \n

    $end(Multiport)$

    \n\"Lingua\n

    Executing this program will yield:

    \n
    Sum of received: 6.
    \n

    The Source reactor has a four-way multiport output and the Destination reactor has a four-way multiport input. These channels are connected all at once on one line, the second line from the last. Notice that the generated diagram shows multiports with hollow triangles. Whether it shows the widths is controlled by an option in the diagram generator.

    \n

    The Source reactor specifies out as an effect of its reaction using the syntax -> out. This brings into scope of the reaction body a way to access the width of the port and a way to write to each channel of the port.

    \n

    NOTE: In Destination, the reaction is triggered by in, not by some individual channel of the multiport input. Hence, it is important when using multiport inputs to test for presence of the input on each channel, as done above with the syntax:

    \n
        if (in[i]->is_present) ...\n
    \n
        if (in[i]->is_present()) ...\n
    \n
        if port.is_present: ...\n
    \n
        if (val) ...\n
    \n
        if let Some(v) = ctx.get(&i) ...\n
    \n

    An event on any one of the channels is sufficient to trigger the reaction.

    \n
    \n

    In the Python target, multiports can be iterated on in a for loop (e.g., for p in out) or enumerated (e.g., for i, p in enumerate(out)) and the length of the multiport can be obtained by using the len() (e.g., len(out)) expression.

    \n
    \n
    \n

    Sparse Inputs

    \n

    Sometimes, a program needs a wide multiport input, but when reactions are triggered by this input, few of the channels are present.\nIn this case, it can be inefficient to iterate over all the channels to determine which are present.\nIf you know that a multiport input will be sparse in this way, then you can provide a hint to the compiler and use a more efficient iterator to access the port. For example:

    \n

    $start(Sparse)$

    \n
    target C;\nreactor Sparse {\n  @sparse\n  input[100] in:int;\n  reaction(in) {=\n    // Create an iterator over the input channels.\n    struct lf_multiport_iterator_t i = lf_multiport_iterator(in);\n    // Get the least index of a channel with present inputs.\n    int channel = lf_multiport_next(&i);\n    // Iterate until no more channels have present inputs.\n    while(channel >= 0) {\n      printf("Received %d on channel %d\\n", in[channel]->value, channel);\n      // Get the next channel with a present input.\n      channel = lf_multiport_next(&i);\n    }\n  =}\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/py/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/ts/src/Sparse.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/Sparse.lf\n
    \n

    $end(Sparse)$

    \n

    Notice the @sparse annotation on the input declaration.\nThis provides a hint to the compiler to optimize for sparse inputs.\nThen, instead of iterating over all input channels, this code uses the built-in function lf_multiport_iterator() to construct an iterator. The function lf_multiport_next() returns the first (and later, the next) channel index that is present. It returns -1 when no more channels have present inputs.

    \n

    The multiport iterator can be used for any input multiport, even if it is not marked sparse.\nBut if it is not marked sparse, then the lf_multiport_next() function will not optimize for sparse inputs and will simply iterate over the channels until it finds one that is present.

    \n
    \n

    Parameterized Widths

    \n

    The width of a port may be given by a parameter. For example, the above Source reactor can be rewritten

    \n
    reactor Source(width:int = 4) {\n  output[width] out:int;\n  reaction(startup) -> out {=\n    ...\n  =}\n}\n
    \n
    \n

    Parameters to the main reactor can be overwritten on the command line interface when running the generated program. As a consequence, the scale of the application can be determined at run time rather than at compile time.

    \n
    \n

    Connecting Reactors with Different Widths

    \n

    Assume that the Source and Destination reactors above both use a parameter width to specify the width of their ports. Then the following connection is valid:

    \n
    main reactor {\n  a1 = new Source(width = 3);\n  a2 = new Source(width = 2);\n  b = new Destination(width = 5);\n  a1.out, a2.out -> b.in;\n}\n
    \n

    The first three ports of b will received input from a1, and the last two ports will receive input from a2. Parallel composition can appear on either side of a connection. For example:

    \n
      a1.out, a2.out -> b1.out, b2.out, b3.out;\n
    \n

    If the total width on the left does not match the total width on the right, then a warning is issued. If the left side is wider than the right, then output data will be discarded. If the right side is wider than the left, then input channels will be absent.

    \n

    Any given port can appear only once on the right side of the -> connection operator, so all connections to a multiport destination must be made in one single connection statement.

    \n

    Banks of Reactors

    \n

    Using a similar notation, it is possible to create a bank of reactors. For example, we can create a bank of four instances of Source and four instances of Destination and connect them as follows:

    \n
    main reactor {\n  a = new[4] Source();\n  b = new[4] Destination();\n  a.out -> b.in;\n}\n
    \n\"Lingua\n

    If the Source and Destination reactors have multiport inputs and outputs, as in the examples above, then a warning will be issued if the total width on the left does not match the total width on the right. For example, the following is balanced:

    \n
    main reactor {\n  a = new[3] Source(width = 4);\n  b = new[4] Destination(width = 3);\n  a.out -> b.in;\n}\n
    \n

    There will be three instances of Source, each with an output of width four, and four instances of Destination, each with an input of width 3, for a total of 12 connections.

    \n

    To distinguish the instances in a bank of reactors, the reactor can define a parameter called bank_index with any type that can be assigned a non-negative integer value (for example, int, size_t, or uint32_t). If such a parameter is defined for the reactor, then when the reactor is instantiated in a bank, each instance will be assigned a number between 0 and n-1, where n is the number of reactor instances in the bank. For example, the following source reactor increments the output it produces by the value of bank_index on each reaction to the timer:

    \n

    $start(MultiportSource)$

    \n
    target C\nreactor MultiportSource(bank_index: int = 0) {\n  timer t(0, 200 msec)\n  output out: int\n  state s: int = 0\n  reaction(t) -> out {=\n    lf_set(out, self->s);\n    self->s += self->bank_index;\n  =}\n}\n
    \n
    target Cpp\nreactor MultiportSource(bank_index: int(0)) {\n  timer t(0, 200 ms)\n  output out: int\n  state s: int(0)\n  reaction(t) -> out {=\n    out.set(s);\n    s += bank_index;\n  =}\n}\n
    \n
    target Python\nreactor MultiportSource(bank_index=0) {\n  timer t(0, 200 msec)\n  output out\n  state s = 0\n  reaction(t) -> out {=\n    out.set(self.s)\n    self.s += self.bank_index\n  =}\n}\n
    \n
    target TypeScript\nreactor MultiportSource {\n  timer t(0, 200 msec)\n  output out: number\n  state s: number = 0\n  reaction(t) -> out {=\n    out = s\n    s += this.getBankIndex()\n  =}\n}\n
    \n
    target Rust\nreactor MultiportSource(bank_index: u32 = 0) {\n  state bank_index = bank_index\n  timer t(0, 200 msec)\n  output out: u32\n  state s: u32 = 0\n  reaction(t) -> out {=\n    ctx.set(out, self.s);\n    self.s += self.bank_index;\n  =}\n}\n
    \n

    $end(MultiportSource)$

    \n

    The width of a bank may also be given by a parameter, as in

    \n
    main reactor(\n  source_bank_width:int = 3,\n  destination_bank_width:int = 4\n) {\n  a = new[source_bank_width] Source(width = 4);\n  b = new[destination_bank_width] Destination(width = 3);\n  a.out -> b.in;\n}\n
    \n
    \n

    Initializing Bank Members from a Table

    \n

    It is often convenient to initialize parameters of bank members from a table.\nHere is an example:

    \n

    $start(BankIndex)$

    \n
    target C;\npreamble {=\n  int table[] = {4, 3, 2, 1};\n=}\nreactor A(bank_index:int = 0, value:int = 0) {\n  reaction (startup) {=\n    printf("bank_index: %d, value: %d\\n", self->bank_index, self->value);\n  =}\n}\nmain reactor {\n  a = new[4] A(value = {= table[bank_index] =});\n}\n
    \n
    WARNING: No source file found: ../code/cpp/src/BankIndex.lf\n
    \n
    target Python;\npreamble {=\n  table = [4, 3, 2, 1]\n=}\nreactor A(bank_index = 0, value = 0) {\n  reaction (startup) {=\n    print("bank_index: {:d}, value: {:d}".format(self.bank_index, self.value))\n  =}\n}\nmain reactor {\n  a = new[4] A(value = {= table[bank_index] =})\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/BankIndex.lf\n
    \n
    WARNING: No source file found: ../code/rs/src/BankIndex.lf\n
    \n

    $end(BankIndex)$

    \n

    The global table defined in the $preamble$ is used to initialize the value parameter of each bank member. The result of running this is something like:

    \n
    bank_index: 0, value: 4\nbank_index: 1, value: 3\nbank_index: 2, value: 2\nbank_index: 3, value: 1
    \n
    \n

    Contained Banks

    \n

    Banks of reactors can be nested. For example, note the following program:

    \n

    $start(ChildBank)$

    \n
    target C;\nreactor Child (\n  bank_index:int = 0\n) {\n  reaction(startup) {=\n    printf("My bank index: %d.\\n", self->bank_index);\n  =}\n}\nreactor Parent (\n  bank_index:int = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target Cpp;\nreactor Child (\n  bank_index:int = 0\n) {\n  reaction(startup) {=\n    std::cout << "My bank index:" << bank_index << std::endl;\n  =}\n}\nreactor Parent (\n  bank_index:int = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target Python;\nreactor Child (\n  bank_index = 0\n) {\n  reaction(startup) {=\n    print(f"My bank index: {self.bank_index}.")\n  =}\n}\nreactor Parent (\n  bank_index = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n
    target TypeScript\nreactor Child {\n  reaction(startup) {=\n    console.log(`My bank index ${this.getBankIndex()}`)\n  =}\n}\nreactor Parent {\n  c = new[2] Child()\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Rust;\nreactor Child (\n  bank_index:usize = 0\n) {\n  state bank_index = bank_index;\n  reaction(startup) {=\n    println!("My bank index: {}.", self.bank_index);\n  =}\n}\nreactor Parent (\n  bank_index:usize = 0\n) {\n  c = new[2] Child();\n}\nmain reactor {\n  p = new[2] Parent();\n}\n
    \n

    $end(ChildBank)$

    \n\"Lingua\n

    In this program, the Parent reactor contains a bank of Child reactor instances\nwith a width of 2. In the main reactor, a bank of Parent reactors is\ninstantiated with a width of 2, therefore, creating 4 Child instances in the program in total.\nThe output of this program will be:

    \n
    My bank index: 0.\nMy bank index: 1.\nMy bank index: 0.\nMy bank index: 1.
    \n

    The order of these outputs will be nondeterministic if the execution is multithreaded (which it will be by default) because there is no dependence between the reactions, and, hence, they can execute in parallel.

    \n

    The bank index of a container (parent) reactor can be passed down to\ncontained (child) reactors. For example, note the following program:

    \n

    $start(ChildParentBank)$

    \n
    target C\nreactor Child(bank_index: int = 0, parent_bank_index: int = 0) {\n  reaction(startup) {=\n    printf(\n        "My bank index: %d. My parent's bank index: %d.\\n",\n        self->bank_index, self->parent_bank_index\n    );\n  =}\n}\nreactor Parent(bank_index: int = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Cpp\nreactor Child(bank_index: int(0), parent_bank_index: int(0)) {\n  reaction(startup) {=\n    std::cout <<"My bank index: " << bank_index << " My parent's bank index: " << parent_bank_index << std::endl;\n  =}\n}\nreactor Parent(bank_index: int(0)) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Python\nreactor Child(bank_index=0, parent_bank_index=0) {\n  reaction(startup) {=\n    print(\n        f"My bank index: {self.bank_index}. "\n        f"My parent's bank index: {self.parent_bank_index}."\n    )\n  =}\n}\nreactor Parent(bank_index=0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target TypeScript\nreactor Child(parentBankIndex: number = 0) {\n  reaction(startup) {=\n    console.log(`My bank index: ${this.getBankIndex()} My parent's bank index: ${parentBankIndex}`)\n  =}\n}\nreactor Parent {\n  c = new[2] Child(parentBankIndex = {= this.getBankIndex() =})\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Rust\nreactor Child(bank_index: usize = 0, parent_bank_index: usize = 0) {\n  state bank_index = bank_index\n  state parent_bank_index = parent_bank_index\n  reaction(startup) {=\n    println!(\n        "My bank index: {}. My parent's bank index: {}.",\n        self.bank_index,\n        self.parent_bank_index,\n    );\n  =}\n}\nreactor Parent(bank_index: usize = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n

    $end(ChildParentBank)$

    \n

    In this example, the bank index of the Parent reactor is passed to the\nparent_bank_index parameter of the Child reactor instances.\nThe output from this program will be:

    \n
    My bank index: 1. My parent's bank index: 1.\nMy bank index: 0. My parent's bank index: 0.\nMy bank index: 0. My parent's bank index: 1.\nMy bank index: 1. My parent's bank index: 0.
    \n

    Again, note that the order of these outputs is nondeterministic.

    \n

    Finally, members of contained banks of reactors can be individually addressed in\nthe body of reactions of the parent reactor if their input/output port appears\nin the reaction signature. For example, note the following program:

    \n

    $start(ChildParentBank2)$

    \n
    target C\nreactor Child(bank_index: int = 0, parent_bank_index: int = 0) {\n  output out: int\n  reaction(startup) -> out {=\n    lf_set(out, self->parent_bank_index * 2 + self->bank_index);\n  =}\n}\nreactor Parent(bank_index: int = 0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for (int i=0; i < c_width; i++) {\n        printf("Received %d from child %d.\\n", c[i].out->value, i);\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Cpp\nreactor Child(bank_index: int(0), parent_bank_index: int(0)) {\n  output out: int\n  reaction(startup) -> out {=\n    out.set(parent_bank_index * 2 + bank_index);\n  =}\n}\nreactor Parent(bank_index: int(0)) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for (auto i = 0ul; i < c.size(); i++) {\n        std::cout << "Received " << *c[i].out.get() <<" from child " << i << std::endl;\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target Python\nreactor Child(bank_index=0, parent_bank_index=0) {\n  output out\n  reaction(startup) -> out {=\n    out.set(self.parent_bank_index * 2 + self.bank_index)\n  =}\n}\nreactor Parent(bank_index=0) {\n  c = new[2] Child(parent_bank_index=bank_index)\n  reaction(c.out) {=\n    for i, child in enumerate(c):\n        print(f"Received {child.out.value} from child {i}.")\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    target TypeScript\nreactor Child(parentBankIndex: number = 0) {\n  output out: number\n  reaction(startup) -> out {=\n    out = parentBankIndex * 2 + this.getBankIndex()\n  =}\n}\nreactor Parent {\n  c = new[2] Child(parentBankIndex = {= this.getBankIndex() =})\n  reaction(c.out) {=\n    for (let i = 0; i < c.length; i++) {\n        console.log(`Received ${c[i].out} from child ${i}`)\n    }\n  =}\n}\nmain reactor {\n  p = new[2] Parent()\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/ChildParentBank2.lf\n
    \n

    $end(ChildParentBank2)$

    \n\"Lingua\n

    Running this program will give something like the following:

    \n
    Received 0 from child 0.\nReceived 1 from child 1.\nReceived 2 from child 0.\nReceived 3 from child 1.
    \n
    \n

    Note the usage of c_width, which holds the width of the c bank of reactors.

    \n
    \n
    \n

    Note that len(c) can be used to get the width of the bank, and for p in c or for (i, p) in enumerate(c) can be used to iterate over the bank members.

    \n
    \n
    \n

    Note that c.size() can be used to get the width of the bank c.

    \n
    \n
    \n

    Note that that bank instance c in TypeScript is an array, so c.length is the width of the bank, and the bank members are referenced by indexing the array, as in c[i].

    \n
    \n
    \n

    FIXME: How to get the width of the bank in target code?

    \n
    \n

    Combining Banks and Multiports

    \n

    Banks of reactors may be combined with multiports, as in the following example:

    \n

    $start(MultiportToBank)$

    \n
    target C\nreactor Source {\n  output[3] out: int\n  reaction(startup) -> out {=\n    for(int i = 0; i < out_width; i++) {\n      lf_set(out[i], i);\n    }\n  =}\n}\nreactor Destination(bank_index: int = 0) {\n  input in: int\n  reaction(in) {=\n    printf("Destination %d received %d.\\n", self->bank_index, in->value);\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.in\n}\n
    \n
    target Cpp\nreactor Source {\n  output[3] out: int\n  reaction(startup) -> out {=\n    for(int i = 0; i < out.size(); i++) {\n      out[i].set(i);\n    }\n  =}\n}\nreactor Destination(bank_index: int(0)) {\n  input in: int\n  reaction(in) {=\n    std::cout << "Destination " << bank_index << " received " << *in.get() << std::endl;\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.in\n}\n
    \n
    target Python\nreactor Source {\n  output[3] out\n  reaction(startup) -> out {=\n    for i, port in enumerate(out):\n      port.set(i)\n  =}\n}\nreactor Destination(bank_index=0) {\n  input inp\n  reaction(inp) {=\n    print(f"Destination {self.bank_index} received {inp.value}.")\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n
    target TypeScript\nreactor Source {\n  output[3] out: number\n  reaction(startup) -> out {=\n     for (let i = 0 ; i < out.length; i++) {\n        out[i] = i\n    }\n  =}\n}\nreactor Destination {\n  input inp: number\n  reaction(inp) {=\n    console.log(`Destination ${this.getBankIndex()} received ${inp}`)\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n
    target Rust\nreactor Source {\n  output[3] out: usize\n  reaction(startup) -> out {=\n    for (i, o) in out.into_iter().enumerate() {\n      ctx.set(o, i);\n    }\n  =}\n}\nreactor Destination(bank_index: usize = 0) {\n  state bank_index = bank_index\n  input inp: usize\n  reaction(inp) {=\n    println!(\n        "Destination {} received {}.",\n        self.bank_index,\n        ctx.get(inp).unwrap(),\n    );\n  =}\n}\nmain reactor MultiportToBank {\n  a = new Source()\n  b = new[3] Destination()\n  a.out -> b.inp\n}\n
    \n

    $end(MultiportToBank)$

    \n\"Lingua\n

    The three outputs from the Source instance a will be sent, respectively, to each of three instances of Destination, b[0], b[1], and b[2]. The result of the program will be something like the following:

    \n
    Destination 0 received 0.\nDestination 1 received 1.\nDestination 2 received 2.
    \n

    Again, the order is nondeterministic in a multithreaded context.

    \n

    The reactors in a bank may themselves have multiports. In all cases, the number of ports on the left of a connection must match the number on the right, unless the ones on the left are iterated, as explained next.

    \n

    Broadcast Connections

    \n

    Occasionally, you will want to have fewer ports on the left of a connection and have their outputs used repeatedly to broadcast to the ports on the right. In the following example, the outputs from an ordinary port are broadcast to the inputs of all instances of a bank of reactors:

    \n
    reactor Source {\n  output out:int;\n  reaction(startup) -> out {=\n    ... write to out ...\n  =}\n}\nreactor Destination {\n  input in:int;\n  reaction(in) {=\n    ... read from in ...\n  =}\n}\nmain reactor ThreadedThreaded(width:int(4)) {\n  a = new Source();\n  d = new[width] Destination();\n  (a.out)+ -> d.in;\n}\n
    \n

    The syntax (a.out)+ means “repeat the output port a.out one or more times as needed to supply all the input ports of d.in.” The content inside the parentheses can be a comma-separated list of ports, the ports inside can be ordinary ports or multiports, and the reactors inside can be ordinary reactors or banks of reactors. In all cases, the number of ports inside the parentheses on the left must divide the number of ports on the right.

    \n

    Interleaved Connections

    \n
    \n

    Sometimes, we don’t want to broadcast messages to all reactors, but need more fine-grained control as to which reactor within a bank receives a message. If we have separate source and destination reactors, this can be done by combining multiports and banks as was shown in Combining Banks and Multiports. Setting a value on the index n of the output multiport, will result in a message to the n-th reactor instance within the destination bank. However, this pattern gets slightly more complicated, if we want to exchange addressable messages between instances of the same bank. This pattern is shown in the following example:

    \n

    $start(Interleaved)$

    \n
    target C\nreactor Node(num_nodes: size_t = 4, bank_index: int = 0) {\n  input[num_nodes] in: int\n  output[num_nodes] out: int\n  reaction(startup) -> out {=\n    lf_set(out[1], 42);\n    printf("Bank index %d sent 42 on channel 1.\\n", self->bank_index);\n  =}\n  reaction(in) {=\n    for (int i = 0; i < in_width; i++) {\n      if (in[i]->is_present) {\n        printf("Bank index %d received %d on channel %d.\\n",\n          self->bank_index, in[i]->value, i\n        );\n      }\n    }\n  =}\n}\nmain reactor(num_nodes: size_t = 4) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.in)\n}\n
    \n
    target Cpp\nreactor Node(num_nodes: size_t(4), bank_index: int(0)) {\n  input[num_nodes] in: int\n  output[num_nodes] out: int\n  reaction(startup) -> out {=\n    out[1].set(42);\n    std::cout << "Bank index " << bank_index << " sent 42 on channel 1." << std::endl;\n  =}\n  reaction(in) {=\n    for (auto i = 0ul; i < in.size(); i++) {\n      if (in[i].is_present()) {\n        std::cout << "Bank index " << bank_index\n          << " received " << *in[i].get() << " on channel" << std::endl;\n      }\n    }\n  =}\n}\nmain reactor(num_nodes: size_t(4)) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.in)\n}\n
    \n
    target Python\nreactor Node(num_nodes=4, bank_index=0) {\n  input[num_nodes] inp\n  output[num_nodes] out\n  reaction(startup) -> out {=\n    out[1].set(42)\n    print(f"Bank index {self.bank_index} sent 42 on channel 1.")\n  =}\n  reaction(inp) {=\n    for i, port in enumerate(inp):\n      if port.is_present:\n        print(\n          f"Bank index {self.bank_index} received {port.value} on channel {i}.",\n        )\n  =}\n}\nmain reactor(num_nodes=4) {\n  nodes = new[num_nodes] Node(num_nodes=num_nodes)\n  nodes.out -> interleaved(nodes.inp)\n}\n
    \n
    WARNING: No source file found: ../code/ts/src/Interleaved.lf\ntarget TypeScript\nreactor Node(numNodes: number(4)) {\n    input[numNodes] inp: number\n    output[numNodes] out: number\n    reaction (startup) -> out {=\n        out[1] = 42\n        console.log(`Bank index ${this.getBankIndex()} sent 42 on channel 1.`)\n    =}\n    reaction (inp) {=\n        for (let i = 0; i < in.length; i++) {\n            if (in[i] !== undefined) {\n                console.log(`Bank index ${this.getBankIndex()} received ${in[i]} on channel ${i}`)\n            }\n        }\n    =}\n}\nmain reactor(numNodes: number(4)) {\n    nodes = new[numNodes] Node(numNodes=numNodes);\n    nodes.out -> interleaved(nodes.inp)\n}\n
    \n
    WARNING: No source file found: ../code/rs/src/Interleaved.lf\n
    \n

    $end(Interleaved)$

    \n\"Lingua\n

    In the above program, four instance of Node are created, and, at startup, each instance sends 42 to its second (index 1) output channel. The result is that the second bank member (bank_index 1) will receive the number 42 on each input channel of its multiport input. Running this program gives something like the following:

    \n
    Bank index 0 sent 42 on channel 1.\nBank index 1 sent 42 on channel 1.\nBank index 2 sent 42 on channel 1.\nBank index 3 sent 42 on channel 1.\nBank index 1 received 42 on channel 0.\nBank index 1 received 42 on channel 1.\nBank index 1 received 42 on channel 2.\nBank index 1 received 42 on channel 3.
    \n

    In bank index 1, the 0-th channel receives from bank_index 0, the 1-th channel from bank_index 1, etc. In effect, the choice of output channel specifies the destination reactor in the bank, and the input channel specifies the source reactor from which the input comes.

    \n

    This style of connection is accomplished using the new keyword $interleaved$ in the connection. Normally, a port reference such as nodes.out where nodes is a bank and out is a multiport, would list all the individual ports by first iterating over the banks and then, for each bank index, iterating over the ports. If we consider the tuple (b,p) to denote the index b within the bank and the index p within the multiport, then the following list is created: (0,0), (0,1), (0,2), (0,3), (1,0), (1,1), (1,2), (1,3), (2,0), (2,1), (2,2), (2,3), (3,0), (3,1), (3,2), (3,3). However, if we use $interleaved$(nodes.out) instead, the connection logic will iterate over the ports first and then the banks, creating the following list: (0,0), (1,0), (2,0), (3,0), (0,1), (1,1), (2,1), (3,1), (0,2), (1,2), (2,2), (3,2), (0,3), (1,3), (2,3), (3,3). By combining a normal port reference with a interleaved reference, we can construct a fully connected network. The figure below visualizes this how this pattern would look without banks or multiports:

    \n\n \n \"Lingua\n \n

    If we were to use a normal connection nodes.out -> nodes.in; instead of the $interleaved$ connection, then the following pattern would be created:

    \n\n \n \"Lingua\n \n

    Effectively, this connects each reactor instance to itself, which isn’t very useful.

    \n
    \n
    \n

    The $interleaved$ keyword is not supported by $target-language$.

    \n
    ","headings":[{"value":"Multiports","depth":2},{"value":"Sparse Inputs","depth":2},{"value":"Parameterized Widths","depth":2},{"value":"Connecting Reactors with Different Widths","depth":2},{"value":"Banks of Reactors","depth":2},{"value":"Initializing Bank Members from a Table","depth":2},{"value":"Contained Banks","depth":2},{"value":"Combining Banks and Multiports","depth":2},{"value":"Broadcast Connections","depth":2},{"value":"Interleaved Connections","depth":2}],"frontmatter":{"permalink":"/docs/handbook/multiports-and-banks","title":"Multiports and Banks","oneline":"Multiports and Banks of Reactors.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Deadlines","oneline":"Deadlines in Lingua Franca.","permalink":"/docs/handbook/deadlines"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","permalink":"/docs/handbook/generics"}}}},"pageContext":{"id":"1-multiports-and-banks","slug":"/docs/handbook/multiports-and-banks","repoPath":"/packages/documentation/copy/en/topics/Multiports and Banks.md","previousID":"cb17ff3f-7a86-5f3b-95f7-e7d1f4e920c0","nextID":"2a0b9619-72b6-5ee7-8a38-83ff8d48005a","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/overview/page-data.json b/page-data/docs/handbook/overview/page-data.json index 627d03b42..7797471d8 100644 --- a/page-data/docs/handbook/overview/page-data.json +++ b/page-data/docs/handbook/overview/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/overview","result":{"data":{"markdownRemark":{"id":"8f275b71-8f6a-5273-83d2-6037d55a4966","excerpt":"Lingua Franca (LF) is a polyglot coordination language for concurrent and possibly time-sensitive applications ranging from low-level embedded code to…","html":"

    Lingua Franca (LF) is a polyglot coordination language for concurrent and possibly time-sensitive applications ranging from low-level embedded code to distributed cloud and edge applications. An LF program specifies the interactions between components called reactors. The emphasis of the framework is on ensuring deterministic interaction with explicit management of timing. The logic of each reactor is written in one of a suite of target languages (currently C, C++, Python, and TypeScript) and can integrate legacy code in those languages. A code generator synthesizes one or more programs in the target language, which are then compiled using standard toolchains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.

    \n

    Lingua Franca programs are compositions of reactors, whose functionality is decomposed into reaction, which are written in the target languages. Reactors are similar to actors, software components that send each other messages, but unlike classical actors, messages are timestamped, and concurrent composition of reactors is deterministic by default. When nondeterministic interactions are tolerable or desired, they must be explicitly coded. LF itself is a polyglot composition language, not a complete programming language. LF describes the interfaces and composition of reactors. See our publications and presentations on reactors and Lingua Franca.

    \n

    The language and compiler infrastructure is very much under development. An IDE based on Eclipse and Xtext is under development, and command-line tools are also provided. LF is, by design, extensible. To support a new target language, a code generator and a runtime system capable of coordinating the execution of a composition of reactors must be developed.

    \n

    The C runtime consists of a few thousand lines of extensively commented code, occupies tens of kilobytes for a minimal application, and is extremely fast, making it suitable even for deeply embedded microcontroller platforms. It has been tested on Linux, Windows, and Mac platforms, as well as some bare-iron platforms. On POSIX-compliant platforms, it supports multithreaded execution, automatically exploiting multiple cores while preserving determinism. It includes features for real-time execution and is particularly well suited to take advantage of platforms with predictable execution times, such as PRET machines. A distributed execution mechanism is under development that takes advantage of clock synchronization when that is available to achieve truly distributed coordination while maintaining determinism.

    \n

    Reactors

    \n

    Reactors are informally described via the following principles:

    \n
      \n
    1. Components — Reactors can have input ports, actions, and timers, all of which are triggers. They can also have output ports, local state, parameters, and an ordered list of reactions.
    2. \n
    3. Composition — A reactor may contain other reactors and manage their connections. The connections define the flow of messages, and two reactors can be connected if they are contained by the same reactor or one is directly contained in the other (i.e., connections span at most one level of hierarchy). An output port may be connected to multiple input ports, but an input port can only be connected to a single output port.
    4. \n
    5. Events — Messages sent from one reactor to another, and timer and action events each have a timestamp, a value on a logical time line. These are timestamped events that can trigger reactions. Each port, timer, and action can have at most one such event at any logical time. An event may carry a value that will be passed as an argument to triggered reactions.
    6. \n
    7. Reactions — A reaction is a procedure in a target language that is invoked in response to a trigger event, and only in response to a trigger event. A reaction can read input ports, even those that do not trigger it, and can produce outputs, but it must declare all inputs that it may read and output ports to which it may write. All inputs that it reads and outputs that it produces bear the same timestamp as its triggering event. I.e., the reaction itself is logically instantaneous, so any output events it produces are logically simultaneous with the triggering event (the two events bear the same timestamp).
    8. \n
    9. Flow of Time — Successive invocations of any single reaction occur at strictly increasing logical times. Any messages that are not read by a reaction triggered at the timestamp of the message are lost.
    10. \n
    11. Mutual Exclusion — The execution of any two reactions of the same reactor are mutually exclusive (atomic with respect to one another). Moreover, any two reactions that are invoked at the same logical time are invoked in the order specified by the reactor definition. This avoids race conditions between reactions accessing the reactor state variables.
    12. \n
    13. Determinism — A Lingua Franca program is deterministic unless the programmer explicit uses nondeterministic constructs. Given the same input data, a composition of reactors has exactly one correct behavior. This makes Lingua Franca programs testable.
    14. \n
    15. Concurrency — Dependencies between reactions are explicitly declared in a Lingua Franca program, and reactions that are not dependent on one another can be executed in parallel on a multicore machine. If the target provides a distributed runtime, using Ptides for example, then execution can also be distributed across networks.
    16. \n
    \n

    Time

    \n

    Lingua Franca has a notion of logical time, where every message occurs at a logical instant and reactions to messages are logically instantaneous. At a logical time instant, each reactor input will either have a message (the input is present) or will not (the input is absent). Reactions belonging to the reactor may be triggered by a present input. Reactions may also be triggered by timers or actions. A reaction may produce outputs, in which case, inputs to which the output is connected will become present at the same logical time instant. Outputs, therefore, are logically simultaneous with the inputs that cause them. A reaction may also schedule actions which will trigger reactions of the same reactor at a later logical time.

    \n

    In the C target,\na timestamp is an unsigned 64-bit integer which, on most platforms,\nspecifies the number of nanoseconds since January 1, 1970.\nSince a 64-bit number has a limited range,\nthis measure of time instants will overflow in approximately the year 2554.\nWhen an LF program starts executing, logical time is (normally) set to the current physical time provided by the operating system.\n(On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    At the starting logical time, reactions that specify a startup trigger will execute. Also, any reactions that are triggered by a timer with a zero offset will execute. Any outputs produced by these reactions will have the same logical time and will trigger execution of any downstream reactions. Those downstream reactions will be invoked at the same logical time unless some connection to the downstream reactor uses the after keyword to specify a time delay. After all reactions at the starting logical time have completed, then time will advance to the logical time of the earliest next event. The earliest next event may be specified by a timer, by an after keyword on a connection, or by a logical or physical action. Once logical time advances, any reactions that are triggered by events at that logical will be invoked, as will any reactions triggered by outputs produced by those reactions.

    \n

    Time in Lingua Franca is actually superdense time, meaning that a logical time may have the same numerical value but also be strictly later than another logical time. When an action is scheduled with a delay of zero, it occurs at such a strictly later time, one microstep later. See the actions description. The after keyword with a time delay of 0 (zero) will also cause the downstream reactions to execute in the next microstep.

    \n

    At any logical time, if any two reactions belonging to the same reactor are triggered, they will be executed\natomically in the order in which they are defined in the reactor. Dependencies across reactors (connections with no after) will also result in sequential execution. Specifically, if any reaction of reactor A produces an output that triggers a reaction of reactor B, then B’s reaction will execute only after A’s reaction has completed execution. Modulo these two ordering constraints, reactions may execute in parallel on multiple cores or even across networks.

    \n

    Reactions are given in a target language, whereas inputs, outputs, actions, and the dependencies among them are defined in Lingua Franca. LF is, therefore, a kind of coordination language rather than a programming language.

    \n

    Real-Time Systems

    \n

    Reactions may have delays and deadlines associated with them. This information can be used to perform earliest deadline first (EDF) scheduling. Also, in combination with execution-time analysis of reactions, it should be possible to determine at compile time whether the imposed deadlines can be met, although these analysis tools have not yet been developed. Of particular interest is to deploy reactors on platforms such as FlexPRET and Patmos, which are designed for predictable timing; execution-time estimates for these architectures will be much tighter than currently possible with ordinary microprocessors.

    \n

    Schedulability Analysis of LF Programs

    \n
      \n
    • Start with classic schedulability test: critical instant (Liu and Layland)\n
        \n
      • Possible not the worst-case?
      • \n
      • Question: can not producing an output lead to a timing anomaly?
      • \n
      \n
    • \n
    • Classic schedulability analysis becomes messy when deadline with locks for communication (priority inversion, locking protocols to avoid it, recursive schedulability analysis)\n
        \n
      • We do not use shared data protected by locks +1
      • \n
      \n
    • \n
    • First (pessimistic) approach: all input events and timers fire at the same time, check all execution chains (reactions in actors) need to finish before the actuator deadlines.
    • \n
    • Delays (timer, after, scheduled actions) break the dependency chain\n
        \n
      • Schedulability analysis can be broken up into sub-chains +1
      • \n
      \n
    • \n
    • For now assume no preemptions, this also enables WCET analysis of reactions
    • \n
    • We could also use a big hammer: model the LF program as timed automata and do model checking (e.g., UupAal)
    • \n
    \n

    To Do List

    \n

    Lingua Franca is a work in progress. See our project page for an overview of ongoing and future work.

    ","headings":[{"value":"Reactors","depth":2},{"value":"Time","depth":2},{"value":"Real-Time Systems","depth":2},{"value":"Schedulability Analysis of LF Programs","depth":3},{"value":"To Do List","depth":2}],"frontmatter":{"permalink":"/docs/handbook/overview","title":"Overview","oneline":"Overview of Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Tutorial Video","oneline":"Tutorial video presented by the Lingua Franca team.","permalink":"/docs/handbook/tutorial-video"}}}},"pageContext":{"id":"0-overview","slug":"/docs/handbook/overview","repoPath":"/packages/documentation/copy/en/topics/Overview.md","nextID":"9c60c2f7-2bab-51e5-bbc5-e7957b224b3a","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/overview","result":{"data":{"markdownRemark":{"id":"8f275b71-8f6a-5273-83d2-6037d55a4966","excerpt":"Lingua Franca (LF) is a polyglot coordination language for concurrent and possibly time-sensitive applications ranging from low-level embedded code to…","html":"

    Lingua Franca (LF) is a polyglot coordination language for concurrent and possibly time-sensitive applications ranging from low-level embedded code to distributed cloud and edge applications. An LF program specifies the interactions between components called reactors. The emphasis of the framework is on ensuring deterministic interaction with explicit management of timing. The logic of each reactor is written in one of a suite of target languages (currently C, C++, Python, and TypeScript) and can integrate legacy code in those languages. A code generator synthesizes one or more programs in the target language, which are then compiled using standard toolchains. If the application has exploitable parallelism, then it executes transparently on multiple cores without compromising determinacy. A distributed application translates into multiple programs and scripts to launch those programs on distributed machines. The communication fabric connecting components is synthesized as part of the programs.

    \n

    Lingua Franca programs are compositions of reactors, whose functionality is decomposed into reaction, which are written in the target languages. Reactors are similar to actors, software components that send each other messages, but unlike classical actors, messages are timestamped, and concurrent composition of reactors is deterministic by default. When nondeterministic interactions are tolerable or desired, they must be explicitly coded. LF itself is a polyglot composition language, not a complete programming language. LF describes the interfaces and composition of reactors. See our publications and presentations on reactors and Lingua Franca.

    \n

    The language and compiler infrastructure is very much under development. An IDE based on Eclipse and Xtext is under development, and command-line tools are also provided. LF is, by design, extensible. To support a new target language, a code generator and a runtime system capable of coordinating the execution of a composition of reactors must be developed.

    \n

    The C runtime consists of a few thousand lines of extensively commented code, occupies tens of kilobytes for a minimal application, and is extremely fast, making it suitable even for deeply embedded microcontroller platforms. It has been tested on Linux, Windows, and Mac platforms, as well as some bare-iron platforms. On POSIX-compliant platforms, it supports multithreaded execution, automatically exploiting multiple cores while preserving determinism. It includes features for real-time execution and is particularly well suited to take advantage of platforms with predictable execution times, such as PRET machines. A distributed execution mechanism is under development that takes advantage of clock synchronization when that is available to achieve truly distributed coordination while maintaining determinism.

    \n

    Reactors

    \n

    Reactors are informally described via the following principles:

    \n
      \n
    1. Components — Reactors can have input ports, actions, and timers, all of which are triggers. They can also have output ports, local state, parameters, and an ordered list of reactions.
    2. \n
    3. Composition — A reactor may contain other reactors and manage their connections. The connections define the flow of messages, and two reactors can be connected if they are contained by the same reactor or one is directly contained in the other (i.e., connections span at most one level of hierarchy). An output port may be connected to multiple input ports, but an input port can only be connected to a single output port.
    4. \n
    5. Events — Messages sent from one reactor to another, and timer and action events each have a timestamp, a value on a logical time line. These are timestamped events that can trigger reactions. Each port, timer, and action can have at most one such event at any logical time. An event may carry a value that will be passed as an argument to triggered reactions.
    6. \n
    7. Reactions — A reaction is a procedure in a target language that is invoked in response to a trigger event, and only in response to a trigger event. A reaction can read input ports, even those that do not trigger it, and can produce outputs, but it must declare all inputs that it may read and output ports to which it may write. All inputs that it reads and outputs that it produces bear the same timestamp as its triggering event. I.e., the reaction itself is logically instantaneous, so any output events it produces are logically simultaneous with the triggering event (the two events bear the same timestamp).
    8. \n
    9. Flow of Time — Successive invocations of any single reaction occur at strictly increasing logical times. Any messages that are not read by a reaction triggered at the timestamp of the message are lost.
    10. \n
    11. Mutual Exclusion — The execution of any two reactions of the same reactor are mutually exclusive (atomic with respect to one another). Moreover, any two reactions that are invoked at the same logical time are invoked in the order specified by the reactor definition. This avoids race conditions between reactions accessing the reactor state variables.
    12. \n
    13. Determinism — A Lingua Franca program is deterministic unless the programmer explicit uses nondeterministic constructs. Given the same input data, a composition of reactors has exactly one correct behavior. This makes Lingua Franca programs testable.
    14. \n
    15. Concurrency — Dependencies between reactions are explicitly declared in a Lingua Franca program, and reactions that are not dependent on one another can be executed in parallel on a multicore machine. If the target provides a distributed runtime, using Ptides for example, then execution can also be distributed across networks.
    16. \n
    \n

    Time

    \n

    Lingua Franca has a notion of logical time, where every message occurs at a logical instant and reactions to messages are logically instantaneous. At a logical time instant, each reactor input will either have a message (the input is present) or will not (the input is absent). Reactions belonging to the reactor may be triggered by a present input. Reactions may also be triggered by timers or actions. A reaction may produce outputs, in which case, inputs to which the output is connected will become present at the same logical time instant. Outputs, therefore, are logically simultaneous with the inputs that cause them. A reaction may also schedule actions which will trigger reactions of the same reactor at a later logical time.

    \n

    In the C target,\na timestamp is an unsigned 64-bit integer which, on most platforms,\nspecifies the number of nanoseconds since January 1, 1970.\nSince a 64-bit number has a limited range,\nthis measure of time instants will overflow in approximately the year 2554.\nWhen an LF program starts executing, logical time is (normally) set to the current physical time provided by the operating system.\n(On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    At the starting logical time, reactions that specify a startup trigger will execute. Also, any reactions that are triggered by a timer with a zero offset will execute. Any outputs produced by these reactions will have the same logical time and will trigger execution of any downstream reactions. Those downstream reactions will be invoked at the same logical time unless some connection to the downstream reactor uses the after keyword to specify a time delay. After all reactions at the starting logical time have completed, then time will advance to the logical time of the earliest next event. The earliest next event may be specified by a timer, by an after keyword on a connection, or by a logical or physical action. Once logical time advances, any reactions that are triggered by events at that logical will be invoked, as will any reactions triggered by outputs produced by those reactions.

    \n

    Time in Lingua Franca is actually superdense time, meaning that a logical time may have the same numerical value but also be strictly later than another logical time. When an action is scheduled with a delay of zero, it occurs at such a strictly later time, one microstep later. See the actions description. The after keyword with a time delay of 0 (zero) will also cause the downstream reactions to execute in the next microstep.

    \n

    At any logical time, if any two reactions belonging to the same reactor are triggered, they will be executed\natomically in the order in which they are defined in the reactor. Dependencies across reactors (connections with no after) will also result in sequential execution. Specifically, if any reaction of reactor A produces an output that triggers a reaction of reactor B, then B’s reaction will execute only after A’s reaction has completed execution. Modulo these two ordering constraints, reactions may execute in parallel on multiple cores or even across networks.

    \n

    Reactions are given in a target language, whereas inputs, outputs, actions, and the dependencies among them are defined in Lingua Franca. LF is, therefore, a kind of coordination language rather than a programming language.

    \n

    Real-Time Systems

    \n

    Reactions may have delays and deadlines associated with them. This information can be used to perform earliest deadline first (EDF) scheduling. Also, in combination with execution-time analysis of reactions, it should be possible to determine at compile time whether the imposed deadlines can be met, although these analysis tools have not yet been developed. Of particular interest is to deploy reactors on platforms such as FlexPRET and Patmos, which are designed for predictable timing; execution-time estimates for these architectures will be much tighter than currently possible with ordinary microprocessors.

    \n

    Schedulability Analysis of LF Programs

    \n
      \n
    • Start with classic schedulability test: critical instant (Liu and Layland)\n
        \n
      • Possible not the worst-case?
      • \n
      • Question: can not producing an output lead to a timing anomaly?
      • \n
      \n
    • \n
    • Classic schedulability analysis becomes messy when deadline with locks for communication (priority inversion, locking protocols to avoid it, recursive schedulability analysis)\n
        \n
      • We do not use shared data protected by locks +1
      • \n
      \n
    • \n
    • First (pessimistic) approach: all input events and timers fire at the same time, check all execution chains (reactions in actors) need to finish before the actuator deadlines.
    • \n
    • Delays (timer, after, scheduled actions) break the dependency chain\n
        \n
      • Schedulability analysis can be broken up into sub-chains +1
      • \n
      \n
    • \n
    • For now assume no preemptions, this also enables WCET analysis of reactions
    • \n
    • We could also use a big hammer: model the LF program as timed automata and do model checking (e.g., UupAal)
    • \n
    \n

    To Do List

    \n

    Lingua Franca is a work in progress. See our project page for an overview of ongoing and future work.

    ","headings":[{"value":"Reactors","depth":2},{"value":"Time","depth":2},{"value":"Real-Time Systems","depth":2},{"value":"Schedulability Analysis of LF Programs","depth":3},{"value":"To Do List","depth":2}],"frontmatter":{"permalink":"/docs/handbook/overview","title":"Overview","oneline":"Overview of Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Tutorial Video","oneline":"Tutorial video presented by the Lingua Franca team.","permalink":"/docs/handbook/tutorial-video"}}}},"pageContext":{"id":"0-overview","slug":"/docs/handbook/overview","repoPath":"/packages/documentation/copy/en/topics/Overview.md","nextID":"9c60c2f7-2bab-51e5-bbc5-e7957b224b3a","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/parameters-and-state-variables/page-data.json b/page-data/docs/handbook/parameters-and-state-variables/page-data.json index 54629401c..6cd9c5602 100644 --- a/page-data/docs/handbook/parameters-and-state-variables/page-data.json +++ b/page-data/docs/handbook/parameters-and-state-variables/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/parameters-and-state-variables","result":{"data":{"markdownRemark":{"id":"6b93641e-db40-5450-88fc-265623991a01","excerpt":"$page-showing-target$ Parameter Declaration A reactor class definition can parameterized as follows: Each parameter has a type annotation, written :$page-showing-target$

    \n

    Parameter Declaration

    \n

    A reactor class definition can parameterized as follows:

    \n
    \n
    reactor <class-name>(<param-name>:<type> = <expr>, ...) {\n    ...\n}\n
    \n

    Each parameter has a type annotation, written :<type>, where <type> has one of the following forms:

    \n
      \n
    • An identifier, such as int, possibly followed by a type argument, e.g. vector<int>.
    • \n
    • An array type type[] and type[integer].
    • \n
    • The keyword $time$, which designates a time value.
    • \n
    • A code block delimited by {= ... =}, where the contents is any valid type in the target language.
    • \n
    \n
    \n
    \n
      \n
    • A pointer type, such as int*.
    • \n
    \n
    \n
    \n

    Types ending with a * are treated specially by the C target. See the Target Language Details.

    \n

    To use strings conveniently in the C target, the “type” string is an alias for {=const char*=}.

    \n
    \n
    \n

    For example, {= int | null =} defines nullable integer type in TypeScript.

    \n
    \n
    \n
    reactor <class-name>(<param-name> = <expr>, ... ) {\n    ...\n}\n
    \n
    \n

    Depending on the target, the type may be a generic type, which means that the type is parameter determined at the time the reactor class is instantiated.

    \n

    Each parameter must have a default value, written <param-name> = <expr>. An expression may be a numeric constant, a string enclosed in quotation marks, a time value such as 10 msec, a list of values, or target-language code enclosed in {= ... =}, for example. See Expressions for full details on what expressions are valid.

    \n

    For example, the Double reactor on the previous page can be replaced with a more general parameterized reactor Scale as follows:

    \n

    $start(Scale)$

    \n
    target C\nreactor Scale(factor: int = 2) {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    lf_set(y, x->value * self->factor);\n  =}\n}\n
    \n
    target Cpp\nreactor Scale(factor: int(2)) {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    y.set(factor * *x.get());\n  =}\n}\n
    \n
    target Python\nreactor Scale(factor=2) {\n  input x\n  output y\n  reaction(x) -> y {=\n    y.set(x.value * self.factor)\n  =}\n}\n
    \n
    target TypeScript\nreactor Scale(factor: number = 2) {\n  input x: number\n  output y: number\n  reaction(x) -> y {=\n    if (x !== undefined) y = x * factor\n  =}\n}\n
    \n
    target Rust\nreactor Scale(factor: u32 = 2) {\n  state factor = factor\n  input x: u32\n  output y: u32\n  reaction(x) -> y {=\n    let x = ctx.get(x).unwrap();\n    ctx.set(y, x * self.factor);\n  =}\n}\n
    \n

    $end(Scale)$

    \n

    This reactor, given any input event x will produce an output y with value equal to the input scaled by the factor parameter. The default value of the factor parameter is 2, but this can be changed when the Scale reactor is instantiated.

    \n

    Notice how, within the body of a reaction, the code accesses the parameter value. This is different for each target language. In the C target, a self struct is provided that contains the parameter values.

    \n

    State Declaration

    \n

    A reactor declares a state variable as follows:

    \n
    \n
      state <name>:<type> = <value>\n
    \n

    The type can any of the same forms as for a parameter.

    \n
    \n
    \n
      state <name> = <value>\n
    \n
    \n

    The <value> is an initial value and, like parameter values, can be given as an expression or target language code with delimiters {= ... =}. The initial value can also be given as a parameter name. The value can be accessed and modified in a target-language-dependent way as illustrated by the following example:

    \n

    $start(Count)$

    \n
    target C\nreactor Count {\n  state count: int = 0\n  output y: int\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    lf_set(y, self->count++);\n  =}\n}\n
    \n
    target Cpp\nreactor Count {\n  state count: int(0)\n  output y: int\n  timer t(0, 100 ms)\n  reaction(t) -> y {=\n    y.set(count++);\n  =}\n}\n
    \n
    target Python\nreactor Count {\n  state count = 0\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(self.count)\n    self.count += 1\n  =}\n}\n
    \n
    target TypeScript\nreactor Count {\n  state count: number = 0\n  output y: number\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y = count++\n  =}\n}\n
    \n
    target Rust\nreactor Count {\n  state count: u32 = 0\n  output y: u32\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    ctx.set(y, self.count);\n    self.count += 1;\n  =}\n}\n
    \n

    $end(Count)$

    \n

    This reactor has an integer state variable named count, and each time its reaction is invoked, it outputs the value of that state variable and increments it. The reaction is triggered by a $timer$, discussed in the next section.

    \n

    Reset State Variables

    \n
    \n

    The $reset$ keyword is not supported in $target-language$ because modal reactors are not supported.

    \n
    \n
    \n

    A state variable declaration may be qualified with a $reset$ keyword as follows:

    \n
      reset state <name>:<type> = <value>\n
    \n
      reset state <name> = <value>\n
    \n

    When this is done, if the state variable or the reactor is within a mode of a modal reactor, then when the mode is entered via a reset transition, the state variable will be reset to its initial value. For details, see the Modal Reactors section.

    \n
    ","headings":[{"value":"Parameter Declaration","depth":2},{"value":"State Declaration","depth":2},{"value":"Reset State Variables","depth":2}],"frontmatter":{"permalink":"/docs/handbook/parameters-and-state-variables","title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","permalink":"/docs/handbook/inputs-and-outputs"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","permalink":"/docs/handbook/time-and-timers"}}}},"pageContext":{"id":"1-parameters-and-state-variables","slug":"/docs/handbook/parameters-and-state-variables","repoPath":"/packages/documentation/copy/en/topics/Parameters and State Variables.md","previousID":"dcdc6b32-76b0-570a-a6f8-23bb570863c7","nextID":"bba867ed-95b9-5017-b4f1-350e621e99da","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/parameters-and-state-variables","result":{"data":{"markdownRemark":{"id":"6b93641e-db40-5450-88fc-265623991a01","excerpt":"$page-showing-target$ Parameter Declaration A reactor class definition can parameterized as follows: Each parameter has a type annotation, written :$page-showing-target$

    \n

    Parameter Declaration

    \n

    A reactor class definition can parameterized as follows:

    \n
    \n
    reactor <class-name>(<param-name>:<type> = <expr>, ...) {\n    ...\n}\n
    \n

    Each parameter has a type annotation, written :<type>, where <type> has one of the following forms:

    \n
      \n
    • An identifier, such as int, possibly followed by a type argument, e.g. vector<int>.
    • \n
    • An array type type[] and type[integer].
    • \n
    • The keyword $time$, which designates a time value.
    • \n
    • A code block delimited by {= ... =}, where the contents is any valid type in the target language.
    • \n
    \n
    \n
    \n
      \n
    • A pointer type, such as int*.
    • \n
    \n
    \n
    \n

    Types ending with a * are treated specially by the C target. See the Target Language Details.

    \n

    To use strings conveniently in the C target, the “type” string is an alias for {=const char*=}.

    \n
    \n
    \n

    For example, {= int | null =} defines nullable integer type in TypeScript.

    \n
    \n
    \n
    reactor <class-name>(<param-name> = <expr>, ... ) {\n    ...\n}\n
    \n
    \n

    Depending on the target, the type may be a generic type, which means that the type is parameter determined at the time the reactor class is instantiated.

    \n

    Each parameter must have a default value, written <param-name> = <expr>. An expression may be a numeric constant, a string enclosed in quotation marks, a time value such as 10 msec, a list of values, or target-language code enclosed in {= ... =}, for example. See Expressions for full details on what expressions are valid.

    \n

    For example, the Double reactor on the previous page can be replaced with a more general parameterized reactor Scale as follows:

    \n

    $start(Scale)$

    \n
    target C\nreactor Scale(factor: int = 2) {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    lf_set(y, x->value * self->factor);\n  =}\n}\n
    \n
    target Cpp\nreactor Scale(factor: int(2)) {\n  input x: int\n  output y: int\n  reaction(x) -> y {=\n    y.set(factor * *x.get());\n  =}\n}\n
    \n
    target Python\nreactor Scale(factor=2) {\n  input x\n  output y\n  reaction(x) -> y {=\n    y.set(x.value * self.factor)\n  =}\n}\n
    \n
    target TypeScript\nreactor Scale(factor: number = 2) {\n  input x: number\n  output y: number\n  reaction(x) -> y {=\n    if (x !== undefined) y = x * factor\n  =}\n}\n
    \n
    target Rust\nreactor Scale(factor: u32 = 2) {\n  state factor = factor\n  input x: u32\n  output y: u32\n  reaction(x) -> y {=\n    let x = ctx.get(x).unwrap();\n    ctx.set(y, x * self.factor);\n  =}\n}\n
    \n

    $end(Scale)$

    \n

    This reactor, given any input event x will produce an output y with value equal to the input scaled by the factor parameter. The default value of the factor parameter is 2, but this can be changed when the Scale reactor is instantiated.

    \n

    Notice how, within the body of a reaction, the code accesses the parameter value. This is different for each target language. In the C target, a self struct is provided that contains the parameter values.

    \n

    State Declaration

    \n

    A reactor declares a state variable as follows:

    \n
    \n
      state <name>:<type> = <value>\n
    \n

    The type can any of the same forms as for a parameter.

    \n
    \n
    \n
      state <name> = <value>\n
    \n
    \n

    The <value> is an initial value and, like parameter values, can be given as an expression or target language code with delimiters {= ... =}. The initial value can also be given as a parameter name. The value can be accessed and modified in a target-language-dependent way as illustrated by the following example:

    \n

    $start(Count)$

    \n
    target C\nreactor Count {\n  state count: int = 0\n  output y: int\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    lf_set(y, self->count++);\n  =}\n}\n
    \n
    target Cpp\nreactor Count {\n  state count: int(0)\n  output y: int\n  timer t(0, 100 ms)\n  reaction(t) -> y {=\n    y.set(count++);\n  =}\n}\n
    \n
    target Python\nreactor Count {\n  state count = 0\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(self.count)\n    self.count += 1\n  =}\n}\n
    \n
    target TypeScript\nreactor Count {\n  state count: number = 0\n  output y: number\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y = count++\n  =}\n}\n
    \n
    target Rust\nreactor Count {\n  state count: u32 = 0\n  output y: u32\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    ctx.set(y, self.count);\n    self.count += 1;\n  =}\n}\n
    \n

    $end(Count)$

    \n

    This reactor has an integer state variable named count, and each time its reaction is invoked, it outputs the value of that state variable and increments it. The reaction is triggered by a $timer$, discussed in the next section.

    \n

    Reset State Variables

    \n
    \n

    The $reset$ keyword is not supported in $target-language$ because modal reactors are not supported.

    \n
    \n
    \n

    A state variable declaration may be qualified with a $reset$ keyword as follows:

    \n
      reset state <name>:<type> = <value>\n
    \n
      reset state <name> = <value>\n
    \n

    When this is done, if the state variable or the reactor is within a mode of a modal reactor, then when the mode is entered via a reset transition, the state variable will be reset to its initial value. For details, see the Modal Reactors section.

    \n
    ","headings":[{"value":"Parameter Declaration","depth":2},{"value":"State Declaration","depth":2},{"value":"Reset State Variables","depth":2}],"frontmatter":{"permalink":"/docs/handbook/parameters-and-state-variables","title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Inputs and Outputs","oneline":"Inputs, outputs, and reactions in Lingua Franca.","permalink":"/docs/handbook/inputs-and-outputs"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","permalink":"/docs/handbook/time-and-timers"}}}},"pageContext":{"id":"1-parameters-and-state-variables","slug":"/docs/handbook/parameters-and-state-variables","repoPath":"/packages/documentation/copy/en/topics/Parameters and State Variables.md","previousID":"dcdc6b32-76b0-570a-a6f8-23bb570863c7","nextID":"bba867ed-95b9-5017-b4f1-350e621e99da","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/preambles/page-data.json b/page-data/docs/handbook/preambles/page-data.json index 6005bc613..b57a5dcf8 100644 --- a/page-data/docs/handbook/preambles/page-data.json +++ b/page-data/docs/handbook/preambles/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/preambles","result":{"data":{"markdownRemark":{"id":"7f5f499e-179a-51f4-a401-ed9dbd103eb9","excerpt":"$page-showing-target$ Preamble Reactions may contain arbitrary target-language code, but often it is convenient for that code to invoke external libraries or to…","html":"

    $page-showing-target$

    \n

    Preamble

    \n

    Reactions may contain arbitrary target-language code, but often it is convenient for that code to invoke external libraries or to share procedure definitions. For either purpose, a reactor may include a $preamble$ section.

    \n
    \n

    For example, the following reactor uses the math C library for its trigonometric functions:

    \n
    main reactor {\n  preamble {=\n    #include <math.h>\n  =}\n  reaction(startup) {=\n    printf("The cosine of 1 is %f.\\n", cos(1));\n  =}\n}\n
    \n

    This will print:

    \n
    The cosine of 1 is 0.540302.
    \n

    By putting the #include in the $preamble$, the library becomes available in all reactions of this reactor.\nIf you wish to have the library available in all reactors in the same file, you can provide the $preamble$ outside the reactor, as shown here:

    \n
    preamble {=\n  #include <math.h>\n=}\nreactor Cos {\n  reaction(startup) {=\n    printf("The cosine of 1 is %f.\\n", cos(1));\n  =}\n}\nreactor Sin {\n  reaction(startup) {=\n    printf("The sine of 1 is %f.\\n", sin(1));\n  =}\n}\nmain reactor {\n  c = new Cos()\n  s = new Sin()\n}\n
    \n

    You can also use the $preamble$ to define functions that are shared across reactions within a reactor, as in this example:

    \n
    main reactor {\n  preamble {=\n    int add_42(int i) {\n      return i + 42;\n    }\n  =}\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\n
    \n

    Not surprisingly, this will print:

    \n
    42 plus 42 is 84.\n42 plus 1 is 43.
    \n

    (The order in which these are printed is arbitrary because the reactions can execute in parallel.)

    \n

    To share a function across reactors, however, is a bit trickers.\nA $preamble$ that is put outside the $reactor$ definition can only contain\ndeclarations not definitions of functions or variables.\nThe following code, for example will fail to compile:

    \n
    preamble {=\n  int add_42(int i) {\n    return i + 42;\n  }\n=}\nreactor Add_42 {\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n}\nreactor Add_1 {\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\nmain reactor {\n  a = new Add_42()\n  b = new Add_1()\n}\n
    \n

    The compiler will issue a duplicate symbol error because the function definition gets repeated in the separate C files generated for the two reactor classes, Add_42 and Add_1. When the compiled C code gets linked, the linker will find two definitions for the function add_42.

    \n

    To correct this compile error, the file-level preamble should contain only a declaration, not a definition, as here:

    \n
    preamble {=\n  int add_42(int i);\n=}\nreactor Add_42 {\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n}\nreactor Add_1 {\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\nmain reactor {\n  preamble {=\n    int add_42(int i) {\n      return i + 42;\n    }\n  =}\n  a = new Add_42()\n  b = new Add_1()\n}\n
    \n

    The function definition here is put into the main reactor, but it can be put in any reactor defined in the file.

    \n

    Most header files contain only declarations, and hence can be safely included\nusing #include in a file-level $preamble$. If you wish to use a header file that includes both declarations and definitions, then you will need to include it within each reactor that uses it.

    \n

    If you wish to share variables across reactors, similar constraints apply.\nNote that sharing variables across reactors is strongly discouraged because it can undermine the determinacy of Lingua Franca, and you may have to implement mutual-exclusion locks to access such variables. But it is occassionaly justfiable, as in the following example:

    \n
    preamble {=\n  extern const char shared_string[];\n=}\nreactor A {\n  reaction(startup) {=\n    printf("Reactor A says %s.\\n", shared_string);\n  =}\n}\nreactor B {\n  reaction(startup) {=\n    printf("Reactor B says %s.\\n", shared_string);\n  =}\n}\nmain reactor {\n  preamble {=\n    const char shared_string[] = "Hello";\n  =}\n  a = new A()\n  b = new B()\n}\n
    \n

    Notice the use of the extern keyword in C, which is required because the definition of the shared_string variable will be in a separate (code-generated) C file, the one for main, not the ones for A and B.

    \n

    One subtlety is that if you define symbols that you will use in $input$, $output$, or $state$ declarations, then the symbols must be defined in a file-level $preamble$.\nSpecifically, the following code will fail to compile:

    \n
    main reactor {\n  preamble {=\n    typedef int foo;\n  =}\n  state x:foo = 0\n  reaction(startup) {=\n    lf_print("State is %d", self->x);\n  =}\n}\n
    \n

    The compiler will issue an unknown type name error. To correct this, just move the declaration to a file-level $preamble$:

    \n
    preamble {=\n  typedef int foo;\n=}\nmain reactor {\n  state x:foo = 0\n  reaction(startup) {=\n    lf_print("State is %d", self->x);\n  =}\n}\n
    \n
    \n
    \n

    For example, the following reactor uses the charconv header from the c++ standard library to convert a string to an integer:

    \n
    target Cpp;\n\nmain reactor {\n  private preamble {=\n    #include <charconv>\n    #include <string>\n  =}\n\n  timer t;\n  reaction(t) {=\n    std::string raw = "42";\n    std::size_t number;\n\n    auto result = std::from_chars(raw.data(), raw.data() + raw.size(), number);\n    if (result.ec == std::errc::invalid_argument) {\n      std::cerr << "Could not convert.";\n    } else {\n      std::cout << "Converted string: " << raw << " to integer: " << number << std::endl;\n    }\n  =}\n}\n
    \n

    This will print:

    \n
    [INFO]  Starting the execution\nConverted string: 42 to integer: 42\n[INFO]  Terminating the execution
    \n

    By putting the #include in the preamble, the library becomes available in all reactions of this reactor. Note the private qualifier before the preamble keyword.\nThis ensures that the preamble is only visible to the reactions defined in this reactor and not to any other reactors. In contrast,\nthe public qualifier ensures that the preamble is also visible to other reactors in files that import the reactor defining the public preamble.

    \n
    reactor Preamble {\n  public preamble {=\n    struct MyStruct {\n      int foo;\n      std::string bar;\n    };\n  =}\n\n  private preamble {=\n    auto add_42(int i) noexcept -> int {\n      return i + 42;\n    }\n  =}\n\n  logical action a:MyStruct;\n\n  reaction(startup) {=\n    a.schedule({add_42(42), "baz"});\n  =}\n\n  reaction(a) {=\n    auto value = *a.get();\n    std::cout << "Received " << value.foo << " and '" << value.bar << "'\\n";\n  =}\n}\n
    \n

    It defines both a public and a private preamble. The public preamble defines the type MyStruct. This type definition will be visible to all elements of the\nPreamble reactor as well as to all reactors defined in files that import Preamble. The private preamble defines the function add_42(int i).\nThis function will only be usable to reactions within the Preamble reactor.

    \n

    You can think of public and private preambles as the equivalent of header files and source files in C++. In fact, the public preamble will be translated to a\nheader file and the private preamble to a source file. As a rule of thumb, all types that are used in port or action definitions as well as in state variables\nor parameters should be defined in a public preamble. Also, declarations of functions to be shared across reactors should be placed in the public preamble.\nEverything else, like function definitions or types that are used only within reactions, should be placed in a private preamble.

    \n

    Note that preambles can also be specified on the file level. These file level preambles are visible to all reactors within the file.\nAn example of this can be found in PreambleFile.lf.

    \n

    Admittedly, the precise interactions of preambles and imports can become confusing. The preamble mechanism will likely be refined in future revisions.

    \n

    Note that functions defined in the preamble cannot access members such as state variables of the reactor unless they are explicitly passed as arguments.\nIf access to the inner state of a reactor is required, methods present a viable and easy to use alternative.

    \n
    \n
    \n

    For example, the following reactor uses the platform module to print the platform information and a defined method to add 42 to an integer:

    \n
    main reactor Preamble {\n\tpreamble {=\n\t\timport platform\n\t\tdef add_42(self, i):\n\t\t\treturn i + 42\n\t=}\n\ttimer t;\n\treaction(t) {=\n\t\ts = "42"\n\t\ti = int(s)\n\t\tprint("Converted string {:s} to int {:d}.".format(s, i))\n\t\tprint("42 plus 42 is ", self.add_42(42))\n\t\tprint("Your platform is ", self.platform.system())\n\t=}\n}\n
    \n

    On a Linux machine, this will print:

    \n
    Converted string 42 to int 42.\n42 plus 42 is 84\nYour platform is Linux
    \n

    By putting import in the $preamble$, the module becomes available in all reactions of this reactor using the self modifier.

    \n

    Note: Preambles will be put in the generated Python class for the given reactor, and thus is part of the instance of the reactor. This means that anything you put in the preamble will be specific to a particular reactor instance and cannot be used to share information between different instantiations of the reactor (this is a feature, not a bug, because it helps ensure determinacy). For more information about implementation details of the Python target, see Implementation Details.

    \n

    Alternatively, you can define a $preamble$ outside any reactor definition. Such a $preamble$ can be used for functions such as import or to define a global function. The following example shows importing the hello module:

    \n
    target Python {\n  files: include/hello.py\n};\npreamble {=\n  import hello\n=}\n
    \n

    Notice the usage of the files target property to move the hello.py module located in the include folder of the test directory into the working directory (located in src-gen/NAME).

    \n

    For another example, the following program uses the built-in Python input() function to get typed input from the user:

    \n
    target Python\nmain reactor {\n  preamble {=\n    import threading\n    def external(self, a):\n      while (True):\n        from_user = input() # Blocking\n        a.schedule(0, from_user)\n  =}\n  state thread\n  physical action a\n  timer t(2 secs, 2 secs)\n\n  reaction(startup) -> a {=\n    self.thread = self.threading.Thread(target=self.external, args=(a,))\n    self.thread.start()\n    print("Type something.")\n  =}\n\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"A time {elapsed_time} nsec after start, received: ", a.value)\n  =}\n\n  reaction(t) {=\n    print("Waiting ...")\n  =}\n}\n
    \n

    Within the $preamble$, we specify to import the threading Python module and define a function that will be started in a separate thread in the reaction to $startup$. The thread function named external blocks when input() is called until the user types something and hits the return or enter key. Usually, you do not want a Lingua Franca program to block waiting for input. In the above reactor, a $timer$ is used to repeatedly trigger a reaction that reminds the user that it is waiting for input.

    \n
    \n
    \n

    For example, the following reactor uses Node’s built-in path module to extract the base name from a path:

    \n
    target TypeScript;\nmain reactor Preamble {\n  preamble {=\n    import * as path from 'path';\n  =}\n  reaction (startup) {=\n    var filename = path.basename('/Users/Refsnes/demo_path.js');\n    console.log(filename);\n  =}\n}\n
    \n

    This will print:

    \n
    demo_path.js
    \n

    By putting the import in the preamble, the library becomes available in all reactions of this reactor. Oddly, it also becomes available in all subsequently defined reactors in the same file. It’s a bit more complicated to set up Node.js modules from npm that aren’t built-in, but the reaction code to import them is the same as what you see here.

    \n

    You can also use the preamble to define functions that are shared across reactions and reactors:

    \n
    main reactor Preamble {\n  preamble {=\n    function add42( i:number) {\n      return i + 42;\n    }\n  =}\n  timer t;\n  reaction(t) {=\n    let s = "42";\n    let radix = 10;\n    let i = parseInt(s, radix);\n    console.log("Converted string " + s + " to number " + i);\n    console.log("42 plus 42 is " + add42(42));\n  =}\n}\n
    \n

    Not surprisingly, this will print:

    \n
    Converted string 42 to number 42\n42 plus 42 is 84
    \n

    Using Node Modules

    \n

    Installing Node.js modules for TypeScript reactors with npm is essentially the same as installing modules for an ordinary Node.js program. First, write a Lingua Franca program (Foo.lf) and compile it. It may not type check if if you’re importing modules in the preamble and you haven’t installed the modules yet, but compiling your program will cause the TypeScript code generator to produce a project for your program. There should now be a package.json file in the same directory as your .lf file. Open a terminal and navigate to that directory. You can use the standard npm install command to install modules for your TypeScript reactors.

    \n

    The important takeaway here is with the package.json file and the compiled JavaScript in the Foo/dist/ directory, you have a standard Node.js program that executes as such. You can modify and debug it just as you would a Node.js program.

    \n
    \n
    \n

    FIXME: Add $preamble$ example.

    \n
    ","headings":[{"value":"Preamble","depth":2},{"value":"Using Node Modules","depth":3}],"frontmatter":{"permalink":"/docs/handbook/preambles","title":"Preambles","oneline":"Defining preambles in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","permalink":"/docs/handbook/generics"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","permalink":"/docs/handbook/distributed-execution"}}}},"pageContext":{"id":"1-preambles","slug":"/docs/handbook/preambles","repoPath":"/packages/documentation/copy/en/topics/Preambles.md","previousID":"2a0b9619-72b6-5ee7-8a38-83ff8d48005a","nextID":"79d9c9b2-eee4-5652-9541-c483de60119e","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/preambles","result":{"data":{"markdownRemark":{"id":"7f5f499e-179a-51f4-a401-ed9dbd103eb9","excerpt":"$page-showing-target$ Preamble Reactions may contain arbitrary target-language code, but often it is convenient for that code to invoke external libraries or to…","html":"

    $page-showing-target$

    \n

    Preamble

    \n

    Reactions may contain arbitrary target-language code, but often it is convenient for that code to invoke external libraries or to share procedure definitions. For either purpose, a reactor may include a $preamble$ section.

    \n
    \n

    For example, the following reactor uses the math C library for its trigonometric functions:

    \n
    main reactor {\n  preamble {=\n    #include <math.h>\n  =}\n  reaction(startup) {=\n    printf("The cosine of 1 is %f.\\n", cos(1));\n  =}\n}\n
    \n

    This will print:

    \n
    The cosine of 1 is 0.540302.
    \n

    By putting the #include in the $preamble$, the library becomes available in all reactions of this reactor.\nIf you wish to have the library available in all reactors in the same file, you can provide the $preamble$ outside the reactor, as shown here:

    \n
    preamble {=\n  #include <math.h>\n=}\nreactor Cos {\n  reaction(startup) {=\n    printf("The cosine of 1 is %f.\\n", cos(1));\n  =}\n}\nreactor Sin {\n  reaction(startup) {=\n    printf("The sine of 1 is %f.\\n", sin(1));\n  =}\n}\nmain reactor {\n  c = new Cos()\n  s = new Sin()\n}\n
    \n

    You can also use the $preamble$ to define functions that are shared across reactions within a reactor, as in this example:

    \n
    main reactor {\n  preamble {=\n    int add_42(int i) {\n      return i + 42;\n    }\n  =}\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\n
    \n

    Not surprisingly, this will print:

    \n
    42 plus 42 is 84.\n42 plus 1 is 43.
    \n

    (The order in which these are printed is arbitrary because the reactions can execute in parallel.)

    \n

    To share a function across reactors, however, is a bit trickers.\nA $preamble$ that is put outside the $reactor$ definition can only contain\ndeclarations not definitions of functions or variables.\nThe following code, for example will fail to compile:

    \n
    preamble {=\n  int add_42(int i) {\n    return i + 42;\n  }\n=}\nreactor Add_42 {\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n}\nreactor Add_1 {\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\nmain reactor {\n  a = new Add_42()\n  b = new Add_1()\n}\n
    \n

    The compiler will issue a duplicate symbol error because the function definition gets repeated in the separate C files generated for the two reactor classes, Add_42 and Add_1. When the compiled C code gets linked, the linker will find two definitions for the function add_42.

    \n

    To correct this compile error, the file-level preamble should contain only a declaration, not a definition, as here:

    \n
    preamble {=\n  int add_42(int i);\n=}\nreactor Add_42 {\n  reaction(startup) {=\n    printf("42 plus 42 is %d.\\n", add_42(42));\n  =}\n}\nreactor Add_1 {\n  reaction(startup) {=\n    printf("42 plus 1 is %d.\\n", add_42(1));\n  =}\n}\nmain reactor {\n  preamble {=\n    int add_42(int i) {\n      return i + 42;\n    }\n  =}\n  a = new Add_42()\n  b = new Add_1()\n}\n
    \n

    The function definition here is put into the main reactor, but it can be put in any reactor defined in the file.

    \n

    Most header files contain only declarations, and hence can be safely included\nusing #include in a file-level $preamble$. If you wish to use a header file that includes both declarations and definitions, then you will need to include it within each reactor that uses it.

    \n

    If you wish to share variables across reactors, similar constraints apply.\nNote that sharing variables across reactors is strongly discouraged because it can undermine the determinacy of Lingua Franca, and you may have to implement mutual-exclusion locks to access such variables. But it is occassionaly justfiable, as in the following example:

    \n
    preamble {=\n  extern const char shared_string[];\n=}\nreactor A {\n  reaction(startup) {=\n    printf("Reactor A says %s.\\n", shared_string);\n  =}\n}\nreactor B {\n  reaction(startup) {=\n    printf("Reactor B says %s.\\n", shared_string);\n  =}\n}\nmain reactor {\n  preamble {=\n    const char shared_string[] = "Hello";\n  =}\n  a = new A()\n  b = new B()\n}\n
    \n

    Notice the use of the extern keyword in C, which is required because the definition of the shared_string variable will be in a separate (code-generated) C file, the one for main, not the ones for A and B.

    \n

    One subtlety is that if you define symbols that you will use in $input$, $output$, or $state$ declarations, then the symbols must be defined in a file-level $preamble$.\nSpecifically, the following code will fail to compile:

    \n
    main reactor {\n  preamble {=\n    typedef int foo;\n  =}\n  state x:foo = 0\n  reaction(startup) {=\n    lf_print("State is %d", self->x);\n  =}\n}\n
    \n

    The compiler will issue an unknown type name error. To correct this, just move the declaration to a file-level $preamble$:

    \n
    preamble {=\n  typedef int foo;\n=}\nmain reactor {\n  state x:foo = 0\n  reaction(startup) {=\n    lf_print("State is %d", self->x);\n  =}\n}\n
    \n
    \n
    \n

    For example, the following reactor uses the charconv header from the c++ standard library to convert a string to an integer:

    \n
    target Cpp;\n\nmain reactor {\n  private preamble {=\n    #include <charconv>\n    #include <string>\n  =}\n\n  timer t;\n  reaction(t) {=\n    std::string raw = "42";\n    std::size_t number;\n\n    auto result = std::from_chars(raw.data(), raw.data() + raw.size(), number);\n    if (result.ec == std::errc::invalid_argument) {\n      std::cerr << "Could not convert.";\n    } else {\n      std::cout << "Converted string: " << raw << " to integer: " << number << std::endl;\n    }\n  =}\n}\n
    \n

    This will print:

    \n
    [INFO]  Starting the execution\nConverted string: 42 to integer: 42\n[INFO]  Terminating the execution
    \n

    By putting the #include in the preamble, the library becomes available in all reactions of this reactor. Note the private qualifier before the preamble keyword.\nThis ensures that the preamble is only visible to the reactions defined in this reactor and not to any other reactors. In contrast,\nthe public qualifier ensures that the preamble is also visible to other reactors in files that import the reactor defining the public preamble.

    \n
    reactor Preamble {\n  public preamble {=\n    struct MyStruct {\n      int foo;\n      std::string bar;\n    };\n  =}\n\n  private preamble {=\n    auto add_42(int i) noexcept -> int {\n      return i + 42;\n    }\n  =}\n\n  logical action a:MyStruct;\n\n  reaction(startup) {=\n    a.schedule({add_42(42), "baz"});\n  =}\n\n  reaction(a) {=\n    auto value = *a.get();\n    std::cout << "Received " << value.foo << " and '" << value.bar << "'\\n";\n  =}\n}\n
    \n

    It defines both a public and a private preamble. The public preamble defines the type MyStruct. This type definition will be visible to all elements of the\nPreamble reactor as well as to all reactors defined in files that import Preamble. The private preamble defines the function add_42(int i).\nThis function will only be usable to reactions within the Preamble reactor.

    \n

    You can think of public and private preambles as the equivalent of header files and source files in C++. In fact, the public preamble will be translated to a\nheader file and the private preamble to a source file. As a rule of thumb, all types that are used in port or action definitions as well as in state variables\nor parameters should be defined in a public preamble. Also, declarations of functions to be shared across reactors should be placed in the public preamble.\nEverything else, like function definitions or types that are used only within reactions, should be placed in a private preamble.

    \n

    Note that preambles can also be specified on the file level. These file level preambles are visible to all reactors within the file.\nAn example of this can be found in PreambleFile.lf.

    \n

    Admittedly, the precise interactions of preambles and imports can become confusing. The preamble mechanism will likely be refined in future revisions.

    \n

    Note that functions defined in the preamble cannot access members such as state variables of the reactor unless they are explicitly passed as arguments.\nIf access to the inner state of a reactor is required, methods present a viable and easy to use alternative.

    \n
    \n
    \n

    For example, the following reactor uses the platform module to print the platform information and a defined method to add 42 to an integer:

    \n
    main reactor Preamble {\n\tpreamble {=\n\t\timport platform\n\t\tdef add_42(self, i):\n\t\t\treturn i + 42\n\t=}\n\ttimer t;\n\treaction(t) {=\n\t\ts = "42"\n\t\ti = int(s)\n\t\tprint("Converted string {:s} to int {:d}.".format(s, i))\n\t\tprint("42 plus 42 is ", self.add_42(42))\n\t\tprint("Your platform is ", self.platform.system())\n\t=}\n}\n
    \n

    On a Linux machine, this will print:

    \n
    Converted string 42 to int 42.\n42 plus 42 is 84\nYour platform is Linux
    \n

    By putting import in the $preamble$, the module becomes available in all reactions of this reactor using the self modifier.

    \n

    Note: Preambles will be put in the generated Python class for the given reactor, and thus is part of the instance of the reactor. This means that anything you put in the preamble will be specific to a particular reactor instance and cannot be used to share information between different instantiations of the reactor (this is a feature, not a bug, because it helps ensure determinacy). For more information about implementation details of the Python target, see Implementation Details.

    \n

    Alternatively, you can define a $preamble$ outside any reactor definition. Such a $preamble$ can be used for functions such as import or to define a global function. The following example shows importing the hello module:

    \n
    target Python {\n  files: include/hello.py\n};\npreamble {=\n  import hello\n=}\n
    \n

    Notice the usage of the files target property to move the hello.py module located in the include folder of the test directory into the working directory (located in src-gen/NAME).

    \n

    For another example, the following program uses the built-in Python input() function to get typed input from the user:

    \n
    target Python\nmain reactor {\n  preamble {=\n    import threading\n    def external(self, a):\n      while (True):\n        from_user = input() # Blocking\n        a.schedule(0, from_user)\n  =}\n  state thread\n  physical action a\n  timer t(2 secs, 2 secs)\n\n  reaction(startup) -> a {=\n    self.thread = self.threading.Thread(target=self.external, args=(a,))\n    self.thread.start()\n    print("Type something.")\n  =}\n\n  reaction(a) {=\n    elapsed_time = lf.time.logical_elapsed()\n    print(f"A time {elapsed_time} nsec after start, received: ", a.value)\n  =}\n\n  reaction(t) {=\n    print("Waiting ...")\n  =}\n}\n
    \n

    Within the $preamble$, we specify to import the threading Python module and define a function that will be started in a separate thread in the reaction to $startup$. The thread function named external blocks when input() is called until the user types something and hits the return or enter key. Usually, you do not want a Lingua Franca program to block waiting for input. In the above reactor, a $timer$ is used to repeatedly trigger a reaction that reminds the user that it is waiting for input.

    \n
    \n
    \n

    For example, the following reactor uses Node’s built-in path module to extract the base name from a path:

    \n
    target TypeScript;\nmain reactor Preamble {\n  preamble {=\n    import * as path from 'path';\n  =}\n  reaction (startup) {=\n    var filename = path.basename('/Users/Refsnes/demo_path.js');\n    console.log(filename);\n  =}\n}\n
    \n

    This will print:

    \n
    demo_path.js
    \n

    By putting the import in the preamble, the library becomes available in all reactions of this reactor. Oddly, it also becomes available in all subsequently defined reactors in the same file. It’s a bit more complicated to set up Node.js modules from npm that aren’t built-in, but the reaction code to import them is the same as what you see here.

    \n

    You can also use the preamble to define functions that are shared across reactions and reactors:

    \n
    main reactor Preamble {\n  preamble {=\n    function add42( i:number) {\n      return i + 42;\n    }\n  =}\n  timer t;\n  reaction(t) {=\n    let s = "42";\n    let radix = 10;\n    let i = parseInt(s, radix);\n    console.log("Converted string " + s + " to number " + i);\n    console.log("42 plus 42 is " + add42(42));\n  =}\n}\n
    \n

    Not surprisingly, this will print:

    \n
    Converted string 42 to number 42\n42 plus 42 is 84
    \n

    Using Node Modules

    \n

    Installing Node.js modules for TypeScript reactors with npm is essentially the same as installing modules for an ordinary Node.js program. First, write a Lingua Franca program (Foo.lf) and compile it. It may not type check if if you’re importing modules in the preamble and you haven’t installed the modules yet, but compiling your program will cause the TypeScript code generator to produce a project for your program. There should now be a package.json file in the same directory as your .lf file. Open a terminal and navigate to that directory. You can use the standard npm install command to install modules for your TypeScript reactors.

    \n

    The important takeaway here is with the package.json file and the compiled JavaScript in the Foo/dist/ directory, you have a standard Node.js program that executes as such. You can modify and debug it just as you would a Node.js program.

    \n
    \n
    \n

    FIXME: Add $preamble$ example.

    \n
    ","headings":[{"value":"Preamble","depth":2},{"value":"Using Node Modules","depth":3}],"frontmatter":{"permalink":"/docs/handbook/preambles","title":"Preambles","oneline":"Defining preambles in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Generic Reactors","oneline":"Defining generic reactors in Lingua Franca.","permalink":"/docs/handbook/generics"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","permalink":"/docs/handbook/distributed-execution"}}}},"pageContext":{"id":"1-preambles","slug":"/docs/handbook/preambles","repoPath":"/packages/documentation/copy/en/topics/Preambles.md","previousID":"2a0b9619-72b6-5ee7-8a38-83ff8d48005a","nextID":"79d9c9b2-eee4-5652-9541-c483de60119e","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/proof-import/page-data.json b/page-data/docs/handbook/proof-import/page-data.json index 3540c122e..f75aa4d2b 100644 --- a/page-data/docs/handbook/proof-import/page-data.json +++ b/page-data/docs/handbook/proof-import/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/proof-import","result":{"data":{"markdownRemark":{"id":"4466937a-36f5-5fc8-a89d-98421824f135","excerpt":"This is a collection of thoughts on the design of a reliable package and import system that is ready for future applications. At this stage, this page mostly…","html":"

    This is a collection of thoughts on the design of a reliable package and import system that is ready for future applications. At this stage, this page mostly represents my personal view (Christian Menard). I will also focus on the C++ target here as this is the target I know best. The C target is not a good example for these considerations as there is a fundamental design issue with the C target. Since the code generator places all code in a single generated .c file and does things like #include reactor.c to avoid the need for Makefiles, it circumvents many of the issues that come with imports that I will outline here. It simply ignores file scopes and namespaces altogether.

    \n

    The status quo

    \n

    The current import system is lean and simple. Write import Bar.lf in Foo.lf and every reactor defined in Bar.lf will be visible in the file scope Foo.lf. Bar.lf is looked up simply by scanning the directory Foo.lf is placed in. This works well for the simple programs and tests we have right now, but does not scale. I identify the following problems:

    \n
      \n
    1. \n

      There is no notion of separate namespaces. Every reactor that Bar.lf defines becomes visible in Foo.lf. If both files define a Reactor Foo, there is a name clash and the import would be ill-formed. There should be a mechanism to distinguish the two definitions of Foo, such as using fully qualified names: Foo.Foo and Bar.Foo.

      \n
    2. \n
    3. \n

      There is no concept for importing files from a directory structure. It is unclear how Foo.lf could import my/lib/Bar.lf.

      \n
    4. \n
    5. \n

      There is no concept for packages or libraries that can be installed on the system. How could we import Reactors from a library that someone else provided?

      \n
    6. \n
    \n

    These are the more obvious issues that we have talked about. However, there are more subtle ones that we haven’t been discussed in depth (or at least not in the context of the import system design discussion). The open question is: What does importing a LF file actually mean? Obviously, an import should bring Reactors defined in another files into local scope. But what should happen with the other structures that are part of an LF file, namely target properties and preambles? That is not specified and our targets use a best practice approach. But this is far away from a good design that is scalable and future proof.

    \n

    A quick dive into the C++ code generator

    \n

    Before I discuss the problems with preambles and target properties, I would like to give you a quick overview of how the C++ code generator works. Consider the following LF program consisting of two files Foo.lf and Bar.lf:

    \n
    // Bar.lf\n\nreactor Bar {\n reaction(startup) {=\n   // do something bar like\n =}\n}
    \n
    // Foo.lf\n\nimport Bar.lf\n\nreactor Foo {\n bar = new Bar();\n reaction(startup) {=\n   // do something foo like\n =}\n}
    \n

    Now let us have a look on what the C++ code generator does. It will produce a file structure like this:

    \n
    CMakeLists.txt\nmain.cc\nBar/\n  Bar.cc\n  Bar.hh\nFoo/\n  Foo.cc\n  Foo.hh
    \n

    We can ignore CMakeLists.txt and main.cc for our discussion here. The former specifies how the whole program can be build and the latter contains the main() function and some code that is required to get the application up and running. For each processed <file>.lf file, the code generator creates a directory <file>. For each reactor <reactor> defined in <file>.lf, it will create <file>/<reactor>.cc and <file>/<reactor>.hh. The header file declares a class representing the reactor like this:

    \n
    // Bar/Bar.hh\n\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n\nclass Bar : public reactor::Reacor {\n private:\n  // default actions\n  reactor::StartupAction startup {"startup", this};\n  reactor::ShutdownAction shutdown {"shutdown", this};\n\n public:\n  /* ... */\n\n private:\n  // reaction bodies\n  r0_body();\n};
    \n

    The corresponding Bar/Bar.cc will look something like this:

    \n
    #include "Bar/Bar.hh"\n\n/* ... */\n\nBar::r0_body() {\n  // do something bar like\n}
    \n

    Similarly, Foo.hh and Foo.cc will be generated. However, since Foo.lf imports Bar.lf and instantiated the reactor Bar it must be made visible. This is done by an include directive in the generated code like so:

    \n
    // Foo/Foo.hh\n###\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n#include "Bar/Bar.hh"\n\nclass Foo : public reactor::Reacor {\n private:\n  // default actions\n  reactor::StartupAction startup;\n  reactor::ShutdownAction shutdown;\n\n  // reactor instances\n  Bar bar;\n public:\n  /* ... */\n\n private:\n  // reaction bodies\n  r0_body();\n};
    \n

    The problem with preambles

    \n

    The problems with preamble in the context of imports were already discussed in a related issue, but I would like to summarize the problem here. While the examples above worked nicely even with imports, things get messy as soon as we introduce a preamble. Let’s try this:

    \n
    // Bar.lf\n\nreactor Bar {\n preamble {=\n   struct bar_t {\n     int x;\n     std::string y;\n   };\n\n   bar_t bar_func {\n     return bar_t(42, "hello")\n   }\n =}\n output out:bar_t;\n reaction(startup) -> out {=\n   out.set(bar_fuc());\n =}\n}
    \n
    // Foo.lf\n\nimport Bar.lf\n\nreactor Foo\n bar = new Bar();\n reaction(bar.out) {=\n   auto& value = bar.out.get();\n   std::cout << "Received {" << value->x << ", " << value->y << "}\\n";\n =}\n}
    \n

    This would be expected to print Received {32, hello}. However, before we can even compile this program, we need to talk about what should happen with the preamble during code generation and how the import affects it. So where should the preamble go? The first thing that comes to mind, is to embed it in the header file Bar.hh something like this:

    \n
    // Bar/Bar.hh\n\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n\n// preamble\nstruct bar_t {\n  int x;\n  std::string y;\n};\n\nbar_t bar_func {\n  return bar_t(42, "hello")\n}\n\nclass Bar : public reactor::Reacor {\n /* ... */\n};
    \n

    If we embed the preamble like this and compile the program ,then the compiler is actually happy and processes all *.cc files without any complaints. But, there is a huge problem while linking the binary. The linker sees multiple definitions of bar_func and has no idea which one to use. Why is that? Well, the definition of bar_func is contained in a header file. This should never be done in C/C++! Since includes translate to a plain-text replacement by the preprocessor, Bar.cc will contain the full definition of bar_func. As Foo.cc imports Foo.hh which imports Bar.hh, also Foo.cc will contain the full definition. And since main.cc also has to include Foo.hh, main.cc will also contain the full definition of bar_func. So we have multiple definitions of the same function and the linker rightfully reports this as an error.

    \n

    So what should we do? We could place the preamble in Bar.cc instead. This ensures that only Bar.cc sees the definition of bar_func. But then the compiler complains. Neither Bar.hh nor Foo.hh see type declaration of bar_t. Note that there is a dependency of Foo.lf on the preamble in Bar.lf. The import system should somehow take care of this dependency! Also note that this has not appeared as a problem in C as the code generator places everything in the same compilation unit. Foo will see the preamble of Bar as long as Foo is generated before Bar.

    \n

    But how to solve it for C++ where the code is split in multiple compilation units (which really should be happening in C as well)? What we do at the moment is annotating the preamble with private and public keywords. This helps to split the preamble up and decide what to place in the header and what to place in the source file. For instance:

    \n
    // Bar.lf\n\nreactor Bar {\n public preamble {=\n   struct bar_t {\n     int x;\n     std::string y;\n   };\n =}\n private preamble {=\n   bar_t bar_func {\n     return bar_t(42, "hello")\n   }\n =}\n output out:bar_t;\n reaction(startup) -> out {=\n   out.set(bar_fuc());\n =}\n}
    \n

    This makes the type bar_t visible as part of the public interface of Bar. Both the code generated for Bar and the code generated for Foo will see the definition of bar_t. This is realized by placing the public preamble in Bar.hh The function bar_func is part of Bar’s private interface. It is only visible with the reactor definition of Bar and is not propagated by an import. This is realized by simply placing the private preamble in Bar.cc. This makes the compiler finally happy and when get an executable program private and public preambles provide a mechanism to define what is propagated on an import and what is not. I think this is an important distinction even in languages other than C/C++ that do not have this weird separation of source and header file.

    \n

    I am sorry for this lengthy diversion into things that happened in the past where we actually want to talk about how things should work in the future. However, understanding this issue is important and when talking about other solutions we should not forget that it exists.

    \n

    The problem with target properties

    \n

    It is also not well-defined what should happen with target properties when importing a .lf file. Apparently the common practice is simply ignoring the existence of other target declarations and only considering the target declaration of the .lf that contains the main reactor. I think this works reasonably well for our small programs. But it will cause problems when either programs become larger or we introduce new target properties where it is unclear what piece of code they reference. Let us have a look at the existing target properties for C++. How should those different properties be handled on an import? Which scope do they actually apply to? We haven’t really talked about this.

    \n

    fast, keepalive, threads and timeout are easy. They apply to the main reactor. Since we do not import main reactors from other files, it is clear that we really want to use the properties defined in the main compilation unit. So our current strategy works in this case. Although there are still some subtleties. For instance, if a library file defines keepalive=true and fast=false because it uses physical actions, should any file importing this library file be allowed to override these properties. Probably not, because it doesn’t make sense if physical actions are involved. But a careless user of the library might not be aware of that. So maybe it isn’t that clear after all.

    \n

    build-type, cmake-include, compile, logging and no-runtime-validation influence how the application is build. They are used for generating the CMakeLists.txt file. So their is quite clear: they apply to the whole compilation of the given application. Again it is a simple solution to only consider the target properties of the file containing the main reactor since this can be considered the file that ‘drives’ the compilation. But what if an imported .lf relies on an external library and uses the cmake-include property to tell CMake to look this library up, make the library header files visible and link our generated code to that library (fortunately this can be done with 2 lines in CMake). Should this target property really be ignored by our import? Probably not, because it will lead to compile errors if the author of the main .lf file does not configure cmake-include properly. So there should be some kind of merging mechanism for cmake-include. Should this be done for the other properties as well? I am not sure and I actually don’t know how the merging would work.

    \n

    So this raises a lot of questions that we currently have no answer to. I believe we need to find answers for these questions in order to create a well working import and package system. This gets only more complicated when we add more properties such as the proposed files directive. We should really consider what properties actually apply to and if they influence the way imports work.

    \n

    The work in progress

    \n

    To be continued… I want to describe here what is happening on the new_import and the (potential) problems this brings.

    \n

    Possible solutions

    \n

    To be continued… I would like to show a few possible solutions that have come to mind and that we discussed already.

    \n

    Concrete proposal

    \n

    With the risk of overlooking some of the issues discussed above, I’d like to outline a concrete proposal. To me, at least, it is easier to reason about these issues in a context with a few more constraints. Hopefully, this can serve as a starting point that we can tweak/adjust as needed. Note: this proposal borrows from the previous proposal written by Soroush. Based on my experience with Xtext, I have confidence that what is described below is feasible to implement.

    \n

    Import/export

    \n
      \n
    1. One LF file can contain multiple reactor definitions.
    2. \n
    3. There can be at most one main reactor per file.
    4. \n
    5. Any reactor class defined outside of the current file has to be imported explicitly.
    6. \n
    7. The visibility of a reactor class can be limited using a modifier in the class definition.
    8. \n
    \n
      \n
    • Should the default visibility be public or private? I have no strong preference either way.
    • \n
    \n
      \n
    1. An import statement must specify which reactor classes to import. This is necessary because if we populate a global scope using the uriImport mechanism, the local scope provider needs to know which definition to link to if there happen to exist multiple among the set of included files. We could potentially relax this constraint and only report the situation where we know for a fact that there is ambiguity and needs to be resolved by making the imports explicit. We could also deprecate the use of unqualified imports (the original syntax), therefore allow it but warn that it might not work as expected.
    2. \n
    3. An LF file in an import statement is specified by a path relative to the directory of the file in which the import statement occurs or relative to a series of directories in a wider search path.
    4. \n
    \n
      \n
    • Eclipse uses .project files to identify the root of a project; we can look for that.
    • \n
    • We can look for our own kind of manifest files as well. These can list additional locations to search. This is compatible with the idea of developing a package system. I personally like this approach better than using an environment variable.
    • \n
    \n
      \n
    1. I personally find fully qualified names excess generality and have never felt compelled to use them in Java. IMO, they lead to code that’s difficult to read and a pain to format. To keep things simple, I suggest we don’t support them. Instead, we should provide a mechanism for renaming imported reactor classes to avoid naming conflicts.
    2. \n
    3. Open question: do we want scope modifiers for imports? It seems that extra import statements could be used to increase visibility, so it might not be needed.
    4. \n
    \n

    Syntax

    \n
    Import := 'import' <ID> <Rename>? (',' <ID> <Rename>?)* 'from' <PATH>\n\nRename := 'as' <ID>
    \n

    Note: This syntax could be extended to support packages in addition to paths. But it doesn’t make much sense to have this until we have a package manager and package registry.

    \n

    Current state of the discussion: one unifying syntax vs. different syntax for references to files and packages.

    \n

    Preambles

    \n

    A preamble allows for the inclusion of verbatim target code that may be necessary for reactors to function. Currently, there are two scopes in which preambles can appear: (1) file scope and (2) reactor class scope. Moreover, there exist visibility modifiers to label preambles private or public. A public preamble is intended to contain code that is necessary for the use of a reactor that is in scope. A private preamble is intended to contain code that is necessary for the implementation of a reactor that is in scope. Only the C++ code generator can currently effectively separate these LF scope levels. It achieves this by putting each reactor class definition in its own file. LF file scope preambles are currently not supported by the C target, but this appears to be unintentional and would be easy to fix. Reactor class scope preambles are supported by the C target, but there is no isolation of scope; the preamble of one reactor is visible to the one defined after it. To fix this, I see two options: (1) follow the same approach as CppGenerator and output separate files, which also means that a Makefile has to be generated in order to compile the result, or (2) leverage block scope within a single file, but this will become complicated and make the generated C code even less humanly readable.

    \n

    We could put aside the problem of name clashes due to the absence of scope isolation in generated C code and fix this later. For the time being, the problem can be circumvented using .h files.

    \n

    Target Properties

    \n
      \n
    1. Each file declares a target.
    2. \n
    3. All code in all reactors in the same file must agree with the specified target.
    4. \n
    5. Additional target properties may be specified.
    6. \n
    7. Target properties are not inherited through imports.
    8. \n
    9. Any property that needs to be inherited through an import (such as the requirement to link against the pthread library) must be specified as a build dependency instead.
    10. \n
    \n

    Build Dependencies

    \n
      \n
    1. It must be possible to specify build dependencies, such as files, sources, and protobufs.
    2. \n
    3. We could either allow these definitions to go directly in the .lf file, or we could decide to specify them in a package description (i.e., separate file). We could potentially allow both.
    4. \n
    5. Build dependencies are inherited through imports (or from package descriptions), and they are never shadowed, always joined.
    6. \n
    \n

    Unique Reactor Names

    \n

    The new import system as described above ensures that reactor names within a single .lf file are unique. In case reactors with the same name are imported from different .lf files, the renaming mechanism needs to be used in order to resolve the name conflict. The same applies if the given .lf file defines some reactors and tries to import other reactors with the same name. For instance, consider the LF file in Example 1 below. In the scope of this file, three reactor declarations are visible: Foo, Bar and Baz, although the actual reactors have the same name Foo.

    \n

    Examples

    \n

    Throughout this section, I will be using two LF example programs. Since the markdown syntax does not provide a simple way to label and refer to code listings, I figure its easiest to place them here in a central place and refer to them later by the heading

    \n

    Example 1

    \n
    \\\\ Foo.lf\nimport Foo as Bar from "Bar.lf"\nimport Foo as Baz from "Baz.lf"\n\nreactor Foo {\n  \\\\ ...\n}
    \n

    Example 2

    \n
    \\\\ Baz.lf\nreactor Foo {\n  \\\\ ...\n}
    \n
    \\\\ Bar.lf\nimport Foo from "Baz.lf"\n\nreactor Foo {\n  foo = new Foo()\n  \\\\ ...\n}
    \n
    \\\\ Foo.lf\nimport Bar from "Bar.lf"\nmain reactor Foo {\n  bar = new Bar()\n  \\\\ ...\n}
    \n

    Unique Reactor Names in Target Code

    \n

    While the mechanism above effectively ensures uniqueness in a single LF file, this uniqueness is surprisingly hard to ensure in generated target code. C has an obvious problem here as it places all generated code in a single file. While the name conflict in the above code can be solved by generating code for three reactors named Bar, Baz and Foo, it breaks as soon as another file of the same LF program uses import Foo from \"Bar.lf\". Then there would be two definitions of the reactor Foo that cannot be resolved.

    \n

    Now you would probably expect that splitting the generated code into multiple files solves the issue, but unfortunately this is not true. If anything, it makes the problem more subtle. The C++ code generated from Example 1 would likely look something like this:

    \n
    // Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n#include "Baz.hh"\n\n// do the renaming\nusing Bar = Foo;\nusing Baz = Foo;\n\nclass Foo : public reactor::Reactor {\n};
    \n

    This will cause a compile error as there are multiple definitions of Foo. While renaming is possible in C++ with the using keyword (typedef works as well), the thing being renamed needs to be already visible in the scope. So there are multiple definitions of Foo as all the files Bar.hh, Baz.hh and Foo.hh define this class. We need a mechanism to distinguish the different definitions of Foo.

    \n

    There is even another issue that stems from the fact that the semantics of imports in LF are different from the include semantics of C++. Consider the code in Example 2, which is valid LF code. Although Bar.lf imports Foo and Foo.lf imports from Bar.lf, the definition of Foo in Baz.lf is not visible in Foo.lf. This ‘hiding’, however, does not easily propagate to the generated code. In C, there will be an error because both definitions of Foo are placed in the same file. In C++, the different definitions of Foo are placed in different files, but there will still be an error. The generated C++ code would look something like this:

    \n
    \\\\ Baz.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\nclass Foo : public reactor::Reactor {\n  // ...\n};
    \n
    \\\\ Bar.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Baz.hh"\n\nclass Bar : public reactor::Reactor {\n  Foo foo;\n  // ...\n};
    \n
    \\\\ Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n\nclass Foo : public reactor::Reactor {\n  Bar bar;\n  // ...\n};
    \n

    This will produce an error due to multiple definitions of Foo being visible in Foo.hh. The problem is that any include in Bar.hh becomes also visible in Foo.hh. So there is a name clash due to the way the C++ compiler processes included and that is hard to work around.

    \n

    Possible Solutions

    \n

    In conclusion from the above section, I can say that translating the file based scoping of reactor names that we have in LF to generated target code is not trivial. Any sensible solution will need to establish a mechanism to ensure that any two distinct reactors in LF are also distinct in target code.

    \n

    Namespaces

    \n

    We could introduce some form of a namespace mechanism that allows us to derive fully-qualified names of reactors. This is the preferred solution for me (Christian). Note that by ‘namespace’ I mean any logical organization of reactors in named groups and not the precise concept of C++ namespaces. In other languages those logical groups are also referred to as modules or packages. Also note that it is only important to be able to assign a fully-qualified name to a reactor, it does not necessarily require that we refer to reactors by their fully-qualified name in LF code.

    \n

    File based namespaces

    \n

    In my view, the easiest way to introduce namespaces in LF would be to leverage file system structure. Everything contained in Foo.lf would automatically be in the namespace Foo. So the FQN of a reactor Foo defined in Foo.lf would be Foo.Foo (or Foo::Foo, or some other delimiter). This would solve the name clashes in both of our examples. For Example 1, the generated code could look like this:

    \n
    // Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n#include "Baz.hh"\nnamespace Foo {\n\n// do the renaming\nusing Bar = Bar::Foo;\nusing Baz = Baz::Foo;\n\nclass Foo : public reactor::Reactor {\n};\n}
    \n

    For Example 2, the generated code could look like this:

    \n
    \\\\ Baz.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\nnamespace Baz {\nclass Foo : public reactor::Reactor {\n  // ...\n};\n}
    \n
    \\\\ Bar.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Baz.hh"\n\nnamespace Bar {\nusing Foo = Baz::Foo; // bring Foo in scope\n\nclass Bar : public reactor::Reactor {\n  Foo foo;\n  // ...\n};\n}
    \n
    \\\\ Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n\nnamespace Foo {\nusing Bar = Bar::Bar; // bring Bar in scope\n\nclass Foo : public reactor::Reactor {\n  Bar bar;\n  // ...\n};\n}
    \n

    While this appears to be a promising solution, it is not sufficient to only consider the name of an *.lf file to derive the namespace\nThere could be two files Foo.lf in different directories that both define the reactor Foo. Thus, we also need to consider the directory structure and introduce hierarchical namespaces. Consider this directory tree:

    \n
    foo/\n  bar/\n    foo.lf  # defines reactor Foo\n  baz/\n    foo.lf  # defines reactor Foo
    \n

    In this example, the two Foo reactors would have the fully qualified names foo.bar.foo.Foo and foo.baz.foo.Foo.\nIn order for this concept to work, we need the notion of a top-level namespace or directory. Naturally, this would be the package. Therefore, this namespace approach would also require a simple mechanism to define a package. For now this could be rudimentary. Simply placing an empty lf.yaml in the foo/ directory in the above example would be sufficient. In addition to the notion of packages, we would also need a simple mechanism to find packages. However, since packages are something we want to have anyway, it would not hurt to start and implement a rudimentary system now.

    \n

    This proposal is a bit at odds with the file based import mechanism described above. While it is clear what the namespace of an *.lf file relative to a package directory is, it is unclear what the namespace of an arbitrary file outside a package is. Marten suggested to resolve this by using a default namespace or the global namespace whenever a *lf that is not part of a package is imported and to make the user responsible for avoiding any conflicts.

    \n

    We would also need to restrict the naming of files and directories and ban the usage of the namespace delimiter (. or :: or some other) in file and directory names. In my opinion this is not much of a problem and common practice for many languages. If we decide to use this namespace mechanism, it would probably be better to drop the file based imports and switch to imports by FQN (e.g. import Foo from foo.bar.foo)

    \n

    A Namespace Directive

    \n

    As an alternative to the file based namespace mechanism described above, we could also introduce a namespace directive in the LF syntax or as part of the target properties. This would effectively allow the user to specify the namespace that any reactor defined in a file should be part of. This solution would allow to augment the file based import system that we have with a namespace mechanism. It is important to note, however, that this entirely shifts the responsibility for ensuring uniqueness within a namespace to the user. When we derive namespaces from the file path as described above, we can be sure that the resulting namespace only contains unique reactors because we ensure that any LF file only contains unique reactors. If we allow the user to specify the namespace, however, there could easily be two files with the same namespace directive that both define the reactor Foo. This approach might also cause problems for target languages where the namespaces relate to concrete file paths such as in Rust, Python or Java.

    \n

    Name Mangling

    \n

    There are other mechanisms to derive unique names apart from namespaces. One that is widely used by compilers is name mangling which replaces or decorates the original name. For instance, we could simply add a number to the name of generated reactors (Foo1, Foo2, …) to distinguish multiple LF reactor definitions named Foo. What separates our approach from traditional compiler though, is that we are not in control of the full build process and only generate source code to be processed by another compiler. Therefore, any renaming we do when compiling LF code to target code needs to be done with care as it could easily introduce new problems because we are not aware of all the identifiers defined in a target language. For instance if our LF program uses a library that defines the class Foo3, adding a third definition of the reactor Foo to the program would lead to an unexpected error that is also hard to debug.

    \n

    Soroush also proposed to use a hashing mechanism (for instance a hash of the file name) to decorate reactor names. This would be less likely\nto clash with any names defined in some library. However, we would need to make sure that any mechanism we use for generating unique decorated names follows strict rules and generates reproducible names. This reproducibility is crucial for several reasons.

    \n
      \n
    1. \n

      Since even a complex name mangling mechanism has still the chance to produce name clashes with identifiers defined outside of the LF program, those clashes should not occur randomly. There should be either an error or no error on each compilation run. Nondeterministic builds are no fun to deal with.

      \n
    2. \n
    3. \n

      In case of any errors, it is crucial to be able to reproduce and compare builds across machines and platforms. A platform dependent name mangling algorithm (for instance one that hashes file paths) would make it unnecessary hard to reproduce and debug compile errors.

      \n
    4. \n
    5. \n

      Somewhere in the future, we might want to be able to compile packages as libraries. Recompilation of the library should never change its API. Moreover, the name mangling algorithm should be robust in the sense that small changes in LF code do not lead to changed identifiers\nin the library API.

      \n
    6. \n
    \n

    All in all, I think it is hard to define an algorithm that generates reproducible and stable names, but maybe someone else has a good idea of how this could be achieved.

    \n

    Another obvious disadvantage of the name mangling approach would be that the generated code is less readable. Also any external target code that might want to reference reactors in a library compiled from LF code, would need to know and use the mangled name.

    \n

    Unique Reactor Names in our Tools

    \n

    In our last meeting (Tue 2020-08-04), I said that there are other places where we care about unique names: our tools such as the diagram view or the trace generator that I implemented for C++ and that we cannot ensure that names are unique at the moment. However, while thinking about it a bit more I realized that this is not much of an issue. Ambiguous names of reactor types are not a big problem for the diagram view. Since clicking on the nodes jumps directly to the reactor definition, the ambiguity in the names can easily be resolved.

    \n

    For the tracing, I realized that it is not the name of the reactor type that matters, but the name of the instance. These are unique fully-qualified names already. For instance main.foo.bar.r0, denotes the reaction with priority 0, of a reactor instance called bar that is contained by the reactor instance foo, which is in turn contained by the main reactor.

    \n

    Summary

    \n

    All in all, I think leveraging the file structure for determining the fully qualified names of reactors is the most promising solution.

    \n
      \n
    1. It works without any changes in our syntax. Only the code generators need to be updated to support the namespacing.
    2. \n
    3. In contrast to name mangling, it allows generation of readable code and also gives the programmer full control of how generated reactors are named.
    4. \n
    5. It fits naturally to languages that also support leveraging the file structure to create namespaces (e.g. python or rust).
    6. \n
    ","headings":[{"value":"The status quo","depth":1},{"value":"A quick dive into the C++ code generator","depth":2},{"value":"The problem with preambles","depth":2},{"value":"The problem with target properties","depth":2},{"value":"The work in progress","depth":3},{"value":"Possible solutions","depth":3},{"value":"Concrete proposal","depth":2},{"value":"Import/export","depth":3},{"value":"Syntax","depth":3},{"value":"Preambles","depth":2},{"value":"Target Properties","depth":2},{"value":"Build Dependencies","depth":2},{"value":"Unique Reactor Names","depth":1},{"value":"Examples","depth":2},{"value":"Example 1","depth":3},{"value":"Example 2","depth":3},{"value":"Unique Reactor Names in Target Code","depth":2},{"value":"Possible Solutions","depth":2},{"value":"Namespaces","depth":3},{"value":"File based namespaces","depth":4},{"value":"A Namespace Directive","depth":4},{"value":"Name Mangling","depth":3},{"value":"Unique Reactor Names in our Tools","depth":2},{"value":"Summary","depth":2}],"frontmatter":{"permalink":"/docs/handbook/proof-import","title":"Future Proof Package/Import System","oneline":"A future proof package and import system","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/proof-import","repoPath":"/packages/documentation/copy/en/less-developed/Future Proof Package and Import System.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/proof-import","result":{"data":{"markdownRemark":{"id":"4466937a-36f5-5fc8-a89d-98421824f135","excerpt":"This is a collection of thoughts on the design of a reliable package and import system that is ready for future applications. At this stage, this page mostly…","html":"

    This is a collection of thoughts on the design of a reliable package and import system that is ready for future applications. At this stage, this page mostly represents my personal view (Christian Menard). I will also focus on the C++ target here as this is the target I know best. The C target is not a good example for these considerations as there is a fundamental design issue with the C target. Since the code generator places all code in a single generated .c file and does things like #include reactor.c to avoid the need for Makefiles, it circumvents many of the issues that come with imports that I will outline here. It simply ignores file scopes and namespaces altogether.

    \n

    The status quo

    \n

    The current import system is lean and simple. Write import Bar.lf in Foo.lf and every reactor defined in Bar.lf will be visible in the file scope Foo.lf. Bar.lf is looked up simply by scanning the directory Foo.lf is placed in. This works well for the simple programs and tests we have right now, but does not scale. I identify the following problems:

    \n
      \n
    1. \n

      There is no notion of separate namespaces. Every reactor that Bar.lf defines becomes visible in Foo.lf. If both files define a Reactor Foo, there is a name clash and the import would be ill-formed. There should be a mechanism to distinguish the two definitions of Foo, such as using fully qualified names: Foo.Foo and Bar.Foo.

      \n
    2. \n
    3. \n

      There is no concept for importing files from a directory structure. It is unclear how Foo.lf could import my/lib/Bar.lf.

      \n
    4. \n
    5. \n

      There is no concept for packages or libraries that can be installed on the system. How could we import Reactors from a library that someone else provided?

      \n
    6. \n
    \n

    These are the more obvious issues that we have talked about. However, there are more subtle ones that we haven’t been discussed in depth (or at least not in the context of the import system design discussion). The open question is: What does importing a LF file actually mean? Obviously, an import should bring Reactors defined in another files into local scope. But what should happen with the other structures that are part of an LF file, namely target properties and preambles? That is not specified and our targets use a best practice approach. But this is far away from a good design that is scalable and future proof.

    \n

    A quick dive into the C++ code generator

    \n

    Before I discuss the problems with preambles and target properties, I would like to give you a quick overview of how the C++ code generator works. Consider the following LF program consisting of two files Foo.lf and Bar.lf:

    \n
    // Bar.lf\n\nreactor Bar {\n reaction(startup) {=\n   // do something bar like\n =}\n}
    \n
    // Foo.lf\n\nimport Bar.lf\n\nreactor Foo {\n bar = new Bar();\n reaction(startup) {=\n   // do something foo like\n =}\n}
    \n

    Now let us have a look on what the C++ code generator does. It will produce a file structure like this:

    \n
    CMakeLists.txt\nmain.cc\nBar/\n  Bar.cc\n  Bar.hh\nFoo/\n  Foo.cc\n  Foo.hh
    \n

    We can ignore CMakeLists.txt and main.cc for our discussion here. The former specifies how the whole program can be build and the latter contains the main() function and some code that is required to get the application up and running. For each processed <file>.lf file, the code generator creates a directory <file>. For each reactor <reactor> defined in <file>.lf, it will create <file>/<reactor>.cc and <file>/<reactor>.hh. The header file declares a class representing the reactor like this:

    \n
    // Bar/Bar.hh\n\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n\nclass Bar : public reactor::Reacor {\n private:\n  // default actions\n  reactor::StartupAction startup {"startup", this};\n  reactor::ShutdownAction shutdown {"shutdown", this};\n\n public:\n  /* ... */\n\n private:\n  // reaction bodies\n  r0_body();\n};
    \n

    The corresponding Bar/Bar.cc will look something like this:

    \n
    #include "Bar/Bar.hh"\n\n/* ... */\n\nBar::r0_body() {\n  // do something bar like\n}
    \n

    Similarly, Foo.hh and Foo.cc will be generated. However, since Foo.lf imports Bar.lf and instantiated the reactor Bar it must be made visible. This is done by an include directive in the generated code like so:

    \n
    // Foo/Foo.hh\n###\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n#include "Bar/Bar.hh"\n\nclass Foo : public reactor::Reacor {\n private:\n  // default actions\n  reactor::StartupAction startup;\n  reactor::ShutdownAction shutdown;\n\n  // reactor instances\n  Bar bar;\n public:\n  /* ... */\n\n private:\n  // reaction bodies\n  r0_body();\n};
    \n

    The problem with preambles

    \n

    The problems with preamble in the context of imports were already discussed in a related issue, but I would like to summarize the problem here. While the examples above worked nicely even with imports, things get messy as soon as we introduce a preamble. Let’s try this:

    \n
    // Bar.lf\n\nreactor Bar {\n preamble {=\n   struct bar_t {\n     int x;\n     std::string y;\n   };\n\n   bar_t bar_func {\n     return bar_t(42, "hello")\n   }\n =}\n output out:bar_t;\n reaction(startup) -> out {=\n   out.set(bar_fuc());\n =}\n}
    \n
    // Foo.lf\n\nimport Bar.lf\n\nreactor Foo\n bar = new Bar();\n reaction(bar.out) {=\n   auto& value = bar.out.get();\n   std::cout << "Received {" << value->x << ", " << value->y << "}\\n";\n =}\n}
    \n

    This would be expected to print Received {32, hello}. However, before we can even compile this program, we need to talk about what should happen with the preamble during code generation and how the import affects it. So where should the preamble go? The first thing that comes to mind, is to embed it in the header file Bar.hh something like this:

    \n
    // Bar/Bar.hh\n\n# pragma once\n\n#include "reactor-cpp/reactor-cpp.hh"\n\n// preamble\nstruct bar_t {\n  int x;\n  std::string y;\n};\n\nbar_t bar_func {\n  return bar_t(42, "hello")\n}\n\nclass Bar : public reactor::Reacor {\n /* ... */\n};
    \n

    If we embed the preamble like this and compile the program ,then the compiler is actually happy and processes all *.cc files without any complaints. But, there is a huge problem while linking the binary. The linker sees multiple definitions of bar_func and has no idea which one to use. Why is that? Well, the definition of bar_func is contained in a header file. This should never be done in C/C++! Since includes translate to a plain-text replacement by the preprocessor, Bar.cc will contain the full definition of bar_func. As Foo.cc imports Foo.hh which imports Bar.hh, also Foo.cc will contain the full definition. And since main.cc also has to include Foo.hh, main.cc will also contain the full definition of bar_func. So we have multiple definitions of the same function and the linker rightfully reports this as an error.

    \n

    So what should we do? We could place the preamble in Bar.cc instead. This ensures that only Bar.cc sees the definition of bar_func. But then the compiler complains. Neither Bar.hh nor Foo.hh see type declaration of bar_t. Note that there is a dependency of Foo.lf on the preamble in Bar.lf. The import system should somehow take care of this dependency! Also note that this has not appeared as a problem in C as the code generator places everything in the same compilation unit. Foo will see the preamble of Bar as long as Foo is generated before Bar.

    \n

    But how to solve it for C++ where the code is split in multiple compilation units (which really should be happening in C as well)? What we do at the moment is annotating the preamble with private and public keywords. This helps to split the preamble up and decide what to place in the header and what to place in the source file. For instance:

    \n
    // Bar.lf\n\nreactor Bar {\n public preamble {=\n   struct bar_t {\n     int x;\n     std::string y;\n   };\n =}\n private preamble {=\n   bar_t bar_func {\n     return bar_t(42, "hello")\n   }\n =}\n output out:bar_t;\n reaction(startup) -> out {=\n   out.set(bar_fuc());\n =}\n}
    \n

    This makes the type bar_t visible as part of the public interface of Bar. Both the code generated for Bar and the code generated for Foo will see the definition of bar_t. This is realized by placing the public preamble in Bar.hh The function bar_func is part of Bar’s private interface. It is only visible with the reactor definition of Bar and is not propagated by an import. This is realized by simply placing the private preamble in Bar.cc. This makes the compiler finally happy and when get an executable program private and public preambles provide a mechanism to define what is propagated on an import and what is not. I think this is an important distinction even in languages other than C/C++ that do not have this weird separation of source and header file.

    \n

    I am sorry for this lengthy diversion into things that happened in the past where we actually want to talk about how things should work in the future. However, understanding this issue is important and when talking about other solutions we should not forget that it exists.

    \n

    The problem with target properties

    \n

    It is also not well-defined what should happen with target properties when importing a .lf file. Apparently the common practice is simply ignoring the existence of other target declarations and only considering the target declaration of the .lf that contains the main reactor. I think this works reasonably well for our small programs. But it will cause problems when either programs become larger or we introduce new target properties where it is unclear what piece of code they reference. Let us have a look at the existing target properties for C++. How should those different properties be handled on an import? Which scope do they actually apply to? We haven’t really talked about this.

    \n

    fast, keepalive, threads and timeout are easy. They apply to the main reactor. Since we do not import main reactors from other files, it is clear that we really want to use the properties defined in the main compilation unit. So our current strategy works in this case. Although there are still some subtleties. For instance, if a library file defines keepalive=true and fast=false because it uses physical actions, should any file importing this library file be allowed to override these properties. Probably not, because it doesn’t make sense if physical actions are involved. But a careless user of the library might not be aware of that. So maybe it isn’t that clear after all.

    \n

    build-type, cmake-include, compile, logging and no-runtime-validation influence how the application is build. They are used for generating the CMakeLists.txt file. So their is quite clear: they apply to the whole compilation of the given application. Again it is a simple solution to only consider the target properties of the file containing the main reactor since this can be considered the file that ‘drives’ the compilation. But what if an imported .lf relies on an external library and uses the cmake-include property to tell CMake to look this library up, make the library header files visible and link our generated code to that library (fortunately this can be done with 2 lines in CMake). Should this target property really be ignored by our import? Probably not, because it will lead to compile errors if the author of the main .lf file does not configure cmake-include properly. So there should be some kind of merging mechanism for cmake-include. Should this be done for the other properties as well? I am not sure and I actually don’t know how the merging would work.

    \n

    So this raises a lot of questions that we currently have no answer to. I believe we need to find answers for these questions in order to create a well working import and package system. This gets only more complicated when we add more properties such as the proposed files directive. We should really consider what properties actually apply to and if they influence the way imports work.

    \n

    The work in progress

    \n

    To be continued… I want to describe here what is happening on the new_import and the (potential) problems this brings.

    \n

    Possible solutions

    \n

    To be continued… I would like to show a few possible solutions that have come to mind and that we discussed already.

    \n

    Concrete proposal

    \n

    With the risk of overlooking some of the issues discussed above, I’d like to outline a concrete proposal. To me, at least, it is easier to reason about these issues in a context with a few more constraints. Hopefully, this can serve as a starting point that we can tweak/adjust as needed. Note: this proposal borrows from the previous proposal written by Soroush. Based on my experience with Xtext, I have confidence that what is described below is feasible to implement.

    \n

    Import/export

    \n
      \n
    1. One LF file can contain multiple reactor definitions.
    2. \n
    3. There can be at most one main reactor per file.
    4. \n
    5. Any reactor class defined outside of the current file has to be imported explicitly.
    6. \n
    7. The visibility of a reactor class can be limited using a modifier in the class definition.
    8. \n
    \n
      \n
    • Should the default visibility be public or private? I have no strong preference either way.
    • \n
    \n
      \n
    1. An import statement must specify which reactor classes to import. This is necessary because if we populate a global scope using the uriImport mechanism, the local scope provider needs to know which definition to link to if there happen to exist multiple among the set of included files. We could potentially relax this constraint and only report the situation where we know for a fact that there is ambiguity and needs to be resolved by making the imports explicit. We could also deprecate the use of unqualified imports (the original syntax), therefore allow it but warn that it might not work as expected.
    2. \n
    3. An LF file in an import statement is specified by a path relative to the directory of the file in which the import statement occurs or relative to a series of directories in a wider search path.
    4. \n
    \n
      \n
    • Eclipse uses .project files to identify the root of a project; we can look for that.
    • \n
    • We can look for our own kind of manifest files as well. These can list additional locations to search. This is compatible with the idea of developing a package system. I personally like this approach better than using an environment variable.
    • \n
    \n
      \n
    1. I personally find fully qualified names excess generality and have never felt compelled to use them in Java. IMO, they lead to code that’s difficult to read and a pain to format. To keep things simple, I suggest we don’t support them. Instead, we should provide a mechanism for renaming imported reactor classes to avoid naming conflicts.
    2. \n
    3. Open question: do we want scope modifiers for imports? It seems that extra import statements could be used to increase visibility, so it might not be needed.
    4. \n
    \n

    Syntax

    \n
    Import := 'import' <ID> <Rename>? (',' <ID> <Rename>?)* 'from' <PATH>\n\nRename := 'as' <ID>
    \n

    Note: This syntax could be extended to support packages in addition to paths. But it doesn’t make much sense to have this until we have a package manager and package registry.

    \n

    Current state of the discussion: one unifying syntax vs. different syntax for references to files and packages.

    \n

    Preambles

    \n

    A preamble allows for the inclusion of verbatim target code that may be necessary for reactors to function. Currently, there are two scopes in which preambles can appear: (1) file scope and (2) reactor class scope. Moreover, there exist visibility modifiers to label preambles private or public. A public preamble is intended to contain code that is necessary for the use of a reactor that is in scope. A private preamble is intended to contain code that is necessary for the implementation of a reactor that is in scope. Only the C++ code generator can currently effectively separate these LF scope levels. It achieves this by putting each reactor class definition in its own file. LF file scope preambles are currently not supported by the C target, but this appears to be unintentional and would be easy to fix. Reactor class scope preambles are supported by the C target, but there is no isolation of scope; the preamble of one reactor is visible to the one defined after it. To fix this, I see two options: (1) follow the same approach as CppGenerator and output separate files, which also means that a Makefile has to be generated in order to compile the result, or (2) leverage block scope within a single file, but this will become complicated and make the generated C code even less humanly readable.

    \n

    We could put aside the problem of name clashes due to the absence of scope isolation in generated C code and fix this later. For the time being, the problem can be circumvented using .h files.

    \n

    Target Properties

    \n
      \n
    1. Each file declares a target.
    2. \n
    3. All code in all reactors in the same file must agree with the specified target.
    4. \n
    5. Additional target properties may be specified.
    6. \n
    7. Target properties are not inherited through imports.
    8. \n
    9. Any property that needs to be inherited through an import (such as the requirement to link against the pthread library) must be specified as a build dependency instead.
    10. \n
    \n

    Build Dependencies

    \n
      \n
    1. It must be possible to specify build dependencies, such as files, sources, and protobufs.
    2. \n
    3. We could either allow these definitions to go directly in the .lf file, or we could decide to specify them in a package description (i.e., separate file). We could potentially allow both.
    4. \n
    5. Build dependencies are inherited through imports (or from package descriptions), and they are never shadowed, always joined.
    6. \n
    \n

    Unique Reactor Names

    \n

    The new import system as described above ensures that reactor names within a single .lf file are unique. In case reactors with the same name are imported from different .lf files, the renaming mechanism needs to be used in order to resolve the name conflict. The same applies if the given .lf file defines some reactors and tries to import other reactors with the same name. For instance, consider the LF file in Example 1 below. In the scope of this file, three reactor declarations are visible: Foo, Bar and Baz, although the actual reactors have the same name Foo.

    \n

    Examples

    \n

    Throughout this section, I will be using two LF example programs. Since the markdown syntax does not provide a simple way to label and refer to code listings, I figure its easiest to place them here in a central place and refer to them later by the heading

    \n

    Example 1

    \n
    \\\\ Foo.lf\nimport Foo as Bar from "Bar.lf"\nimport Foo as Baz from "Baz.lf"\n\nreactor Foo {\n  \\\\ ...\n}
    \n

    Example 2

    \n
    \\\\ Baz.lf\nreactor Foo {\n  \\\\ ...\n}
    \n
    \\\\ Bar.lf\nimport Foo from "Baz.lf"\n\nreactor Foo {\n  foo = new Foo()\n  \\\\ ...\n}
    \n
    \\\\ Foo.lf\nimport Bar from "Bar.lf"\nmain reactor Foo {\n  bar = new Bar()\n  \\\\ ...\n}
    \n

    Unique Reactor Names in Target Code

    \n

    While the mechanism above effectively ensures uniqueness in a single LF file, this uniqueness is surprisingly hard to ensure in generated target code. C has an obvious problem here as it places all generated code in a single file. While the name conflict in the above code can be solved by generating code for three reactors named Bar, Baz and Foo, it breaks as soon as another file of the same LF program uses import Foo from \"Bar.lf\". Then there would be two definitions of the reactor Foo that cannot be resolved.

    \n

    Now you would probably expect that splitting the generated code into multiple files solves the issue, but unfortunately this is not true. If anything, it makes the problem more subtle. The C++ code generated from Example 1 would likely look something like this:

    \n
    // Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n#include "Baz.hh"\n\n// do the renaming\nusing Bar = Foo;\nusing Baz = Foo;\n\nclass Foo : public reactor::Reactor {\n};
    \n

    This will cause a compile error as there are multiple definitions of Foo. While renaming is possible in C++ with the using keyword (typedef works as well), the thing being renamed needs to be already visible in the scope. So there are multiple definitions of Foo as all the files Bar.hh, Baz.hh and Foo.hh define this class. We need a mechanism to distinguish the different definitions of Foo.

    \n

    There is even another issue that stems from the fact that the semantics of imports in LF are different from the include semantics of C++. Consider the code in Example 2, which is valid LF code. Although Bar.lf imports Foo and Foo.lf imports from Bar.lf, the definition of Foo in Baz.lf is not visible in Foo.lf. This ‘hiding’, however, does not easily propagate to the generated code. In C, there will be an error because both definitions of Foo are placed in the same file. In C++, the different definitions of Foo are placed in different files, but there will still be an error. The generated C++ code would look something like this:

    \n
    \\\\ Baz.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\nclass Foo : public reactor::Reactor {\n  // ...\n};
    \n
    \\\\ Bar.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Baz.hh"\n\nclass Bar : public reactor::Reactor {\n  Foo foo;\n  // ...\n};
    \n
    \\\\ Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n\nclass Foo : public reactor::Reactor {\n  Bar bar;\n  // ...\n};
    \n

    This will produce an error due to multiple definitions of Foo being visible in Foo.hh. The problem is that any include in Bar.hh becomes also visible in Foo.hh. So there is a name clash due to the way the C++ compiler processes included and that is hard to work around.

    \n

    Possible Solutions

    \n

    In conclusion from the above section, I can say that translating the file based scoping of reactor names that we have in LF to generated target code is not trivial. Any sensible solution will need to establish a mechanism to ensure that any two distinct reactors in LF are also distinct in target code.

    \n

    Namespaces

    \n

    We could introduce some form of a namespace mechanism that allows us to derive fully-qualified names of reactors. This is the preferred solution for me (Christian). Note that by ‘namespace’ I mean any logical organization of reactors in named groups and not the precise concept of C++ namespaces. In other languages those logical groups are also referred to as modules or packages. Also note that it is only important to be able to assign a fully-qualified name to a reactor, it does not necessarily require that we refer to reactors by their fully-qualified name in LF code.

    \n

    File based namespaces

    \n

    In my view, the easiest way to introduce namespaces in LF would be to leverage file system structure. Everything contained in Foo.lf would automatically be in the namespace Foo. So the FQN of a reactor Foo defined in Foo.lf would be Foo.Foo (or Foo::Foo, or some other delimiter). This would solve the name clashes in both of our examples. For Example 1, the generated code could look like this:

    \n
    // Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n#include "Baz.hh"\nnamespace Foo {\n\n// do the renaming\nusing Bar = Bar::Foo;\nusing Baz = Baz::Foo;\n\nclass Foo : public reactor::Reactor {\n};\n}
    \n

    For Example 2, the generated code could look like this:

    \n
    \\\\ Baz.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\nnamespace Baz {\nclass Foo : public reactor::Reactor {\n  // ...\n};\n}
    \n
    \\\\ Bar.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Baz.hh"\n\nnamespace Bar {\nusing Foo = Baz::Foo; // bring Foo in scope\n\nclass Bar : public reactor::Reactor {\n  Foo foo;\n  // ...\n};\n}
    \n
    \\\\ Foo.hh\n#include <reactor-cpp/reactor-cpp.hh>\n\n#include "Bar.hh"\n\nnamespace Foo {\nusing Bar = Bar::Bar; // bring Bar in scope\n\nclass Foo : public reactor::Reactor {\n  Bar bar;\n  // ...\n};\n}
    \n

    While this appears to be a promising solution, it is not sufficient to only consider the name of an *.lf file to derive the namespace\nThere could be two files Foo.lf in different directories that both define the reactor Foo. Thus, we also need to consider the directory structure and introduce hierarchical namespaces. Consider this directory tree:

    \n
    foo/\n  bar/\n    foo.lf  # defines reactor Foo\n  baz/\n    foo.lf  # defines reactor Foo
    \n

    In this example, the two Foo reactors would have the fully qualified names foo.bar.foo.Foo and foo.baz.foo.Foo.\nIn order for this concept to work, we need the notion of a top-level namespace or directory. Naturally, this would be the package. Therefore, this namespace approach would also require a simple mechanism to define a package. For now this could be rudimentary. Simply placing an empty lf.yaml in the foo/ directory in the above example would be sufficient. In addition to the notion of packages, we would also need a simple mechanism to find packages. However, since packages are something we want to have anyway, it would not hurt to start and implement a rudimentary system now.

    \n

    This proposal is a bit at odds with the file based import mechanism described above. While it is clear what the namespace of an *.lf file relative to a package directory is, it is unclear what the namespace of an arbitrary file outside a package is. Marten suggested to resolve this by using a default namespace or the global namespace whenever a *lf that is not part of a package is imported and to make the user responsible for avoiding any conflicts.

    \n

    We would also need to restrict the naming of files and directories and ban the usage of the namespace delimiter (. or :: or some other) in file and directory names. In my opinion this is not much of a problem and common practice for many languages. If we decide to use this namespace mechanism, it would probably be better to drop the file based imports and switch to imports by FQN (e.g. import Foo from foo.bar.foo)

    \n

    A Namespace Directive

    \n

    As an alternative to the file based namespace mechanism described above, we could also introduce a namespace directive in the LF syntax or as part of the target properties. This would effectively allow the user to specify the namespace that any reactor defined in a file should be part of. This solution would allow to augment the file based import system that we have with a namespace mechanism. It is important to note, however, that this entirely shifts the responsibility for ensuring uniqueness within a namespace to the user. When we derive namespaces from the file path as described above, we can be sure that the resulting namespace only contains unique reactors because we ensure that any LF file only contains unique reactors. If we allow the user to specify the namespace, however, there could easily be two files with the same namespace directive that both define the reactor Foo. This approach might also cause problems for target languages where the namespaces relate to concrete file paths such as in Rust, Python or Java.

    \n

    Name Mangling

    \n

    There are other mechanisms to derive unique names apart from namespaces. One that is widely used by compilers is name mangling which replaces or decorates the original name. For instance, we could simply add a number to the name of generated reactors (Foo1, Foo2, …) to distinguish multiple LF reactor definitions named Foo. What separates our approach from traditional compiler though, is that we are not in control of the full build process and only generate source code to be processed by another compiler. Therefore, any renaming we do when compiling LF code to target code needs to be done with care as it could easily introduce new problems because we are not aware of all the identifiers defined in a target language. For instance if our LF program uses a library that defines the class Foo3, adding a third definition of the reactor Foo to the program would lead to an unexpected error that is also hard to debug.

    \n

    Soroush also proposed to use a hashing mechanism (for instance a hash of the file name) to decorate reactor names. This would be less likely\nto clash with any names defined in some library. However, we would need to make sure that any mechanism we use for generating unique decorated names follows strict rules and generates reproducible names. This reproducibility is crucial for several reasons.

    \n
      \n
    1. \n

      Since even a complex name mangling mechanism has still the chance to produce name clashes with identifiers defined outside of the LF program, those clashes should not occur randomly. There should be either an error or no error on each compilation run. Nondeterministic builds are no fun to deal with.

      \n
    2. \n
    3. \n

      In case of any errors, it is crucial to be able to reproduce and compare builds across machines and platforms. A platform dependent name mangling algorithm (for instance one that hashes file paths) would make it unnecessary hard to reproduce and debug compile errors.

      \n
    4. \n
    5. \n

      Somewhere in the future, we might want to be able to compile packages as libraries. Recompilation of the library should never change its API. Moreover, the name mangling algorithm should be robust in the sense that small changes in LF code do not lead to changed identifiers\nin the library API.

      \n
    6. \n
    \n

    All in all, I think it is hard to define an algorithm that generates reproducible and stable names, but maybe someone else has a good idea of how this could be achieved.

    \n

    Another obvious disadvantage of the name mangling approach would be that the generated code is less readable. Also any external target code that might want to reference reactors in a library compiled from LF code, would need to know and use the mangled name.

    \n

    Unique Reactor Names in our Tools

    \n

    In our last meeting (Tue 2020-08-04), I said that there are other places where we care about unique names: our tools such as the diagram view or the trace generator that I implemented for C++ and that we cannot ensure that names are unique at the moment. However, while thinking about it a bit more I realized that this is not much of an issue. Ambiguous names of reactor types are not a big problem for the diagram view. Since clicking on the nodes jumps directly to the reactor definition, the ambiguity in the names can easily be resolved.

    \n

    For the tracing, I realized that it is not the name of the reactor type that matters, but the name of the instance. These are unique fully-qualified names already. For instance main.foo.bar.r0, denotes the reaction with priority 0, of a reactor instance called bar that is contained by the reactor instance foo, which is in turn contained by the main reactor.

    \n

    Summary

    \n

    All in all, I think leveraging the file structure for determining the fully qualified names of reactors is the most promising solution.

    \n
      \n
    1. It works without any changes in our syntax. Only the code generators need to be updated to support the namespacing.
    2. \n
    3. In contrast to name mangling, it allows generation of readable code and also gives the programmer full control of how generated reactors are named.
    4. \n
    5. It fits naturally to languages that also support leveraging the file structure to create namespaces (e.g. python or rust).
    6. \n
    ","headings":[{"value":"The status quo","depth":1},{"value":"A quick dive into the C++ code generator","depth":2},{"value":"The problem with preambles","depth":2},{"value":"The problem with target properties","depth":2},{"value":"The work in progress","depth":3},{"value":"Possible solutions","depth":3},{"value":"Concrete proposal","depth":2},{"value":"Import/export","depth":3},{"value":"Syntax","depth":3},{"value":"Preambles","depth":2},{"value":"Target Properties","depth":2},{"value":"Build Dependencies","depth":2},{"value":"Unique Reactor Names","depth":1},{"value":"Examples","depth":2},{"value":"Example 1","depth":3},{"value":"Example 2","depth":3},{"value":"Unique Reactor Names in Target Code","depth":2},{"value":"Possible Solutions","depth":2},{"value":"Namespaces","depth":3},{"value":"File based namespaces","depth":4},{"value":"A Namespace Directive","depth":4},{"value":"Name Mangling","depth":3},{"value":"Unique Reactor Names in our Tools","depth":2},{"value":"Summary","depth":2}],"frontmatter":{"permalink":"/docs/handbook/proof-import","title":"Future Proof Package/Import System","oneline":"A future proof package and import system","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/proof-import","repoPath":"/packages/documentation/copy/en/less-developed/Future Proof Package and Import System.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/reaction-declarations/page-data.json b/page-data/docs/handbook/reaction-declarations/page-data.json index f8a8ac015..6fe76d41f 100644 --- a/page-data/docs/handbook/reaction-declarations/page-data.json +++ b/page-data/docs/handbook/reaction-declarations/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reaction-declarations","result":{"data":{"markdownRemark":{"id":"9562e9fd-8d70-5f52-b5b1-0a79a3e238c2","excerpt":"Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also…","html":"

    Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also possible to just declare them and provide implementations in a separate file. The syntax of reaction declarations is the same as for reaction definitions, except they have no implementation. Reaction declarations can be thought of as function prototypes.

    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target C {\n  cmake-include: ["hello.cmake"],\n  files: ["hello.c"]\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The cmake-include target property is used to make the build system aware of an externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.c)\n
    \n

    The files target property is used to make the file that has the implementation in hello.c accessible,\nwhich could look something like this:

    \n
    #include <stdio.h>\n#include "../include/HelloDecl/HelloDecl.h"\n\nvoid hello(hellodecl_self_t* self) {\n    printf("Hello declaration!\\n");\n}\n
    \n

    File Structure

    \n

    In the above example, the C file used #include to import a file called HelloDecl.h. This file\nwas generated from the Lingua Franca source file when the LF program was compiled. The file\nHelloDecl.h is named after the main reactor, which is called HelloDecl, and its parent\ndirectory, include/HelloDecl, is named after the file, HelloDecl.lf.

    \n

    In general, compiling a Lingua Franca program that uses reaction declarations will always generate a\ndirectory in the include directory for each file in the program. This directory will contain a\nheader file for each reactor that is defined in the file.

    \n

    As another example, if an LF program consists of files F1 and F2, where F1 defines reactors\nA and B and F2 defines the reactor C and the main reactor F2, then the directory structure\nwill look something like this:

    \n
    include/\n├ F1/\n│ ├ A.h\n│ └ B.h\n└ F2/\n  ├ C.h\n  └ F2.h\nsrc/\n├ F1.lf  // defines A and B\n└ F2.lf  // defines C and F2\nsrc-gen/
    \n

    There is no particular location where you are required to place your C files or your CMake files.\nFor example, you may choose to place them in a directory called c that is a sibling of the src\ndirectory.

    \n

    The Generated Header Files

    \n

    The generated header files are necessary in order to separate your C code from your LF code because\nthe describe the signatures of the reaction functions that you must implement.

    \n

    In addition, they define structs that will be referenced by the reaction bodies. This includes the\nself struct of the reactor to which the header file corresponds, as well as structs for its ports,\nits actions, and the ports of its child reactors.

    \n

    As with preambles, programmer discipline is required to avoid breaking the deterministic semantics\nof Lingua Franca. In particular, although the information exposed in these header files allows\nregular C code to operate on ports and self structs, such information must not be saved in global or\nstatic variables.

    \n

    Linking Your C Code

    \n

    As with any Lingua Franca project that uses external C files, projects involving external reactions\nmust use the cmake-include target property to link those files into the main target.

    \n

    This is done using the syntax

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE <files>)\n
    \n

    where <files> is a list of the C files you need to link, with paths given relative to the project\nroot (the parent of the src directory).

    \n
    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target Cpp {\n  cmake-include: ["hello.cmake"],\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The behavior of the hello reaction is provided using a method definition in an external C++ file hello.cc.

    \n
    #include "HelloDecl/HelloDecl.hh" // include the code generated reactor class\n\n// define the reaction implementation\nvoid HelloDecl::Inner::hello([[maybe_unused]] const reactor::StartupTrigger& startup) {\n  std::cout << "Hello World." << std::endl;\n}\n
    \n

    Using the cmake-include target property, we can make the build system aware of this externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.cc)\n
    \n

    Note that this mechanism can be used to add arbitrary additional resources such as additional headers and implementation files or 3rd party libraries to the compilation.

    \n

    Header Files and Method Signatures

    \n

    In order to provide an implementation of a reaction method, it is important to know the header file that declares the reactor class as well as the precise signature of the method implementing the reaction body.

    \n

    The LF compiler generates a header file for each reactor that gets defined in LF code. The header file is named after the corresponding reactor and prefixed by the path to the LF file that defines the reactor. Consider the following example project structure:

    \n
    src/\n├ A.lf   // defines Foo\n└ sub/\n  └ B.lf // defines Bar
    \n

    In this case, the compiler will generate two header files A/Foo.hh and sub/B/Bar.hh, which would need to be included by an external implementation file.

    \n

    The precise method signature depends on the name of the reactor, the name of the reactions, and the precise triggers, sources, and effects that are defined in the LF code.\nThe return type is always void. A reaction foo in a reactor Bar will be named Bar::Inner::foo. Note that each reactor class in the C++ target defines an Inner class which contains all reactions as well as the parameters and state variables. This is done to deliberately restrict the scope of reaction bodies in order to avoid accidental violations of reactor semantics.\nAny declared triggers, sources or effects are given to the reaction method via method arguments. The precise arguments and their types depend on the LF code. If in doubt, please check the signature used in the generated header file under src-gen/<lf-file>, where <lf-file> corresponds to the LF file that you are compiling.

    \n
    \n
    \n

    The $target-language$ target does not currently support reaction declarations.

    \n
    ","headings":[{"value":"Example","depth":2},{"value":"File Structure","depth":2},{"value":"The Generated Header Files","depth":2},{"value":"Linking Your C Code","depth":2},{"value":"Example","depth":2},{"value":"Header Files and Method Signatures","depth":2}],"frontmatter":{"permalink":"/docs/handbook/reaction-declarations","title":"Reaction Declarations","oneline":"Reaction declarations in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/reaction-declarations","repoPath":"/packages/documentation/copy/en/topics/Reaction Declarations.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reaction-declarations","result":{"data":{"markdownRemark":{"id":"9562e9fd-8d70-5f52-b5b1-0a79a3e238c2","excerpt":"Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also…","html":"

    Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also possible to just declare them and provide implementations in a separate file. The syntax of reaction declarations is the same as for reaction definitions, except they have no implementation. Reaction declarations can be thought of as function prototypes.

    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target C {\n  cmake-include: ["hello.cmake"],\n  files: ["hello.c"]\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The cmake-include target property is used to make the build system aware of an externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.c)\n
    \n

    The files target property is used to make the file that has the implementation in hello.c accessible,\nwhich could look something like this:

    \n
    #include <stdio.h>\n#include "../include/HelloDecl/HelloDecl.h"\n\nvoid hello(hellodecl_self_t* self) {\n    printf("Hello declaration!\\n");\n}\n
    \n

    File Structure

    \n

    In the above example, the C file used #include to import a file called HelloDecl.h. This file\nwas generated from the Lingua Franca source file when the LF program was compiled. The file\nHelloDecl.h is named after the main reactor, which is called HelloDecl, and its parent\ndirectory, include/HelloDecl, is named after the file, HelloDecl.lf.

    \n

    In general, compiling a Lingua Franca program that uses reaction declarations will always generate a\ndirectory in the include directory for each file in the program. This directory will contain a\nheader file for each reactor that is defined in the file.

    \n

    As another example, if an LF program consists of files F1 and F2, where F1 defines reactors\nA and B and F2 defines the reactor C and the main reactor F2, then the directory structure\nwill look something like this:

    \n
    include/\n├ F1/\n│ ├ A.h\n│ └ B.h\n└ F2/\n  ├ C.h\n  └ F2.h\nsrc/\n├ F1.lf  // defines A and B\n└ F2.lf  // defines C and F2\nsrc-gen/
    \n

    There is no particular location where you are required to place your C files or your CMake files.\nFor example, you may choose to place them in a directory called c that is a sibling of the src\ndirectory.

    \n

    The Generated Header Files

    \n

    The generated header files are necessary in order to separate your C code from your LF code because\nthe describe the signatures of the reaction functions that you must implement.

    \n

    In addition, they define structs that will be referenced by the reaction bodies. This includes the\nself struct of the reactor to which the header file corresponds, as well as structs for its ports,\nits actions, and the ports of its child reactors.

    \n

    As with preambles, programmer discipline is required to avoid breaking the deterministic semantics\nof Lingua Franca. In particular, although the information exposed in these header files allows\nregular C code to operate on ports and self structs, such information must not be saved in global or\nstatic variables.

    \n

    Linking Your C Code

    \n

    As with any Lingua Franca project that uses external C files, projects involving external reactions\nmust use the cmake-include target property to link those files into the main target.

    \n

    This is done using the syntax

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE <files>)\n
    \n

    where <files> is a list of the C files you need to link, with paths given relative to the project\nroot (the parent of the src directory).

    \n
    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target Cpp {\n  cmake-include: ["hello.cmake"],\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The behavior of the hello reaction is provided using a method definition in an external C++ file hello.cc.

    \n
    #include "HelloDecl/HelloDecl.hh" // include the code generated reactor class\n\n// define the reaction implementation\nvoid HelloDecl::Inner::hello([[maybe_unused]] const reactor::StartupTrigger& startup) {\n  std::cout << "Hello World." << std::endl;\n}\n
    \n

    Using the cmake-include target property, we can make the build system aware of this externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.cc)\n
    \n

    Note that this mechanism can be used to add arbitrary additional resources such as additional headers and implementation files or 3rd party libraries to the compilation.

    \n

    Header Files and Method Signatures

    \n

    In order to provide an implementation of a reaction method, it is important to know the header file that declares the reactor class as well as the precise signature of the method implementing the reaction body.

    \n

    The LF compiler generates a header file for each reactor that gets defined in LF code. The header file is named after the corresponding reactor and prefixed by the path to the LF file that defines the reactor. Consider the following example project structure:

    \n
    src/\n├ A.lf   // defines Foo\n└ sub/\n  └ B.lf // defines Bar
    \n

    In this case, the compiler will generate two header files A/Foo.hh and sub/B/Bar.hh, which would need to be included by an external implementation file.

    \n

    The precise method signature depends on the name of the reactor, the name of the reactions, and the precise triggers, sources, and effects that are defined in the LF code.\nThe return type is always void. A reaction foo in a reactor Bar will be named Bar::Inner::foo. Note that each reactor class in the C++ target defines an Inner class which contains all reactions as well as the parameters and state variables. This is done to deliberately restrict the scope of reaction bodies in order to avoid accidental violations of reactor semantics.\nAny declared triggers, sources or effects are given to the reaction method via method arguments. The precise arguments and their types depend on the LF code. If in doubt, please check the signature used in the generated header file under src-gen/<lf-file>, where <lf-file> corresponds to the LF file that you are compiling.

    \n
    \n
    \n

    The $target-language$ target does not currently support reaction declarations.

    \n
    ","headings":[{"value":"Example","depth":2},{"value":"File Structure","depth":2},{"value":"The Generated Header Files","depth":2},{"value":"Linking Your C Code","depth":2},{"value":"Example","depth":2},{"value":"Header Files and Method Signatures","depth":2}],"frontmatter":{"permalink":"/docs/handbook/reaction-declarations","title":"Reaction Declarations","oneline":"Reaction declarations in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/reaction-declarations","repoPath":"/packages/documentation/copy/en/topics/Reaction Declarations.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/reactions/page-data.json b/page-data/docs/handbook/reactions/page-data.json index 6cc29a125..a25fd0b01 100644 --- a/page-data/docs/handbook/reactions/page-data.json +++ b/page-data/docs/handbook/reactions/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reactions","result":{"data":{"markdownRemark":{"id":"1432f795-b6dc-5a03-84b2-2f09c3c01d21","excerpt":"Reaction Declaration A reaction declaration has the following form: Each reaction declares its triggers, uses, and effects: The triggers field can be a comma…","html":"

    Reaction Declaration

    \n

    A reaction declaration has the following form:

    \n
      reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n
    \n

    Each reaction declares its triggers, uses, and effects:

    \n
      \n
    • The triggers field can be a comma-separated list of input ports, output ports of contained reactors, timers, actions, or the special events $startup$, $shutdown$, and $reset$ (explained here). There must be at least one trigger for each reaction.
    • \n
    • The uses field, which is optional, specifies input ports (or output ports of contained reactors) that do not trigger execution of the reaction but may be read by the reaction.
    • \n
    • The effects field, which is also optional, is a comma-separated lists of output ports ports, input ports of contained reactors, or actions.
    • \n
    \n

    Reactions may optionally be named. The name is cosmetic and may serve as additional documentation. Note that reactions cannot be called like functions, even if they are named.

    \n

    The reaction’s behavior is defined by its body, which should be given in the target programming language. Note that the reaction body may only read from actions and ports that it has declared as triggers or uses, and it may only write to actions and ports that is has declared as an effect. The target code generators implement a scoping mechanism, such that only variables that are declared in the reaction signature are accessible in the reaction body.

    \n

    In some targets, the reaction body may be omitted and the body can be defined natively in the target language in an external file. See the section on Bodyless Reactions for details.

    \n

    Reaction Order

    \n

    A reactor may have multiple reactions, and more than one reaction may be enabled at any given tag. In Lingua Franca semantics, if two or more reactions of the same reactor are simultaneously enabled, then they will be invoked sequentially in the order in which they are declared. More strongly, the reactions of a reactor are mutually exclusive and are invoked in tag order primarily and declaration order secondarily. Consider the following example:

    \n

    $start(Alignment)$

    \n
    target C {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s: int = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self->s += 1;\n  =}\n  reaction(t2) {=\n    self->s -= 2;\n  =}\n  reaction(t4) {=\n    printf("s = %d\\n", self->s);\n  =}\n}\n
    \n
    target Cpp {\n  timeout: 3 s\n}\nmain reactor Alignment {\n  state s: int(0)\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  timer t4(400 ms, 400 ms)\n  reaction(t1) {=\n    s += 1;\n  =}\n  reaction(t2) {=\n    s -= 2;\n  =}\n  reaction(t4) {=\n    std::cout << "s = " << std::to_string(s) << std::endl;\n  =}\n}\n
    \n
    target Python {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self.s += 1\n  =}\n  reaction(t2) {=\n    self.s -= 2\n  =}\n  reaction(t4) {=\n    print(f"s = {self.s}")\n  =}\n}\n
    \n
    target TypeScript {\n  timeout: 3 s\n}\nmain reactor Alignment {\n  state s: number = 0\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  timer t4(400 ms, 400 ms)\n  reaction(t1) {=\n    s += 1\n  =}\n  reaction(t2) {=\n    s -= 2\n  =}\n  reaction(t4) {=\n    console.log(`s = ${s}`)\n  =}\n}\n
    \n
    target Rust {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s: u32 = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self.s += 1;\n  =}\n  reaction(t2) {=\n    self.s -= 2;\n  =}\n  reaction(t4) {=\n    println!("s = {}", self.s);\n  =}\n}\n
    \n

    $end(Alignment)$

    \n

    Every 100 ms, this increments the state variable s by 1, every 200 ms, it decrements s by 2, and every 400 ms, it prints the value of s. When these reactions align, they are invoked in declaration order, and, as a result, the printed value of s is always 0.

    \n

    Overwriting Outputs

    \n

    Just as the reactions of the Alignment reactor overwrite the state variable s, logically simultaneous reactions can overwrite outputs. Consider the following example:

    \n

    $start(Overwriting)$

    \n
    target C\nreactor Overwriting {\n  output y: int\n  state s: int = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self->s += 1;\n    lf_set(y, self->s);\n  =}\n  reaction(t2) -> y {=\n    self->s -= 2;\n    lf_set(y, self->s);\n  =}\n}\n
    \n
    target Cpp\nreactor Overwriting {\n  output y: int\n  state s: int(0)\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  reaction(t1) -> y {=\n    s += 1;\n    y.set(s);\n  =}\n  reaction(t2) -> y {=\n    s -= 2;\n    y.set(s);\n  =}\n}\n
    \n
    target Python\nreactor Overwriting {\n  output y\n  state s = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self.s += 1\n    y.set(self.s)\n  =}\n  reaction(t2) -> y {=\n    self.s -= 2\n    y.set(self.s)\n  =}\n}\n
    \n
    target TypeScript\nreactor Overwriting {\n  output y: number\n  state s: number = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    s += 1\n    y = s\n  =}\n  reaction(t2) -> y {=\n    s -= 2\n    y = s\n  =}\n}\n
    \n
    target Rust\nreactor Overwriting {\n  output y: u32\n  state s: u32 = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self.s += 1;\n    ctx.set(y, self.s);\n  =}\n  reaction(t2) -> y {=\n    self.s -= 2;\n    ctx.set(y, self.s);\n  =}\n}\n
    \n

    $end(Overwriting)$

    \n

    Here, the reaction to t1 will set the output to 1 or 2, but every time it sets it to 2, the second reaction (to t2) will overwrite the output with the value 0. As a consequence, the outputs will be 1, 0, 1, 0, … deterministically.

    \n

    Reacting to Outputs of Contained Reactors

    \n

    A reaction may be triggered by the an input to the reactor, but also by an output of a contained reactor, as illustrated in the following example:

    \n

    $start(Contained)$

    \n
    target C\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if (s.y->value != 0 && s.y->value != 1) {\n      lf_print_error_and_exit("Outputs should only be 0 or 1!");\n    }\n  =}\n}\n
    \n
    target Cpp\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    auto is_correct = [](auto value){\n      return value == 0 || value == 1;\n    };\n    if (s.y.is_present() && !is_correct(*s.y.get())) {\n      std::cout << "Output shoudl only be 0 or 1!" << std::endl;\n    }\n  =}\n}\n
    \n
    target Python\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if s.y.value != 0 and s.y.value != 1:\n      sys.stderr.write("ERROR: Outputs should only be 0 or 1!\\n")\n      exit(1)\n  =}\n}\n
    \n
    target TypeScript\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if (s.y != 0 && s.y != 1) {\n      util.requestErrorStop("Outputs should only be 0 or 1!")\n    }\n  =}\n}\n
    \n
    target Rust\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    let value = ctx.get(s__y).unwrap();\n    if value != 0 && value != 1 {\n      eprintln!("Output schould only be 0 or 1!");\n      ctx.request_stop(Asap);\n    }\n  =}\n}\n
    \n

    $end(Contained)$

    \n\"Lingua\n

    This instantiates the above Overwriting reactor and monitors its outputs.

    \n

    Triggering Contained Reactors

    \n

    A reaction can set the input of a contained reactor, thereby triggering its reactions, as illustrated in the following example:

    \n

    $start(Triggering)$

    \n
    target C\nreactor Inside {\n  input x: int\n  reaction(x) {=\n    printf("Received %d\\n", x->value);=\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    lf_set(i.x, 42);\n  =}\n}\n
    \n
    target Cpp\nreactor Inside {\n  input x: int\n  reaction(x) {=\n    std::cout << "Received " << std::to_string(*x.get()) << std::endl;\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x.set(42);\n  =}\n}\n
    \n
    target Python\nreactor Inside {\n  input x\n  reaction(x) {=\n    print(f"Received {x.value}")\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x.set(42);\n  =}\n}\n
    \n
    target TypeScript\nreactor Inside {\n  input x: number\n  reaction(x) {=\n    console.log("Received ${x}");\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x = 42\n  =}\n}\n
    \n
    target Rust\nreactor Inside {\n  input x: u32\n  reaction(x) {=\n    println!("Received {}", ctx.get(x).unwrap());\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    ctx.set(i__x, 42);\n  =}\n}\n
    \n

    $end(Triggering)$

    \n

    The reaction to $startup$ declares the input port of the inside reactor as an effect and then sets it with value 42.\nThis will cause the inside reactor’s reaction to execute and print Received 42.

    \n

    Startup, Shutdown, and Reset Reactions

    \n

    Reactions may be triggered by the special events $startup$, $shutdown$, or $reset$.\nFor example,

    \n
      reaction(startup) {=\n    // ... Do something\n  =}\n
    \n

    A reaction to $startup$ is triggered at the very first tag when the program begins (or, if within a mode of a modal reactor, when the mode is first entered).\nThis reaction will be logically simultaneous with reactions to timers that have a zero offset.\nAs usual, for logically simultaneous reactions declared within the same reactor, the order in which they are invoked will be governed by the order in which they are declared.

    \n

    A reaction to $shutdown$ is invoked at program termination.\nSee the Termination section for details.

    \n
    \n

    Reactions to the $reset$ event are not supported in $target-language$ because modal reactors are not supported.

    \n
    \n
    \n

    A reaction to the $reset$ event is invoked if the reaction or reactor is within a mode of a modal reactor and the mode is entered via a reset transition.\nFor details, see the Modal Reactors section.

    \n
    \n

    Bodyless Reactions

    \n

    Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also possible to just declare them and provide implementations in a separate file. The syntax of reaction declarations is the same as for reaction definitions, except they have no implementation. Reaction declarations can be thought of as function prototypes.

    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target C {\n  cmake-include: ["hello.cmake"],\n  files: ["hello.c"]\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The cmake-include target property is used to make the build system aware of an externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.c)\n
    \n

    The files target property is used to make accessible the file that has the implementation in `hello.c,\nwhich could look something like this:

    \n
    #include <stdio.h>\n#include "../include/HelloDecl/HelloDecl.h"\n\nvoid hello(hellodecl_self_t* self) {\n    printf("Hello declaration!\\n");\n}\n
    \n

    File Structure

    \n

    In the above example, the C file uses #include to import a file called HelloDecl.h. The HelloDecl.h file\nis generated from the Lingua Franca source file when the LF program is compiled. The file\nHelloDecl.h is named after the main reactor, which is called HelloDecl, and its parent\ndirectory, include/HelloDecl, is named after the file, HelloDecl.lf.

    \n

    In general, compiling a Lingua Franca program that uses reaction declarations will always generate a\ndirectory in the include directory for each file in the program. This directory will contain a\nheader file for each reactor that is defined in the file.

    \n

    As another example, if an LF program consists of files F1.lf and F2.lf, where F1.lf defines reactors\nA and B and F2.lf defines the reactor C and the main reactor F2, then the directory structure\nwill look something like this:

    \n
    include/\n├ F1/\n│ ├ A.h\n│ └ B.h\n└ F2/\n  ├ C.h\n  └ F2.h\nsrc/\n├ F1.lf  // defines A and B\n└ F2.lf  // defines C and F2\nsrc-gen/
    \n

    There is no particular location where you are required to place your C files or your CMake files.\nFor example, you may choose to place them in a directory called c that is a sibling of the src\ndirectory.

    \n

    The Generated Header Files

    \n

    The generated header files are necessary in order to separate your C code from your LF code because\nthey describe the signatures of the reaction functions that you must implement.

    \n

    In addition, they define structs that will be referenced by the reaction bodies. This includes the\nself struct of the reactor to which the header file corresponds, as well as structs for its ports,\nits actions, and the ports of its child reactors.

    \n

    As with preambles, programmer discipline is required to avoid breaking the deterministic semantics\nof Lingua Franca. In particular, although the information exposed in these header files allows\nregular C code to operate on ports and self structs, such information must not be saved in global or\nstatic variables.

    \n

    Linking Your C Code

    \n

    As with any Lingua Franca project that uses external C files, projects involving external reactions\nmust use the cmake-include target property to link those files into the main target.

    \n

    The file referenced by the cmake-include target property has the following syntax:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE <files>)\n
    \n

    where <files> is a list of the C files you need to link, with paths given relative to the project\nroot (the parent of the src directory).

    \n
    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target Cpp {\n  cmake-include: ["hello.cmake"],\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The behavior of the hello reaction is provided using a method definition in an external C++ file hello.cc.

    \n
    #include "HelloDecl/HelloDecl.hh" // include the code generated reactor class\n\n// define the reaction implementation\nvoid HelloDecl::Inner::hello([[maybe_unused]] const reactor::StartupTrigger& startup) {\n  std::cout << "Hello World." << std::endl;\n}\n
    \n

    Using the cmake-include target property, we can make the build system aware of this externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.cc)\n
    \n

    Note that this mechanism can be used to add arbitrary additional resources such as additional headers and implementation files or 3rd party libraries to the compilation.

    \n

    Header Files and Method Signatures

    \n

    In order to provide an implementation of a reaction method, it is important to know the header file that declares the reactor class as well as the precise signature of the method implementing the reaction body.

    \n

    The LF compiler generates a header file for each reactor that gets defined in LF code. The header file is named after the corresponding reactor and prefixed by the path to the LF file that defines the reactor. Consider the following example project structure:

    \n
    src/\n├ A.lf   // defines Foo\n└ sub/\n  └ B.lf // defines Bar
    \n

    In this case, the compiler will generate two header files A/Foo.hh and sub/B/Bar.hh, which would need to be included by an external implementation file.

    \n

    The precise method signature depends on the name of the reactor, the name of the reactions, and the precise triggers, sources, and effects that are defined in the LF code.\nThe return type is always void. A reaction foo in a reactor Bar will be named Bar::Inner::foo. Note that each reactor class in the C++ target defines an Inner class which contains all reactions as well as the parameters and state variables. This is done to deliberately restrict the scope of reaction bodies in order to avoid accidental violations of reactor semantics.\nAny declared triggers, sources or effects are given to the reaction method via method arguments. The precise arguments and their types depend on the LF code. If in doubt, please check the signature used in the generated header file under src-gen/<lf-file>, where <lf-file> corresponds to the LF file that you are compiling.

    \n
    \n
    \n

    The $target-language$ target does not currently support reaction declarations.

    \n
    ","headings":[{"value":"Reaction Declaration","depth":2},{"value":"Reaction Order","depth":2},{"value":"Overwriting Outputs","depth":2},{"value":"Reacting to Outputs of Contained Reactors","depth":2},{"value":"Triggering Contained Reactors","depth":2},{"value":"Startup, Shutdown, and Reset Reactions","depth":2},{"value":"Bodyless Reactions","depth":2},{"value":"Example","depth":3},{"value":"File Structure","depth":3},{"value":"The Generated Header Files","depth":3},{"value":"Linking Your C Code","depth":3},{"value":"Example","depth":3},{"value":"Header Files and Method Signatures","depth":3}],"frontmatter":{"permalink":"/docs/handbook/reactions","title":"Reactions","oneline":"Reactions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","permalink":"/docs/handbook/composing-reactors"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Methods","oneline":"Methods in Lingua Franca.","permalink":"/docs/handbook/methods"}}}},"pageContext":{"id":"1-reactions","slug":"/docs/handbook/reactions","repoPath":"/packages/documentation/copy/en/topics/Reactions.md","previousID":"834f9d0d-f7c6-5732-8c60-bad1954701f7","nextID":"c2fc2a0b-a389-5c48-9cb0-c079bf2e6034","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reactions","result":{"data":{"markdownRemark":{"id":"1432f795-b6dc-5a03-84b2-2f09c3c01d21","excerpt":"Reaction Declaration A reaction declaration has the following form: Each reaction declares its triggers, uses, and effects: The triggers field can be a comma…","html":"

    Reaction Declaration

    \n

    A reaction declaration has the following form:

    \n
      reaction [<name>] (<triggers>) [<uses>] [-> <effects>] [{= ... body ...=}]\n
    \n

    Each reaction declares its triggers, uses, and effects:

    \n
      \n
    • The triggers field can be a comma-separated list of input ports, output ports of contained reactors, timers, actions, or the special events $startup$, $shutdown$, and $reset$ (explained here). There must be at least one trigger for each reaction.
    • \n
    • The uses field, which is optional, specifies input ports (or output ports of contained reactors) that do not trigger execution of the reaction but may be read by the reaction.
    • \n
    • The effects field, which is also optional, is a comma-separated lists of output ports ports, input ports of contained reactors, or actions.
    • \n
    \n

    Reactions may optionally be named. The name is cosmetic and may serve as additional documentation. Note that reactions cannot be called like functions, even if they are named.

    \n

    The reaction’s behavior is defined by its body, which should be given in the target programming language. Note that the reaction body may only read from actions and ports that it has declared as triggers or uses, and it may only write to actions and ports that is has declared as an effect. The target code generators implement a scoping mechanism, such that only variables that are declared in the reaction signature are accessible in the reaction body.

    \n

    In some targets, the reaction body may be omitted and the body can be defined natively in the target language in an external file. See the section on Bodyless Reactions for details.

    \n

    Reaction Order

    \n

    A reactor may have multiple reactions, and more than one reaction may be enabled at any given tag. In Lingua Franca semantics, if two or more reactions of the same reactor are simultaneously enabled, then they will be invoked sequentially in the order in which they are declared. More strongly, the reactions of a reactor are mutually exclusive and are invoked in tag order primarily and declaration order secondarily. Consider the following example:

    \n

    $start(Alignment)$

    \n
    target C {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s: int = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self->s += 1;\n  =}\n  reaction(t2) {=\n    self->s -= 2;\n  =}\n  reaction(t4) {=\n    printf("s = %d\\n", self->s);\n  =}\n}\n
    \n
    target Cpp {\n  timeout: 3 s\n}\nmain reactor Alignment {\n  state s: int(0)\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  timer t4(400 ms, 400 ms)\n  reaction(t1) {=\n    s += 1;\n  =}\n  reaction(t2) {=\n    s -= 2;\n  =}\n  reaction(t4) {=\n    std::cout << "s = " << std::to_string(s) << std::endl;\n  =}\n}\n
    \n
    target Python {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self.s += 1\n  =}\n  reaction(t2) {=\n    self.s -= 2\n  =}\n  reaction(t4) {=\n    print(f"s = {self.s}")\n  =}\n}\n
    \n
    target TypeScript {\n  timeout: 3 s\n}\nmain reactor Alignment {\n  state s: number = 0\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  timer t4(400 ms, 400 ms)\n  reaction(t1) {=\n    s += 1\n  =}\n  reaction(t2) {=\n    s -= 2\n  =}\n  reaction(t4) {=\n    console.log(`s = ${s}`)\n  =}\n}\n
    \n
    target Rust {\n  timeout: 3 secs\n}\nmain reactor Alignment {\n  state s: u32 = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  timer t4(400 msec, 400 msec)\n  reaction(t1) {=\n    self.s += 1;\n  =}\n  reaction(t2) {=\n    self.s -= 2;\n  =}\n  reaction(t4) {=\n    println!("s = {}", self.s);\n  =}\n}\n
    \n

    $end(Alignment)$

    \n

    Every 100 ms, this increments the state variable s by 1, every 200 ms, it decrements s by 2, and every 400 ms, it prints the value of s. When these reactions align, they are invoked in declaration order, and, as a result, the printed value of s is always 0.

    \n

    Overwriting Outputs

    \n

    Just as the reactions of the Alignment reactor overwrite the state variable s, logically simultaneous reactions can overwrite outputs. Consider the following example:

    \n

    $start(Overwriting)$

    \n
    target C\nreactor Overwriting {\n  output y: int\n  state s: int = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self->s += 1;\n    lf_set(y, self->s);\n  =}\n  reaction(t2) -> y {=\n    self->s -= 2;\n    lf_set(y, self->s);\n  =}\n}\n
    \n
    target Cpp\nreactor Overwriting {\n  output y: int\n  state s: int(0)\n  timer t1(100 ms, 100 ms)\n  timer t2(200 ms, 200 ms)\n  reaction(t1) -> y {=\n    s += 1;\n    y.set(s);\n  =}\n  reaction(t2) -> y {=\n    s -= 2;\n    y.set(s);\n  =}\n}\n
    \n
    target Python\nreactor Overwriting {\n  output y\n  state s = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self.s += 1\n    y.set(self.s)\n  =}\n  reaction(t2) -> y {=\n    self.s -= 2\n    y.set(self.s)\n  =}\n}\n
    \n
    target TypeScript\nreactor Overwriting {\n  output y: number\n  state s: number = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    s += 1\n    y = s\n  =}\n  reaction(t2) -> y {=\n    s -= 2\n    y = s\n  =}\n}\n
    \n
    target Rust\nreactor Overwriting {\n  output y: u32\n  state s: u32 = 0\n  timer t1(100 msec, 100 msec)\n  timer t2(200 msec, 200 msec)\n  reaction(t1) -> y {=\n    self.s += 1;\n    ctx.set(y, self.s);\n  =}\n  reaction(t2) -> y {=\n    self.s -= 2;\n    ctx.set(y, self.s);\n  =}\n}\n
    \n

    $end(Overwriting)$

    \n

    Here, the reaction to t1 will set the output to 1 or 2, but every time it sets it to 2, the second reaction (to t2) will overwrite the output with the value 0. As a consequence, the outputs will be 1, 0, 1, 0, … deterministically.

    \n

    Reacting to Outputs of Contained Reactors

    \n

    A reaction may be triggered by the an input to the reactor, but also by an output of a contained reactor, as illustrated in the following example:

    \n

    $start(Contained)$

    \n
    target C\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if (s.y->value != 0 && s.y->value != 1) {\n      lf_print_error_and_exit("Outputs should only be 0 or 1!");\n    }\n  =}\n}\n
    \n
    target Cpp\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    auto is_correct = [](auto value){\n      return value == 0 || value == 1;\n    };\n    if (s.y.is_present() && !is_correct(*s.y.get())) {\n      std::cout << "Output shoudl only be 0 or 1!" << std::endl;\n    }\n  =}\n}\n
    \n
    target Python\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if s.y.value != 0 and s.y.value != 1:\n      sys.stderr.write("ERROR: Outputs should only be 0 or 1!\\n")\n      exit(1)\n  =}\n}\n
    \n
    target TypeScript\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    if (s.y != 0 && s.y != 1) {\n      util.requestErrorStop("Outputs should only be 0 or 1!")\n    }\n  =}\n}\n
    \n
    target Rust\nimport Overwriting from "Overwriting.lf"\nmain reactor {\n  s = new Overwriting()\n  reaction(s.y) {=\n    let value = ctx.get(s__y).unwrap();\n    if value != 0 && value != 1 {\n      eprintln!("Output schould only be 0 or 1!");\n      ctx.request_stop(Asap);\n    }\n  =}\n}\n
    \n

    $end(Contained)$

    \n\"Lingua\n

    This instantiates the above Overwriting reactor and monitors its outputs.

    \n

    Triggering Contained Reactors

    \n

    A reaction can set the input of a contained reactor, thereby triggering its reactions, as illustrated in the following example:

    \n

    $start(Triggering)$

    \n
    target C\nreactor Inside {\n  input x: int\n  reaction(x) {=\n    printf("Received %d\\n", x->value);=\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    lf_set(i.x, 42);\n  =}\n}\n
    \n
    target Cpp\nreactor Inside {\n  input x: int\n  reaction(x) {=\n    std::cout << "Received " << std::to_string(*x.get()) << std::endl;\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x.set(42);\n  =}\n}\n
    \n
    target Python\nreactor Inside {\n  input x\n  reaction(x) {=\n    print(f"Received {x.value}")\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x.set(42);\n  =}\n}\n
    \n
    target TypeScript\nreactor Inside {\n  input x: number\n  reaction(x) {=\n    console.log("Received ${x}");\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    i.x = 42\n  =}\n}\n
    \n
    target Rust\nreactor Inside {\n  input x: u32\n  reaction(x) {=\n    println!("Received {}", ctx.get(x).unwrap());\n  =}\n}\nmain reactor {\n  i = new Inside()\n  reaction(startup) -> i.x {=\n    ctx.set(i__x, 42);\n  =}\n}\n
    \n

    $end(Triggering)$

    \n

    The reaction to $startup$ declares the input port of the inside reactor as an effect and then sets it with value 42.\nThis will cause the inside reactor’s reaction to execute and print Received 42.

    \n

    Startup, Shutdown, and Reset Reactions

    \n

    Reactions may be triggered by the special events $startup$, $shutdown$, or $reset$.\nFor example,

    \n
      reaction(startup) {=\n    // ... Do something\n  =}\n
    \n

    A reaction to $startup$ is triggered at the very first tag when the program begins (or, if within a mode of a modal reactor, when the mode is first entered).\nThis reaction will be logically simultaneous with reactions to timers that have a zero offset.\nAs usual, for logically simultaneous reactions declared within the same reactor, the order in which they are invoked will be governed by the order in which they are declared.

    \n

    A reaction to $shutdown$ is invoked at program termination.\nSee the Termination section for details.

    \n
    \n

    Reactions to the $reset$ event are not supported in $target-language$ because modal reactors are not supported.

    \n
    \n
    \n

    A reaction to the $reset$ event is invoked if the reaction or reactor is within a mode of a modal reactor and the mode is entered via a reset transition.\nFor details, see the Modal Reactors section.

    \n
    \n

    Bodyless Reactions

    \n

    Sometimes, it is inconvenient to mix Lingua Franca code with target code. Rather than defining reactions (i.e., complete with inlined target code), it is also possible to just declare them and provide implementations in a separate file. The syntax of reaction declarations is the same as for reaction definitions, except they have no implementation. Reaction declarations can be thought of as function prototypes.

    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target C {\n  cmake-include: ["hello.cmake"],\n  files: ["hello.c"]\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The cmake-include target property is used to make the build system aware of an externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.c)\n
    \n

    The files target property is used to make accessible the file that has the implementation in `hello.c,\nwhich could look something like this:

    \n
    #include <stdio.h>\n#include "../include/HelloDecl/HelloDecl.h"\n\nvoid hello(hellodecl_self_t* self) {\n    printf("Hello declaration!\\n");\n}\n
    \n

    File Structure

    \n

    In the above example, the C file uses #include to import a file called HelloDecl.h. The HelloDecl.h file\nis generated from the Lingua Franca source file when the LF program is compiled. The file\nHelloDecl.h is named after the main reactor, which is called HelloDecl, and its parent\ndirectory, include/HelloDecl, is named after the file, HelloDecl.lf.

    \n

    In general, compiling a Lingua Franca program that uses reaction declarations will always generate a\ndirectory in the include directory for each file in the program. This directory will contain a\nheader file for each reactor that is defined in the file.

    \n

    As another example, if an LF program consists of files F1.lf and F2.lf, where F1.lf defines reactors\nA and B and F2.lf defines the reactor C and the main reactor F2, then the directory structure\nwill look something like this:

    \n
    include/\n├ F1/\n│ ├ A.h\n│ └ B.h\n└ F2/\n  ├ C.h\n  └ F2.h\nsrc/\n├ F1.lf  // defines A and B\n└ F2.lf  // defines C and F2\nsrc-gen/
    \n

    There is no particular location where you are required to place your C files or your CMake files.\nFor example, you may choose to place them in a directory called c that is a sibling of the src\ndirectory.

    \n

    The Generated Header Files

    \n

    The generated header files are necessary in order to separate your C code from your LF code because\nthey describe the signatures of the reaction functions that you must implement.

    \n

    In addition, they define structs that will be referenced by the reaction bodies. This includes the\nself struct of the reactor to which the header file corresponds, as well as structs for its ports,\nits actions, and the ports of its child reactors.

    \n

    As with preambles, programmer discipline is required to avoid breaking the deterministic semantics\nof Lingua Franca. In particular, although the information exposed in these header files allows\nregular C code to operate on ports and self structs, such information must not be saved in global or\nstatic variables.

    \n

    Linking Your C Code

    \n

    As with any Lingua Franca project that uses external C files, projects involving external reactions\nmust use the cmake-include target property to link those files into the main target.

    \n

    The file referenced by the cmake-include target property has the following syntax:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE <files>)\n
    \n

    where <files> is a list of the C files you need to link, with paths given relative to the project\nroot (the parent of the src directory).

    \n
    \n
    \n

    Example

    \n

    Consider the following program that has a single reaction named hello and is triggered at startup.\nIt has no implementation.

    \n
    target Cpp {\n  cmake-include: ["hello.cmake"],\n}\n\nmain reactor HelloDecl {\n\n  reaction hello(startup)\n\n}\n
    \n

    The behavior of the hello reaction is provided using a method definition in an external C++ file hello.cc.

    \n
    #include "HelloDecl/HelloDecl.hh" // include the code generated reactor class\n\n// define the reaction implementation\nvoid HelloDecl::Inner::hello([[maybe_unused]] const reactor::StartupTrigger& startup) {\n  std::cout << "Hello World." << std::endl;\n}\n
    \n

    Using the cmake-include target property, we can make the build system aware of this externally supplied implementation. The contents of hello.cmake is as follows:

    \n
    target_sources(${LF_MAIN_TARGET} PRIVATE hello.cc)\n
    \n

    Note that this mechanism can be used to add arbitrary additional resources such as additional headers and implementation files or 3rd party libraries to the compilation.

    \n

    Header Files and Method Signatures

    \n

    In order to provide an implementation of a reaction method, it is important to know the header file that declares the reactor class as well as the precise signature of the method implementing the reaction body.

    \n

    The LF compiler generates a header file for each reactor that gets defined in LF code. The header file is named after the corresponding reactor and prefixed by the path to the LF file that defines the reactor. Consider the following example project structure:

    \n
    src/\n├ A.lf   // defines Foo\n└ sub/\n  └ B.lf // defines Bar
    \n

    In this case, the compiler will generate two header files A/Foo.hh and sub/B/Bar.hh, which would need to be included by an external implementation file.

    \n

    The precise method signature depends on the name of the reactor, the name of the reactions, and the precise triggers, sources, and effects that are defined in the LF code.\nThe return type is always void. A reaction foo in a reactor Bar will be named Bar::Inner::foo. Note that each reactor class in the C++ target defines an Inner class which contains all reactions as well as the parameters and state variables. This is done to deliberately restrict the scope of reaction bodies in order to avoid accidental violations of reactor semantics.\nAny declared triggers, sources or effects are given to the reaction method via method arguments. The precise arguments and their types depend on the LF code. If in doubt, please check the signature used in the generated header file under src-gen/<lf-file>, where <lf-file> corresponds to the LF file that you are compiling.

    \n
    \n
    \n

    The $target-language$ target does not currently support reaction declarations.

    \n
    ","headings":[{"value":"Reaction Declaration","depth":2},{"value":"Reaction Order","depth":2},{"value":"Overwriting Outputs","depth":2},{"value":"Reacting to Outputs of Contained Reactors","depth":2},{"value":"Triggering Contained Reactors","depth":2},{"value":"Startup, Shutdown, and Reset Reactions","depth":2},{"value":"Bodyless Reactions","depth":2},{"value":"Example","depth":3},{"value":"File Structure","depth":3},{"value":"The Generated Header Files","depth":3},{"value":"Linking Your C Code","depth":3},{"value":"Example","depth":3},{"value":"Header Files and Method Signatures","depth":3}],"frontmatter":{"permalink":"/docs/handbook/reactions","title":"Reactions","oneline":"Reactions in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","permalink":"/docs/handbook/composing-reactors"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Methods","oneline":"Methods in Lingua Franca.","permalink":"/docs/handbook/methods"}}}},"pageContext":{"id":"1-reactions","slug":"/docs/handbook/reactions","repoPath":"/packages/documentation/copy/en/topics/Reactions.md","previousID":"834f9d0d-f7c6-5732-8c60-bad1954701f7","nextID":"c2fc2a0b-a389-5c48-9cb0-c079bf2e6034","lang":"en","modifiedTime":"2023-11-08T00:42:45.532Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/reactors-on-patmos/page-data.json b/page-data/docs/handbook/reactors-on-patmos/page-data.json index 0ed7cfbb5..1aeaccb2c 100644 --- a/page-data/docs/handbook/reactors-on-patmos/page-data.json +++ b/page-data/docs/handbook/reactors-on-patmos/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reactors-on-patmos","result":{"data":{"markdownRemark":{"id":"1b9eb6de-b3ca-54c1-bc79-972ea7a0fd22","excerpt":"Reactors on Patmos Reactors can be executed on Patmos, a bare-metal execution platform\nthat is optimized for time-predictable execution. Well written C programs…","html":"

    Reactors on Patmos

    \n

    Reactors can be executed on Patmos, a bare-metal execution platform\nthat is optimized for time-predictable execution. Well written C programs can be analyzed for their\nworst-case execution time (WCET).

    \n

    Compiling and Running Reactors

    \n

    Patmos can run in an FPGA, but there are also two\nsimulators available:

    \n
      \n
    1. pasim a software ISA simulator that is written in C++.
    2. \n
    3. patemu a cycle-accurate hardware emulator generated from the hardware description.
    4. \n
    \n

    To execute reactions on Patmos, the Patmos toolchain needs\nto be installed. The web page contains a quick start, detailed information including how to\nperform WCET analysis is available in the\nPatmos Reference Handbook.

    \n

    To execute the “hello world” reactor on Patmos use the LF compiler to generate the C code.\nCompile the reactor with the Patmos compiler (in src-gen):

    \n
    patmos-clang Minimal.c -o Minimal.elf
    \n

    The reactor can be executed on the SW simulator with:

    \n
    pasim Minimal.elf
    \n

    As Patmos is a bare metal runtime that has no notion of calendar time, its start time\nis considered the epoch and the following output will be observed:

    \n
    Start execution at time Thu Jan  1 00:00:00 1970\nplus 640000 nanoseconds.\nHello World.\nElapsed logical time (in nsec): 0\nElapsed physical time (in nsec): 3970000
    \n

    The reactor can also be executed on the hardware emulator of Patmos:

    \n
    patemu Minimal.elf
    \n

    This execution is considerably slower than the SW simulator, as the concrete hardware\nof Patmos is simulated cycle-accurate.

    \n

    Worst-Case Execution Time Analysis

    \n

    Following example is a code fragment from\nWcet.lf.

    \n
    reactor Work {\n    input in1: int;\n    input in2: int;\n    output out:int;\n    reaction(in1, in2) -> out {=\n    \tint ret;\n    \tif (in1 > 10) {\n    \t\tret = in2 * in1;\n    \t} else {\n    \t\tret = in2 + in1;\n    \t}\n        lf_set(out, ret);\n    =}\n}\n
    \n

    We want to perform WCET analysis of the single reaction of the Work reactor.\nThis reaction, depending on the input data, will either perform a multiplication,\nwhich is more expensive in Patmos, or an addition. The WCET analysis shall consider\nthe multiplication path as the worst-case path. To generate the information for\nWCET analysis by the compiler we have to compile the application as follows:

    \n
    patmos-clang -O2 -mserialize=wcet.pml Wcet.c
    \n

    We investigate the C source code Wcet.c and find that the reaction we\nare interested is named reaction_function1. Therefore, we invoke WCET analysis\nas follows:

    \n
    platin wcet -i wcet.pml -b a.out -e reaction_function1 --report
    \n

    This results in following report:

    \n
    ...\n[platin] INFO: Finished run WCET analysis (platin)          in 62 ms\n[platin] INFO: best WCET bound: 242 cycles\n---\n- analysis-entry: reaction_function1\n  source: platin\n  cycles: 242\n...
    \n

    The analysis gives the WCET of 242 clock cycles for the reaction,\nwhich includes clock cycles for data cache misses.\nFurther details on the WCET analysis\ntool platin and e.g., how to annotate loop bounds can be found in the\nPatmos Reference Handbook.

    \n

    Note, that the WCET analysis of a reaction does only include the code of the\nreaction function, not the cache miss cost of calling the function from\nthe scheduler or the cache miss cost when returning to the scheduler.

    ","headings":[{"value":"Reactors on Patmos","depth":2},{"value":"Compiling and Running Reactors","depth":3},{"value":"Worst-Case Execution Time Analysis","depth":3}],"frontmatter":{"permalink":"/docs/handbook/reactors-on-patmos","title":"Reactors on Patmos","oneline":"Reactors on Patmos (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/reactors-on-patmos","repoPath":"/packages/documentation/copy/en/preliminary/Reactors on Patmos.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/reactors-on-patmos","result":{"data":{"markdownRemark":{"id":"1b9eb6de-b3ca-54c1-bc79-972ea7a0fd22","excerpt":"Reactors on Patmos Reactors can be executed on Patmos, a bare-metal execution platform\nthat is optimized for time-predictable execution. Well written C programs…","html":"

    Reactors on Patmos

    \n

    Reactors can be executed on Patmos, a bare-metal execution platform\nthat is optimized for time-predictable execution. Well written C programs can be analyzed for their\nworst-case execution time (WCET).

    \n

    Compiling and Running Reactors

    \n

    Patmos can run in an FPGA, but there are also two\nsimulators available:

    \n
      \n
    1. pasim a software ISA simulator that is written in C++.
    2. \n
    3. patemu a cycle-accurate hardware emulator generated from the hardware description.
    4. \n
    \n

    To execute reactions on Patmos, the Patmos toolchain needs\nto be installed. The web page contains a quick start, detailed information including how to\nperform WCET analysis is available in the\nPatmos Reference Handbook.

    \n

    To execute the “hello world” reactor on Patmos use the LF compiler to generate the C code.\nCompile the reactor with the Patmos compiler (in src-gen):

    \n
    patmos-clang Minimal.c -o Minimal.elf
    \n

    The reactor can be executed on the SW simulator with:

    \n
    pasim Minimal.elf
    \n

    As Patmos is a bare metal runtime that has no notion of calendar time, its start time\nis considered the epoch and the following output will be observed:

    \n
    Start execution at time Thu Jan  1 00:00:00 1970\nplus 640000 nanoseconds.\nHello World.\nElapsed logical time (in nsec): 0\nElapsed physical time (in nsec): 3970000
    \n

    The reactor can also be executed on the hardware emulator of Patmos:

    \n
    patemu Minimal.elf
    \n

    This execution is considerably slower than the SW simulator, as the concrete hardware\nof Patmos is simulated cycle-accurate.

    \n

    Worst-Case Execution Time Analysis

    \n

    Following example is a code fragment from\nWcet.lf.

    \n
    reactor Work {\n    input in1: int;\n    input in2: int;\n    output out:int;\n    reaction(in1, in2) -> out {=\n    \tint ret;\n    \tif (in1 > 10) {\n    \t\tret = in2 * in1;\n    \t} else {\n    \t\tret = in2 + in1;\n    \t}\n        lf_set(out, ret);\n    =}\n}\n
    \n

    We want to perform WCET analysis of the single reaction of the Work reactor.\nThis reaction, depending on the input data, will either perform a multiplication,\nwhich is more expensive in Patmos, or an addition. The WCET analysis shall consider\nthe multiplication path as the worst-case path. To generate the information for\nWCET analysis by the compiler we have to compile the application as follows:

    \n
    patmos-clang -O2 -mserialize=wcet.pml Wcet.c
    \n

    We investigate the C source code Wcet.c and find that the reaction we\nare interested is named reaction_function1. Therefore, we invoke WCET analysis\nas follows:

    \n
    platin wcet -i wcet.pml -b a.out -e reaction_function1 --report
    \n

    This results in following report:

    \n
    ...\n[platin] INFO: Finished run WCET analysis (platin)          in 62 ms\n[platin] INFO: best WCET bound: 242 cycles\n---\n- analysis-entry: reaction_function1\n  source: platin\n  cycles: 242\n...
    \n

    The analysis gives the WCET of 242 clock cycles for the reaction,\nwhich includes clock cycles for data cache misses.\nFurther details on the WCET analysis\ntool platin and e.g., how to annotate loop bounds can be found in the\nPatmos Reference Handbook.

    \n

    Note, that the WCET analysis of a reaction does only include the code of the\nreaction function, not the cache miss cost of calling the function from\nthe scheduler or the cache miss cost when returning to the scheduler.

    ","headings":[{"value":"Reactors on Patmos","depth":2},{"value":"Compiling and Running Reactors","depth":3},{"value":"Worst-Case Execution Time Analysis","depth":3}],"frontmatter":{"permalink":"/docs/handbook/reactors-on-patmos","title":"Reactors on Patmos","oneline":"Reactors on Patmos (preliminary)","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/reactors-on-patmos","repoPath":"/packages/documentation/copy/en/preliminary/Reactors on Patmos.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/regression-tests/page-data.json b/page-data/docs/handbook/regression-tests/page-data.json index 1a13d19be..8cc541511 100644 --- a/page-data/docs/handbook/regression-tests/page-data.json +++ b/page-data/docs/handbook/regression-tests/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/regression-tests","result":{"data":{"markdownRemark":{"id":"96f63950-6ca6-530f-9c7a-ed0605046575","excerpt":"Lingua Franca comes with an extensive set of regression tests that are executed on various platforms automatically whenever an update is pushed to the LF…","html":"

    Lingua Franca comes with an extensive set of regression tests that are executed on various platforms automatically whenever an update is pushed to the LF repository. There are two categories of tests:

    \n
      \n
    • Unit tests are Java or Kotlin methods in our code base that are labeled with the @Test directive. These tests check individual functions of the code generation infrastructure. These are located in the src/test directory of each subroject within the repository.
    • \n
    • Integration tests are complete Lingua Franca programs that are compiled and executed automatically. A test passes if it successfully compiles and runs to completion with normal termination (return code 0). These tests are located in the test directory at the root of the LF repo, with one subdirectory per target language.
    • \n
    \n

    Their implementation can be found in the core/src/integrationTest directory.\nThe integration tests are also executed through JUnit using methods with @Test directives, but they are executed separately.

    \n

    Running the Tests From the Command Line

    \n

    To run all unit tests, simply run ./gradlew test. Note that also the normal build tasks ./gradlew build runs all the unit tests.

    \n

    The integration tests can be run using the integrationTest task. However, typically it is not desired to run all tests for all targets locally as it will need the right target tooling and will take a long time.

    \n

    To run only the integration tests for one target, we provide the targetTest gradle task. For instance, you can use the following command to run all Rust tests:

    \n
    ./gradlew targetTest -Ptarget=Rust
    \n

    You can specify any valid target. If you run the task without specifying the target property ./gradlew targetTest it will produce an error message and list all available targets.

    \n

    The targetTest task is essentially a convenient shortcut for the following:

    \n
    ./gradew core:integrationTest --test org.lflang.tests.runtime.<target>Test.*
    \n

    If you prefer have more control over which tests are executed, you can also use this more verbose version.

    \n

    It is also possible to run a subset of the tests. For example, the C tests are organized into the following categories:

    \n
      \n
    • generic tests are .lf files located in $LF/test/C/src.
    • \n
    • concurrent tests are .lf files located in $LF/test/C/src/concurrent.
    • \n
    • federated tests are .lf files located in $LF/test/C/src/federated.
    • \n
    • multiport tests are .lf files located in $LF/test/C/src/multiport.
    • \n
    \n

    To invoke only the C tests in the concurrent category, for example, run this:

    \n
    ./gradlew core:integrationTest --tests org.lflang.tests.runtime.CTest.runConcurrentTests
    \n

    Sometimes it is convenient to only run a single specific test case. This can be done with the singleTest task. For instance:

    \n
    ./gradlew singleTest -DsingleTest=test/C/src/Minimal.lf
    \n

    Reporting Bugs

    \n

    If you encounter a bug or add some enhancement to Lingua Franca, then you should create a regression test either as a system test or a unit test and issue a pull request. System tests are particularly easy to create since they are simply Lingua Franca programs that either compile and execute successfully (the test passes) or fail either to compile or execute.

    \n

    Testing Architecture

    \n

    System tests can be put in any subdirectory of $LF/test or $LF/example.\nAny .lf file within these directories will be treated as a system test unless they are within a directory named failing, in which case they will be ignored.\nThe tests are automatically indexed by our JUnit-based test infrastructure, which is located in the package core/src/integrationTest. Each target has its own class in the runtime package, with a number of test methods that correspond to particular test categories, such as generic, concurrent, federated, etc. A test can be associated with a particular category by placing it in a directory that matches its name. For instance, we can create a test (e.g., Foo.lf) in test/C/src/concurrent, which will then get indexed under the target C in the category concurrent. Files placed directly in test/C/src will be considered generic C tests, and a file in a directory concurrent/federated will be indexed as federated (corresponding to the nearest containing directory).

    \n

    Caution: adding a new category requires updating an enum in TestRegistry.java and adding a @Test-labeled method to TestBase.

    \n

    Known Failures

    \n

    Sometimes it is useful to retain tests that have a known failure that should be addressed at a later point. Such tests can simply be put in a directory called failing, which will tell our test indexing code to exclude it.

    \n

    Test Output

    \n

    Tests are grouped by target and category. It is also reported when, for a given category, there are other targets that feature tests that are missing for the target under test. Tests that either do not have a main reactor or are marked as known failures are reported as “ignored.” For all the tests that were successfully indexed, it is reported how many passed. For each failing test, diagnostics are reported that should help explain the failure. Here is some sample output for Ctest.runConcurrentTests, which runs tests categorized as concurrent for the C target:

    \n
    CTest > runConcurrentTests() STANDARD_OUT\n    ==============================================================================\n    Target: C\n    Description: Run concurrent tests.\n    ==============================================================================\n\n    ==============================================================================\n    Category: CONCURRENT\n    ==============================================================================\n    ------------------------------------------------------------------------------\n    Ignored: 0\n    ------------------------------------------------------------------------------\n\n    ------------------------------------------------------------------------------\n    Covered: 29/33\n    ------------------------------------------------------------------------------\n    Missing: src/concurrent/BankToBank.lf\n    Missing: src/concurrent/BankToBankMultiport.lf\n    Missing: src/concurrent/BankToBankMultiportAfter.lf\n    Missing: src/concurrent/BankToMultiport.lf\n\n    ------------------------------------------------------------------------------\n    Passing: 29/29\n    ------------------------------------------------------------------------------\n
    \n

    Code Coverage

    \n

    Code coverage is automatically recorded when running tests.\nA combined report for each subproject can be created by running ./gradlew jacocoTestReport.\nFor the core subproject, the html report will be located in build/reports/html/index.html.\nNote that this report will only reflect the coverage of the test that have actually executed.

    \n

    Continuous Integration

    \n

    Each push or pull request will trigger all tests to be run on GitHub Actions. It’s configuration can be found here.

    ","headings":[{"value":"Running the Tests From the Command Line","depth":2},{"value":"Reporting Bugs","depth":2},{"value":"Testing Architecture","depth":2},{"value":"Known Failures","depth":3},{"value":"Test Output","depth":3},{"value":"Code Coverage","depth":2},{"value":"Continuous Integration","depth":2}],"frontmatter":{"permalink":"/docs/handbook/regression-tests","title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","permalink":"/docs/handbook/intellij"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Running Benchmarks","oneline":"Running Benchmarks.","permalink":"/docs/handbook/running-benchmarks"}}}},"pageContext":{"id":"5-regression-tests","slug":"/docs/handbook/regression-tests","repoPath":"/packages/documentation/copy/en/developer/Regression Tests.md","previousID":"1d9f0442-2300-5615-9c04-6ee5f2c33793","nextID":"8d78ab4e-cebd-5116-bbe9-871de58f9aeb","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/regression-tests","result":{"data":{"markdownRemark":{"id":"96f63950-6ca6-530f-9c7a-ed0605046575","excerpt":"Lingua Franca comes with an extensive set of regression tests that are executed on various platforms automatically whenever an update is pushed to the LF…","html":"

    Lingua Franca comes with an extensive set of regression tests that are executed on various platforms automatically whenever an update is pushed to the LF repository. There are two categories of tests:

    \n
      \n
    • Unit tests are Java or Kotlin methods in our code base that are labeled with the @Test directive. These tests check individual functions of the code generation infrastructure. These are located in the src/test directory of each subroject within the repository.
    • \n
    • Integration tests are complete Lingua Franca programs that are compiled and executed automatically. A test passes if it successfully compiles and runs to completion with normal termination (return code 0). These tests are located in the test directory at the root of the LF repo, with one subdirectory per target language.
    • \n
    \n

    Their implementation can be found in the core/src/integrationTest directory.\nThe integration tests are also executed through JUnit using methods with @Test directives, but they are executed separately.

    \n

    Running the Tests From the Command Line

    \n

    To run all unit tests, simply run ./gradlew test. Note that also the normal build tasks ./gradlew build runs all the unit tests.

    \n

    The integration tests can be run using the integrationTest task. However, typically it is not desired to run all tests for all targets locally as it will need the right target tooling and will take a long time.

    \n

    To run only the integration tests for one target, we provide the targetTest gradle task. For instance, you can use the following command to run all Rust tests:

    \n
    ./gradlew targetTest -Ptarget=Rust
    \n

    You can specify any valid target. If you run the task without specifying the target property ./gradlew targetTest it will produce an error message and list all available targets.

    \n

    The targetTest task is essentially a convenient shortcut for the following:

    \n
    ./gradew core:integrationTest --test org.lflang.tests.runtime.<target>Test.*
    \n

    If you prefer have more control over which tests are executed, you can also use this more verbose version.

    \n

    It is also possible to run a subset of the tests. For example, the C tests are organized into the following categories:

    \n
      \n
    • generic tests are .lf files located in $LF/test/C/src.
    • \n
    • concurrent tests are .lf files located in $LF/test/C/src/concurrent.
    • \n
    • federated tests are .lf files located in $LF/test/C/src/federated.
    • \n
    • multiport tests are .lf files located in $LF/test/C/src/multiport.
    • \n
    \n

    To invoke only the C tests in the concurrent category, for example, run this:

    \n
    ./gradlew core:integrationTest --tests org.lflang.tests.runtime.CTest.runConcurrentTests
    \n

    Sometimes it is convenient to only run a single specific test case. This can be done with the singleTest task. For instance:

    \n
    ./gradlew singleTest -DsingleTest=test/C/src/Minimal.lf
    \n

    Reporting Bugs

    \n

    If you encounter a bug or add some enhancement to Lingua Franca, then you should create a regression test either as a system test or a unit test and issue a pull request. System tests are particularly easy to create since they are simply Lingua Franca programs that either compile and execute successfully (the test passes) or fail either to compile or execute.

    \n

    Testing Architecture

    \n

    System tests can be put in any subdirectory of $LF/test or $LF/example.\nAny .lf file within these directories will be treated as a system test unless they are within a directory named failing, in which case they will be ignored.\nThe tests are automatically indexed by our JUnit-based test infrastructure, which is located in the package core/src/integrationTest. Each target has its own class in the runtime package, with a number of test methods that correspond to particular test categories, such as generic, concurrent, federated, etc. A test can be associated with a particular category by placing it in a directory that matches its name. For instance, we can create a test (e.g., Foo.lf) in test/C/src/concurrent, which will then get indexed under the target C in the category concurrent. Files placed directly in test/C/src will be considered generic C tests, and a file in a directory concurrent/federated will be indexed as federated (corresponding to the nearest containing directory).

    \n

    Caution: adding a new category requires updating an enum in TestRegistry.java and adding a @Test-labeled method to TestBase.

    \n

    Known Failures

    \n

    Sometimes it is useful to retain tests that have a known failure that should be addressed at a later point. Such tests can simply be put in a directory called failing, which will tell our test indexing code to exclude it.

    \n

    Test Output

    \n

    Tests are grouped by target and category. It is also reported when, for a given category, there are other targets that feature tests that are missing for the target under test. Tests that either do not have a main reactor or are marked as known failures are reported as “ignored.” For all the tests that were successfully indexed, it is reported how many passed. For each failing test, diagnostics are reported that should help explain the failure. Here is some sample output for Ctest.runConcurrentTests, which runs tests categorized as concurrent for the C target:

    \n
    CTest > runConcurrentTests() STANDARD_OUT\n    ==============================================================================\n    Target: C\n    Description: Run concurrent tests.\n    ==============================================================================\n\n    ==============================================================================\n    Category: CONCURRENT\n    ==============================================================================\n    ------------------------------------------------------------------------------\n    Ignored: 0\n    ------------------------------------------------------------------------------\n\n    ------------------------------------------------------------------------------\n    Covered: 29/33\n    ------------------------------------------------------------------------------\n    Missing: src/concurrent/BankToBank.lf\n    Missing: src/concurrent/BankToBankMultiport.lf\n    Missing: src/concurrent/BankToBankMultiportAfter.lf\n    Missing: src/concurrent/BankToMultiport.lf\n\n    ------------------------------------------------------------------------------\n    Passing: 29/29\n    ------------------------------------------------------------------------------\n
    \n

    Code Coverage

    \n

    Code coverage is automatically recorded when running tests.\nA combined report for each subproject can be created by running ./gradlew jacocoTestReport.\nFor the core subproject, the html report will be located in build/reports/html/index.html.\nNote that this report will only reflect the coverage of the test that have actually executed.

    \n

    Continuous Integration

    \n

    Each push or pull request will trigger all tests to be run on GitHub Actions. It’s configuration can be found here.

    ","headings":[{"value":"Running the Tests From the Command Line","depth":2},{"value":"Reporting Bugs","depth":2},{"value":"Testing Architecture","depth":2},{"value":"Known Failures","depth":3},{"value":"Test Output","depth":3},{"value":"Code Coverage","depth":2},{"value":"Continuous Integration","depth":2}],"frontmatter":{"permalink":"/docs/handbook/regression-tests","title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Developer IntelliJ Setup","oneline":"Developer IntelliJ Setup.","permalink":"/docs/handbook/intellij"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Running Benchmarks","oneline":"Running Benchmarks.","permalink":"/docs/handbook/running-benchmarks"}}}},"pageContext":{"id":"5-regression-tests","slug":"/docs/handbook/regression-tests","repoPath":"/packages/documentation/copy/en/developer/Regression Tests.md","previousID":"1d9f0442-2300-5615-9c04-6ee5f2c33793","nextID":"8d78ab4e-cebd-5116-bbe9-871de58f9aeb","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/related-work/page-data.json b/page-data/docs/handbook/related-work/page-data.json index acd3d0719..31281ad65 100644 --- a/page-data/docs/handbook/related-work/page-data.json +++ b/page-data/docs/handbook/related-work/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/related-work","result":{"data":{"markdownRemark":{"id":"9b70f0f3-50a7-5c9f-a0fd-7a858bff6a59","excerpt":"Lingua Franca is focused more on using the best ideas than on being innovative.\nHere, we list most closely related work first, then other work with which it may…","html":"

    Lingua Franca is focused more on using the best ideas than on being innovative.\nHere, we list most closely related work first, then other work with which it may be useful to contrast.

    \n

    Software Frameworks

    \n
      \n
    • \n

      Rubus.

      \n
    • \n
    • \n

      Akka framework for distributed Fog computing.

      \n
    • \n
    • \n

      Accessors, from Berkeley, a JavaScript-based framework for IoT: This framework is the most direct inspiration for Lingua Franca. The idea behind accessors is to componentize IoT resources by encapsulating them in actors. As such, their interactions can be coordinated under a discrete event semantics paper.

      \n
    • \n
    • \n

      Rebecca.

      \n
    • \n
    • \n

      The Kiel Integrated Environment for Layout Eclipse Rich Client (KIELER) is a graphical environment for programming using SCCharts (see the 2014 PLDI paper).

      \n
    • \n
    • \n

      RTMAPS: From Intempora. It has a graphical syntax in a UI and advertises “data is acquired asynchronously and each data sample is captured along with its time stamp at its own pace.” You can build your own blocks in C++ or Python. It does, however, look like its not deterministic.

      \n
    • \n
    \n

    Usage of the Term Reactor

    \n\n

    Other Pointers

    \n
      \n
    • Reactive Manifesto: Version 2.0, Published in 2014, this position paper defines Reactive Systems as those that are Responsive, Resilient, Elastic and Message Driven.
    • \n
    \n

    Academic Projects

    \n
      \n
    • \n

      I/O Automata, from MIT, is a formalism that could be used to model the semantics of Lingua Franca. Timed I/O Automata FIXME: link extend I/O Automata with temporal semantics. They share with LF the notion of reactions to input messages and internal events that change the state of an actor and produce outputs. The behavior of a component is given as a state machine.

      \n
    • \n
    • \n

      FIXME Hewitt actors.

      \n
    • \n
    • \n

      SyncCharts: By Charles André. See the 1996 technical report.

      \n
    • \n
    • \n

      ReactiveML: Website

      \n
    • \n
    \n

    Contrasting Work

    \n
      \n
    • \n

      CAPH (a recursive acronym for CAPH Ain’t plain HDL), a hardware description language from CNRS, is a fine-grained dataflow language for compiling into FPGAs. The language has no temporal semantics, and although it has a notion of firing rules, it is not clear which of the many variants of dataflow is realized nor whether the MoC is deterministic. The paper does not cite any of the prior work on dataflow MoCs.

      \n
    • \n
    • \n

      Robot Operating System (ROS), an open-source project originally from Willow Garage: ROS provides a publish-and-subscribe server for interaction between components. Version 1 has no timing properties at all. Version 2 has some timing properties such as priorities, but it makes no effort to be deterministic.

      \n
    • \n
    • \n

      RADLER framework from SRI, which is based on a publish-and-subscribe architecture similar to ROS. It introduces some timing constructs such as periodic execution and scheduling constraints, but it makes no effort to be deterministic.

      \n
    • \n
    ","headings":[{"value":"Software Frameworks","depth":2},{"value":"Usage of the Term Reactor","depth":3},{"value":"Other Pointers","depth":3},{"value":"Academic Projects","depth":2},{"value":"Contrasting Work","depth":2}],"frontmatter":{"permalink":"/docs/handbook/related-work","title":"Related Work","oneline":"Related Work","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/related-work","repoPath":"/packages/documentation/copy/en/less-developed/Related Work.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/related-work","result":{"data":{"markdownRemark":{"id":"9b70f0f3-50a7-5c9f-a0fd-7a858bff6a59","excerpt":"Lingua Franca is focused more on using the best ideas than on being innovative.\nHere, we list most closely related work first, then other work with which it may…","html":"

    Lingua Franca is focused more on using the best ideas than on being innovative.\nHere, we list most closely related work first, then other work with which it may be useful to contrast.

    \n

    Software Frameworks

    \n
      \n
    • \n

      Rubus.

      \n
    • \n
    • \n

      Akka framework for distributed Fog computing.

      \n
    • \n
    • \n

      Accessors, from Berkeley, a JavaScript-based framework for IoT: This framework is the most direct inspiration for Lingua Franca. The idea behind accessors is to componentize IoT resources by encapsulating them in actors. As such, their interactions can be coordinated under a discrete event semantics paper.

      \n
    • \n
    • \n

      Rebecca.

      \n
    • \n
    • \n

      The Kiel Integrated Environment for Layout Eclipse Rich Client (KIELER) is a graphical environment for programming using SCCharts (see the 2014 PLDI paper).

      \n
    • \n
    • \n

      RTMAPS: From Intempora. It has a graphical syntax in a UI and advertises “data is acquired asynchronously and each data sample is captured along with its time stamp at its own pace.” You can build your own blocks in C++ or Python. It does, however, look like its not deterministic.

      \n
    • \n
    \n

    Usage of the Term Reactor

    \n\n

    Other Pointers

    \n
      \n
    • Reactive Manifesto: Version 2.0, Published in 2014, this position paper defines Reactive Systems as those that are Responsive, Resilient, Elastic and Message Driven.
    • \n
    \n

    Academic Projects

    \n
      \n
    • \n

      I/O Automata, from MIT, is a formalism that could be used to model the semantics of Lingua Franca. Timed I/O Automata FIXME: link extend I/O Automata with temporal semantics. They share with LF the notion of reactions to input messages and internal events that change the state of an actor and produce outputs. The behavior of a component is given as a state machine.

      \n
    • \n
    • \n

      FIXME Hewitt actors.

      \n
    • \n
    • \n

      SyncCharts: By Charles André. See the 1996 technical report.

      \n
    • \n
    • \n

      ReactiveML: Website

      \n
    • \n
    \n

    Contrasting Work

    \n
      \n
    • \n

      CAPH (a recursive acronym for CAPH Ain’t plain HDL), a hardware description language from CNRS, is a fine-grained dataflow language for compiling into FPGAs. The language has no temporal semantics, and although it has a notion of firing rules, it is not clear which of the many variants of dataflow is realized nor whether the MoC is deterministic. The paper does not cite any of the prior work on dataflow MoCs.

      \n
    • \n
    • \n

      Robot Operating System (ROS), an open-source project originally from Willow Garage: ROS provides a publish-and-subscribe server for interaction between components. Version 1 has no timing properties at all. Version 2 has some timing properties such as priorities, but it makes no effort to be deterministic.

      \n
    • \n
    • \n

      RADLER framework from SRI, which is based on a publish-and-subscribe architecture similar to ROS. It introduces some timing constructs such as periodic execution and scheduling constraints, but it makes no effort to be deterministic.

      \n
    • \n
    ","headings":[{"value":"Software Frameworks","depth":2},{"value":"Usage of the Term Reactor","depth":3},{"value":"Other Pointers","depth":3},{"value":"Academic Projects","depth":2},{"value":"Contrasting Work","depth":2}],"frontmatter":{"permalink":"/docs/handbook/related-work","title":"Related Work","oneline":"Related Work","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/related-work","repoPath":"/packages/documentation/copy/en/less-developed/Related Work.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/running-benchmarks/page-data.json b/page-data/docs/handbook/running-benchmarks/page-data.json index 7209f7d78..915a9b529 100644 --- a/page-data/docs/handbook/running-benchmarks/page-data.json +++ b/page-data/docs/handbook/running-benchmarks/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/running-benchmarks","result":{"data":{"markdownRemark":{"id":"ac793a6f-ced3-5dd9-972e-a6891620a738","excerpt":"Running Benchmarks The LF repository contains a series of benchmarks in the benchmark directory. There is also a flexible benchmark runner that automates the…","html":"

    Running Benchmarks

    \n

    The LF repository contains a series of benchmarks in the benchmark directory. There is also a flexible benchmark runner that automates the process of running benchmarks for various settings and collecting results from those benchmarks. It is located in benchmark/runner.\nThe runner is written in python and is based on hydra, a tool for dynamically creating hierarchical configurations by composition

    \n

    Prerequisites

    \n

    Install Python dependencies

    \n

    The benchmark runner is written in Python and requires a working Python3 installation. It also requires a few python packages to be installed. Namely, hydra-core, cogapp and pandas.

    \n

    It is recommended to install the dependencies and execute the benchmark runner in a virtual environment. For instance, this can be done with virtualenv:

    \n
    virtualenv ~/virtualenvs/lfrunner -p python3\nsource ~/virtualenvs/lfrunner/bin/activate\n
    \n

    Then the dependencies can be installed by running:

    \n
    pip install -r benchmark/runner/requirements.txt\n
    \n

    Compile lfc

    \n

    For running LF benchmarks, the command-line compiler lfc needs to be built. Simply run

    \n
    bin/build-lfc\n
    \n

    in the root directory of the LF repository.

    \n

    Also, the environment variable LF_PATH needs to be set and point to the location of the LF repository. This needs to be an absolute path.

    \n
    export LF_PATH=/path/to/lf\n
    \n

    Setup Savina

    \n

    Currently all of our benchmarks are ported from the Savina actor benchmark suite. In order to compare our LF implementations with actor based implementation, the Savina benchmark suite needs to be downloaded and compiled. Note that we require a modified version of the Savina suite, that adds support for specifying the number of worker threads and that includes CAF implementations of most benchmarks.

    \n

    To download and build Savina, run the following commands:

    \n
    git clone https://github.com/lf-lang/savina.git\ncd savina\nmvn install\n
    \n

    Building Savina requires a Java 8 JDK. Depending on the local setup, JAVA_HOME might need to be adjusted before running mvn in order to point to the correct JDK.

    \n
    export JAVA_HOME=/path/to/jdk8\n
    \n

    Before invoking the benchmark runner, the environment variable SAVINA_PATH needs to be set and point to the location of the Savina repository using an absolute path.

    \n
    export SAVINA_PATH=/path/to/savina\n
    \n

    CAF

    \n

    To further build the CAF benchmarks, CAF 0.16.5 needs to be downloaded, compiled and installed first:

    \n
    git clone --branch "0.16.5" git@github.com:actor-framework/actor-framework.git\nmkdir actor-framework/build && cd actor-framework/build\ncmake -DCMAKE_INSTALL_PREFIX=<preferred/install/location> ..\nmake install\n
    \n

    Then, from within the Savina directory, the CAF benchmarks can be build:

    \n
    cmake -DCAF_ROOT_DIR=<path/to/caf/install/location> ..\nmake\n
    \n

    The CAF benchmarks are used in these two publications:

    \n\n

    Running a benchmark

    \n

    A benchmark can simply be run by specifying a benchmark and a target. For instance

    \n
    cd benchmark/runner\n./run_benchmark.py benchmark=savina_micro_pingpong target=lf-c\n
    \n

    runs the Ping Pong benchmark from the Savina suite using the C-target of LF. Currently, supported targets are lf-c, lf-cpp, akka, and caf where akka corresponds to the Akka implementation in the original Savina suite and caf corresponds to a implementation using the C++ Actor Framework .

    \n

    The benchmarks can also be configured. The threads and iterations parameters apply to every benchmark and specify the number of worker threads as well as how many times the benchmark should be run. Most benchmarks allow additional parameters. For instance, the Ping Pong benchmark sends a configurable number of pings that be set via the benchmark.params.messages configuration key. Running the Akka version of the Ping Pong benchmark for 1000 messages, 1 thread and 12 iterations could be done like this:

    \n
    ./run_benchmark.py benchmark=savina_micro_pingpong target=akka threads=1 iterations=12 benchmark.params.messages=1000\n
    \n

    Each benchmark run produces an output directory in the scheme outputs/<date>/<time>/ (e.g. outputs/2020-12-17/16-46-16/). This directory contains a files results.csv which contains the measured execution time for each iteration and all the parameters used for running this particular benchmark. The CSV file contains precisely one row per iteration.

    \n

    Running a series of benchmarks (multirun)

    \n

    The runner also allows to automatically run a single benchmark or a series of benchmarks with a range of settings. The multirun feature is simply used by the -m switch. For instance:

    \n
    ./run_benchmark.py -m benchmark=savina_micro_pingpong target="glob(*)" threads=1,2,4 iterations=12 benchmark.params.messages="range(1000000,10000000,1000000)"\n
    \n

    runs the Ping Pong benchmark for all targets using 1, 2 and 4 threads and for a number of messages ranging from 1M to 10M (in 1M steps).

    \n

    This mechanism can also be used to run multiple benchmarks. For instance,

    \n
    ./run_benchmark.py -m benchmark="glob(*)" target="glob(*)" threads=4 iterations=12\n
    \n

    runs all benchmarks for all targets using 4 threads and 12 iterations.

    \n

    The results for a multirun are written to a directory in the scheme multirun/<date>/<time>/<n> (e.g. multirun/2020-12-17/17-11-03/0/) where <n> denotes the particular run. Each of the <n> subdirectories contains a results.csv for this particular run.

    \n

    Collecting results from multirun

    \n

    A second script called collect_results.py provides a convenient way for collecting results from a multirun and merging them into a single CSV file. Simply running

    \n
    ./collect_results.py multirun/<date>/<time>/ out.csv\n
    \n

    collects all results from the particular multirun and stores the merged data structure in out.csv. collect_results.py not only merges the results, but it also calculates minimum, maximum and median execution time for each individual run. The resulting CSV does not contain the measured values of individual iterations anymore and only contains a single row per run. This behavior can be disabled with the --raw command line flag. With the flag set, the results from all runs are merged as say are and the resulting file contains rows for all individual runs, but no minimum, maximum and median values.

    \n

    How it works

    \n

    The benchmark runner itself is actually relatively simple. Most of the complexity is dealt with by hydra. Hydra is a complex and convenient tool for handling configurations. These configurations can be merged from different sources and be overridden via command line arguments as you have seen above. The actual benchmark runner receives the configuration represented as nested dictionaries from hydra. It then executes the benchmarks precisely as instructed by the configuration.

    \n

    The configuration is split into two big parts: the benchmark configuration and the target configuration. The benchmark configuration describes a particular benchmark instance. This is described in more detail in the next section. The target configuration specifies how to run a benchmark for a specific target (e.g. akka, lf-c, lf-cpp). This is not intended to be changed by the user and therefore isn’t explained in detail here. Essentially a benchmark run is split into 5 steps as is outlined in the following. The target configuration precisely specifies what needs to be done in each step

    \n
      \n
    1. copy The command used to copy relevant source files to a temporary directory.
    2. \n
    3. gen The command used to generate a configured LF file. This is intended to apply a code generation tool like cog to the source code in order to make benchmarks parameterized.
    4. \n
    5. compile The command used to compile the benchmark.
    6. \n
    7. run The command used to generate the benchmark.
    8. \n
    9. parser A parser (a python method) that is used to process the output of the benchmark run and that returns the execution times of individual benchmark runs in a list.
    10. \n
    \n

    Adding new benchmarks

    \n

    In order to add new benchmarks, a new configuration file needs to be created in the conf/benchmark subdirectory. Benchmarks may be grouped by the underscore-delimited segments in their file name. For instance, the PingPong benchmark is part of the micro-benchmarks of the Savina suite, and consequently its configuration file is named in conf/benchmark/savina_micro_pingpong.yaml. This allows to later specify benchmark=savina/micro/pingpong on the command line. Below you can see the contents of savina_micro_pingpong.yaml which we will break down in the following.

    \n
    # @package benchmark\nname: "Ping Pong"\nparams:\n  pings: 1000000\n\n# target specific configuration\ntargets:\n  akka:\n    jar: "${savina_path}/target/savina-0.0.1-SNAPSHOT-jar-with-dependencies.jar"\n    class: "edu.rice.habanero.benchmarks.pingpong.PingPongAkkaActorBenchmark"\n    run_args:\n      pings: ["-n", "<value>"]\n  caf:\n    bin: "caf_01_pingpong"\n    run_args:\n      pings: ["-n", "<value>"]\n  lf-cpp:\n    copy_sources:\n      - "${lf_path}/benchmark/Cpp/Savina/src/BenchmarkRunner.lf"\n      - "${lf_path}/benchmark/Cpp/Savina/src/micro"\n    lf_file: "micro/PingPong.lf"\n    binary: "PingPong"\n    gen_args: null\n    run_args:\n      pings: ["--count", "<value>"]\n  lf-c:\n    copy_sources:\n      - "${lf_path}/benchmark/C/Savina/src/micro/PingPong.lf"\n    lf_file: "PingPong.lf"\n    binary: "PingPong"\n    gen_args:\n      pings: ["-D", "count=<value>"]\n
    \n

    The first line # @package benchmark is hydra specific. It specifies that this configuration is part of the benchmark package. Essentially this enables the configuration to be assigned to benchmark on the command line.

    \n
    name: "Ping Pong"\nparams:\n  pings: 1000000\n
    \n

    This part sets the benchmark name to “Ping Pong” and declares that there is one benchmark specific parameter: pings. This configuration also set the default value for pings to 1000000. Note that the params dictionary may specify an arbitrary number of parameters.

    \n

    The remainder of the configuration file contains target specific configurations that provide instructions on how the particular benchmark can be run for the various targets. This block

    \n
    # target specific configuration\ntargets:\n  akka:\n    jar: "${savina_path}/target/savina-0.0.1-SNAPSHOT-jar-with-dependencies.jar"\n    class: "edu.rice.habanero.benchmarks.pingpong.PingPongAkkaActorBenchmark"\n    run_args:\n      pings: ["-n", "<value>"]\n
    \n

    specifies how the benchmark is executed using Akka. The jar and class configuration keys simply instruct the benchmark runner which class in which jar to run. Note that hydra automatically resolves ${savina_path} to the value you set in the SAVINA_PATH environment variable.

    \n

    The run_args configuration key allows specification of further arguments that are added to the command to be executed when running the benchmark. It expects a dictionary, where the keys are names of parameters as specified above in the params configuration key, and the values are a list of arguments to be added to the executed command. In the case of the pings parameter, the Akka implementation of the benchmark expects the -n flag followed by the parameter value. Note that the special string <value> is automatically resolved by the runner to the actual parameter value when executing the command.

    \n

    Instructions for the C++ target are specified as follows.

    \n
    lf-cpp:\n  copy_sources:\n    - "${lf_path}/benchmark/Cpp/Savina/src/BenchmarkRunner.lf"\n    - "${lf_path}/benchmark/Cpp/Savina/src/micro"\n  lf_file: "micro/PingPong.lf"\n  binary: "PingPong"\n  gen_args: null\n  run_args:\n    pings: ["--count", "<value>"]\n
    \n

    For C and C++ programs, we cannot run a precompiled program as it is the case for Akka, but we need to compile the benchmark first. The benchmark handler automatically performs the build in a temporary directory, so that it doesn’t interfere with the source tree. First, it copies all files listed under copy_sources to the temporary directory. If the specified source path is a directory, the whole directory is copied recursively. The lf_file configuration file specifies the file to be compiled with lfc. binary indicates the name of the binary file resulting from the compilation process.

    \n

    For some benchmarks, not all parameters can be applied at runtime. In such cases, the gen_args configuration key can be used to provide additional arguments that should be passed to cog. cog then applies the parameters to the source file (assuming that the source LF file uses cog directives to generate code according to the configuration). Similarly run_args specifies any additional arguments that should be passed to the binary when running the benchmark. In the case of the C++ configuration for the Ping Pong benchmark, the number of pings is a runtime parameter and specified with --count. Since this particular benchmark does not have any parameter that need to be set during generation, gen_args is set to null.

    \n

    Finally, we have the C part of the target configuration.

    \n
    lf-c:\n  copy_sources:\n    - "${lf_path}/benchmark/C/Savina/src/micro/PingPong.lf"\n  lf_file: "PingPong.lf"\n  binary: "PingPong"\n  gen_args:\n    pings: ["-D", "count=<value>"]\n
    \n

    This is very similar to the C++ configuration. However, the C target of LF currently does not support overriding of parameter values at runtime. Therefore, all parameters need to be provided as arguments to the code generator and the benchmark needs to provide corresponding cog directives.

    \n

    New benchmarks can be simply added by replicating this example and adjusting the precise configuration values and parameters to the specific benchmark.

    ","headings":[{"value":"Running Benchmarks","depth":1},{"value":"Prerequisites","depth":2},{"value":"Install Python dependencies","depth":3},{"value":"Compile lfc","depth":3},{"value":"Setup Savina","depth":3},{"value":"CAF","depth":4},{"value":"Running a benchmark","depth":2},{"value":"Running a series of benchmarks (multirun)","depth":2},{"value":"Collecting results from multirun","depth":2},{"value":"How it works","depth":2},{"value":"Adding new benchmarks","depth":2}],"frontmatter":{"permalink":"/docs/handbook/running-benchmarks","title":"Running Benchmarks","oneline":"Running Benchmarks.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","permalink":"/docs/handbook/regression-tests"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Website Development","oneline":"Development of the Lingua Franca website.","permalink":"/docs/handbook/website-development"}}}},"pageContext":{"id":"5-running-benchmarks","slug":"/docs/handbook/running-benchmarks","repoPath":"/packages/documentation/copy/en/developer/Running Benchmarks.md","previousID":"b004db16-2d0f-54e3-a2a8-d9d6f510eea1","nextID":"6fe7623e-c5ce-509d-9532-9d282ae790cc","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/running-benchmarks","result":{"data":{"markdownRemark":{"id":"ac793a6f-ced3-5dd9-972e-a6891620a738","excerpt":"Running Benchmarks The LF repository contains a series of benchmarks in the benchmark directory. There is also a flexible benchmark runner that automates the…","html":"

    Running Benchmarks

    \n

    The LF repository contains a series of benchmarks in the benchmark directory. There is also a flexible benchmark runner that automates the process of running benchmarks for various settings and collecting results from those benchmarks. It is located in benchmark/runner.\nThe runner is written in python and is based on hydra, a tool for dynamically creating hierarchical configurations by composition

    \n

    Prerequisites

    \n

    Install Python dependencies

    \n

    The benchmark runner is written in Python and requires a working Python3 installation. It also requires a few python packages to be installed. Namely, hydra-core, cogapp and pandas.

    \n

    It is recommended to install the dependencies and execute the benchmark runner in a virtual environment. For instance, this can be done with virtualenv:

    \n
    virtualenv ~/virtualenvs/lfrunner -p python3\nsource ~/virtualenvs/lfrunner/bin/activate\n
    \n

    Then the dependencies can be installed by running:

    \n
    pip install -r benchmark/runner/requirements.txt\n
    \n

    Compile lfc

    \n

    For running LF benchmarks, the command-line compiler lfc needs to be built. Simply run

    \n
    bin/build-lfc\n
    \n

    in the root directory of the LF repository.

    \n

    Also, the environment variable LF_PATH needs to be set and point to the location of the LF repository. This needs to be an absolute path.

    \n
    export LF_PATH=/path/to/lf\n
    \n

    Setup Savina

    \n

    Currently all of our benchmarks are ported from the Savina actor benchmark suite. In order to compare our LF implementations with actor based implementation, the Savina benchmark suite needs to be downloaded and compiled. Note that we require a modified version of the Savina suite, that adds support for specifying the number of worker threads and that includes CAF implementations of most benchmarks.

    \n

    To download and build Savina, run the following commands:

    \n
    git clone https://github.com/lf-lang/savina.git\ncd savina\nmvn install\n
    \n

    Building Savina requires a Java 8 JDK. Depending on the local setup, JAVA_HOME might need to be adjusted before running mvn in order to point to the correct JDK.

    \n
    export JAVA_HOME=/path/to/jdk8\n
    \n

    Before invoking the benchmark runner, the environment variable SAVINA_PATH needs to be set and point to the location of the Savina repository using an absolute path.

    \n
    export SAVINA_PATH=/path/to/savina\n
    \n

    CAF

    \n

    To further build the CAF benchmarks, CAF 0.16.5 needs to be downloaded, compiled and installed first:

    \n
    git clone --branch "0.16.5" git@github.com:actor-framework/actor-framework.git\nmkdir actor-framework/build && cd actor-framework/build\ncmake -DCMAKE_INSTALL_PREFIX=<preferred/install/location> ..\nmake install\n
    \n

    Then, from within the Savina directory, the CAF benchmarks can be build:

    \n
    cmake -DCAF_ROOT_DIR=<path/to/caf/install/location> ..\nmake\n
    \n

    The CAF benchmarks are used in these two publications:

    \n\n

    Running a benchmark

    \n

    A benchmark can simply be run by specifying a benchmark and a target. For instance

    \n
    cd benchmark/runner\n./run_benchmark.py benchmark=savina_micro_pingpong target=lf-c\n
    \n

    runs the Ping Pong benchmark from the Savina suite using the C-target of LF. Currently, supported targets are lf-c, lf-cpp, akka, and caf where akka corresponds to the Akka implementation in the original Savina suite and caf corresponds to a implementation using the C++ Actor Framework .

    \n

    The benchmarks can also be configured. The threads and iterations parameters apply to every benchmark and specify the number of worker threads as well as how many times the benchmark should be run. Most benchmarks allow additional parameters. For instance, the Ping Pong benchmark sends a configurable number of pings that be set via the benchmark.params.messages configuration key. Running the Akka version of the Ping Pong benchmark for 1000 messages, 1 thread and 12 iterations could be done like this:

    \n
    ./run_benchmark.py benchmark=savina_micro_pingpong target=akka threads=1 iterations=12 benchmark.params.messages=1000\n
    \n

    Each benchmark run produces an output directory in the scheme outputs/<date>/<time>/ (e.g. outputs/2020-12-17/16-46-16/). This directory contains a files results.csv which contains the measured execution time for each iteration and all the parameters used for running this particular benchmark. The CSV file contains precisely one row per iteration.

    \n

    Running a series of benchmarks (multirun)

    \n

    The runner also allows to automatically run a single benchmark or a series of benchmarks with a range of settings. The multirun feature is simply used by the -m switch. For instance:

    \n
    ./run_benchmark.py -m benchmark=savina_micro_pingpong target="glob(*)" threads=1,2,4 iterations=12 benchmark.params.messages="range(1000000,10000000,1000000)"\n
    \n

    runs the Ping Pong benchmark for all targets using 1, 2 and 4 threads and for a number of messages ranging from 1M to 10M (in 1M steps).

    \n

    This mechanism can also be used to run multiple benchmarks. For instance,

    \n
    ./run_benchmark.py -m benchmark="glob(*)" target="glob(*)" threads=4 iterations=12\n
    \n

    runs all benchmarks for all targets using 4 threads and 12 iterations.

    \n

    The results for a multirun are written to a directory in the scheme multirun/<date>/<time>/<n> (e.g. multirun/2020-12-17/17-11-03/0/) where <n> denotes the particular run. Each of the <n> subdirectories contains a results.csv for this particular run.

    \n

    Collecting results from multirun

    \n

    A second script called collect_results.py provides a convenient way for collecting results from a multirun and merging them into a single CSV file. Simply running

    \n
    ./collect_results.py multirun/<date>/<time>/ out.csv\n
    \n

    collects all results from the particular multirun and stores the merged data structure in out.csv. collect_results.py not only merges the results, but it also calculates minimum, maximum and median execution time for each individual run. The resulting CSV does not contain the measured values of individual iterations anymore and only contains a single row per run. This behavior can be disabled with the --raw command line flag. With the flag set, the results from all runs are merged as say are and the resulting file contains rows for all individual runs, but no minimum, maximum and median values.

    \n

    How it works

    \n

    The benchmark runner itself is actually relatively simple. Most of the complexity is dealt with by hydra. Hydra is a complex and convenient tool for handling configurations. These configurations can be merged from different sources and be overridden via command line arguments as you have seen above. The actual benchmark runner receives the configuration represented as nested dictionaries from hydra. It then executes the benchmarks precisely as instructed by the configuration.

    \n

    The configuration is split into two big parts: the benchmark configuration and the target configuration. The benchmark configuration describes a particular benchmark instance. This is described in more detail in the next section. The target configuration specifies how to run a benchmark for a specific target (e.g. akka, lf-c, lf-cpp). This is not intended to be changed by the user and therefore isn’t explained in detail here. Essentially a benchmark run is split into 5 steps as is outlined in the following. The target configuration precisely specifies what needs to be done in each step

    \n
      \n
    1. copy The command used to copy relevant source files to a temporary directory.
    2. \n
    3. gen The command used to generate a configured LF file. This is intended to apply a code generation tool like cog to the source code in order to make benchmarks parameterized.
    4. \n
    5. compile The command used to compile the benchmark.
    6. \n
    7. run The command used to generate the benchmark.
    8. \n
    9. parser A parser (a python method) that is used to process the output of the benchmark run and that returns the execution times of individual benchmark runs in a list.
    10. \n
    \n

    Adding new benchmarks

    \n

    In order to add new benchmarks, a new configuration file needs to be created in the conf/benchmark subdirectory. Benchmarks may be grouped by the underscore-delimited segments in their file name. For instance, the PingPong benchmark is part of the micro-benchmarks of the Savina suite, and consequently its configuration file is named in conf/benchmark/savina_micro_pingpong.yaml. This allows to later specify benchmark=savina/micro/pingpong on the command line. Below you can see the contents of savina_micro_pingpong.yaml which we will break down in the following.

    \n
    # @package benchmark\nname: "Ping Pong"\nparams:\n  pings: 1000000\n\n# target specific configuration\ntargets:\n  akka:\n    jar: "${savina_path}/target/savina-0.0.1-SNAPSHOT-jar-with-dependencies.jar"\n    class: "edu.rice.habanero.benchmarks.pingpong.PingPongAkkaActorBenchmark"\n    run_args:\n      pings: ["-n", "<value>"]\n  caf:\n    bin: "caf_01_pingpong"\n    run_args:\n      pings: ["-n", "<value>"]\n  lf-cpp:\n    copy_sources:\n      - "${lf_path}/benchmark/Cpp/Savina/src/BenchmarkRunner.lf"\n      - "${lf_path}/benchmark/Cpp/Savina/src/micro"\n    lf_file: "micro/PingPong.lf"\n    binary: "PingPong"\n    gen_args: null\n    run_args:\n      pings: ["--count", "<value>"]\n  lf-c:\n    copy_sources:\n      - "${lf_path}/benchmark/C/Savina/src/micro/PingPong.lf"\n    lf_file: "PingPong.lf"\n    binary: "PingPong"\n    gen_args:\n      pings: ["-D", "count=<value>"]\n
    \n

    The first line # @package benchmark is hydra specific. It specifies that this configuration is part of the benchmark package. Essentially this enables the configuration to be assigned to benchmark on the command line.

    \n
    name: "Ping Pong"\nparams:\n  pings: 1000000\n
    \n

    This part sets the benchmark name to “Ping Pong” and declares that there is one benchmark specific parameter: pings. This configuration also set the default value for pings to 1000000. Note that the params dictionary may specify an arbitrary number of parameters.

    \n

    The remainder of the configuration file contains target specific configurations that provide instructions on how the particular benchmark can be run for the various targets. This block

    \n
    # target specific configuration\ntargets:\n  akka:\n    jar: "${savina_path}/target/savina-0.0.1-SNAPSHOT-jar-with-dependencies.jar"\n    class: "edu.rice.habanero.benchmarks.pingpong.PingPongAkkaActorBenchmark"\n    run_args:\n      pings: ["-n", "<value>"]\n
    \n

    specifies how the benchmark is executed using Akka. The jar and class configuration keys simply instruct the benchmark runner which class in which jar to run. Note that hydra automatically resolves ${savina_path} to the value you set in the SAVINA_PATH environment variable.

    \n

    The run_args configuration key allows specification of further arguments that are added to the command to be executed when running the benchmark. It expects a dictionary, where the keys are names of parameters as specified above in the params configuration key, and the values are a list of arguments to be added to the executed command. In the case of the pings parameter, the Akka implementation of the benchmark expects the -n flag followed by the parameter value. Note that the special string <value> is automatically resolved by the runner to the actual parameter value when executing the command.

    \n

    Instructions for the C++ target are specified as follows.

    \n
    lf-cpp:\n  copy_sources:\n    - "${lf_path}/benchmark/Cpp/Savina/src/BenchmarkRunner.lf"\n    - "${lf_path}/benchmark/Cpp/Savina/src/micro"\n  lf_file: "micro/PingPong.lf"\n  binary: "PingPong"\n  gen_args: null\n  run_args:\n    pings: ["--count", "<value>"]\n
    \n

    For C and C++ programs, we cannot run a precompiled program as it is the case for Akka, but we need to compile the benchmark first. The benchmark handler automatically performs the build in a temporary directory, so that it doesn’t interfere with the source tree. First, it copies all files listed under copy_sources to the temporary directory. If the specified source path is a directory, the whole directory is copied recursively. The lf_file configuration file specifies the file to be compiled with lfc. binary indicates the name of the binary file resulting from the compilation process.

    \n

    For some benchmarks, not all parameters can be applied at runtime. In such cases, the gen_args configuration key can be used to provide additional arguments that should be passed to cog. cog then applies the parameters to the source file (assuming that the source LF file uses cog directives to generate code according to the configuration). Similarly run_args specifies any additional arguments that should be passed to the binary when running the benchmark. In the case of the C++ configuration for the Ping Pong benchmark, the number of pings is a runtime parameter and specified with --count. Since this particular benchmark does not have any parameter that need to be set during generation, gen_args is set to null.

    \n

    Finally, we have the C part of the target configuration.

    \n
    lf-c:\n  copy_sources:\n    - "${lf_path}/benchmark/C/Savina/src/micro/PingPong.lf"\n  lf_file: "PingPong.lf"\n  binary: "PingPong"\n  gen_args:\n    pings: ["-D", "count=<value>"]\n
    \n

    This is very similar to the C++ configuration. However, the C target of LF currently does not support overriding of parameter values at runtime. Therefore, all parameters need to be provided as arguments to the code generator and the benchmark needs to provide corresponding cog directives.

    \n

    New benchmarks can be simply added by replicating this example and adjusting the precise configuration values and parameters to the specific benchmark.

    ","headings":[{"value":"Running Benchmarks","depth":1},{"value":"Prerequisites","depth":2},{"value":"Install Python dependencies","depth":3},{"value":"Compile lfc","depth":3},{"value":"Setup Savina","depth":3},{"value":"CAF","depth":4},{"value":"Running a benchmark","depth":2},{"value":"Running a series of benchmarks (multirun)","depth":2},{"value":"Collecting results from multirun","depth":2},{"value":"How it works","depth":2},{"value":"Adding new benchmarks","depth":2}],"frontmatter":{"permalink":"/docs/handbook/running-benchmarks","title":"Running Benchmarks","oneline":"Running Benchmarks.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Regression Tests","oneline":"Regression Tests for Lingua Franca.","permalink":"/docs/handbook/regression-tests"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Website Development","oneline":"Development of the Lingua Franca website.","permalink":"/docs/handbook/website-development"}}}},"pageContext":{"id":"5-running-benchmarks","slug":"/docs/handbook/running-benchmarks","repoPath":"/packages/documentation/copy/en/developer/Running Benchmarks.md","previousID":"b004db16-2d0f-54e3-a2a8-d9d6f510eea1","nextID":"6fe7623e-c5ce-509d-9532-9d282ae790cc","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/security/page-data.json b/page-data/docs/handbook/security/page-data.json index 6c9d10a3f..61a2e360c 100644 --- a/page-data/docs/handbook/security/page-data.json +++ b/page-data/docs/handbook/security/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/security","result":{"data":{"markdownRemark":{"id":"21be3a62-9007-5e8f-a8ab-0e0e5a549b99","excerpt":"Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations…","html":"

    Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations will be secure if the auth target property is enabled.

    \n

    By default, there is no secure authentication happening when a federate joins a federation, and data exchanged by federates is not encrypted. For targets that support it, the auth target property can be used to enable authentication between federates. Messages exchanged between federates after authentication are not encrypted, but this capability is planned for the future.

    \n
    \n

    The $target-language$ target does not currently support the auth target option.

    \n
    \n
    \n

    Authentication

    \n

    For the C target, federated execution is able to apply security with authentication by using HMAC authentication between RTI and federates. To enable this, include the auth property in your target specification, as follows:

    \n
    target C {\n    auth: true\n};\n
    \n

    The RTI build must include CMake options to enable simple HMAC-based authentication of federates. Add -DAUTH=ON option to the CMake command as shown below:

    \n
    mkdir build && cd build\ncmake -DAUTH=ON ../\nmake\nsudo make install\n
    \n

    If you would like to go back to non-AUTH mode, you would have to remove all contents of the build folder.

    \n
    ","headings":[{"value":"Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations will be secure if the auth target property is enabled.","depth":3},{"value":"Authentication","depth":2}],"frontmatter":{"permalink":"/docs/handbook/security","title":"Security","oneline":"Secure Federated Execution","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Containerized Execution","oneline":"Containerized Execution using Docker","permalink":"/docs/handbook/containerized-execution"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"id":"3-security","slug":"/docs/handbook/security","repoPath":"/packages/documentation/copy/en/reference/Security.md","previousID":"0d63a4b3-f226-54ba-a299-23b69782aba6","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/security","result":{"data":{"markdownRemark":{"id":"21be3a62-9007-5e8f-a8ab-0e0e5a549b99","excerpt":"Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations…","html":"

    Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations will be secure if the auth target property is enabled.

    \n

    By default, there is no secure authentication happening when a federate joins a federation, and data exchanged by federates is not encrypted. For targets that support it, the auth target property can be used to enable authentication between federates. Messages exchanged between federates after authentication are not encrypted, but this capability is planned for the future.

    \n
    \n

    The $target-language$ target does not currently support the auth target option.

    \n
    \n
    \n

    Authentication

    \n

    For the C target, federated execution is able to apply security with authentication by using HMAC authentication between RTI and federates. To enable this, include the auth property in your target specification, as follows:

    \n
    target C {\n    auth: true\n};\n
    \n

    The RTI build must include CMake options to enable simple HMAC-based authentication of federates. Add -DAUTH=ON option to the CMake command as shown below:

    \n
    mkdir build && cd build\ncmake -DAUTH=ON ../\nmake\nsudo make install\n
    \n

    If you would like to go back to non-AUTH mode, you would have to remove all contents of the build folder.

    \n
    ","headings":[{"value":"Warning: the experimental security features described on this page are under development and not production ready. Users should not expect their federations will be secure if the auth target property is enabled.","depth":3},{"value":"Authentication","depth":2}],"frontmatter":{"permalink":"/docs/handbook/security","title":"Security","oneline":"Secure Federated Execution","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Containerized Execution","oneline":"Containerized Execution using Docker","permalink":"/docs/handbook/containerized-execution"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"id":"3-security","slug":"/docs/handbook/security","repoPath":"/packages/documentation/copy/en/reference/Security.md","previousID":"0d63a4b3-f226-54ba-a299-23b69782aba6","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/superdense-time/page-data.json b/page-data/docs/handbook/superdense-time/page-data.json index 2b728aaf0..8f7b03067 100644 --- a/page-data/docs/handbook/superdense-time/page-data.json +++ b/page-data/docs/handbook/superdense-time/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/superdense-time","result":{"data":{"markdownRemark":{"id":"20889c15-4d9e-53ef-b371-5b83af6a6440","excerpt":"$page-showing-target$ Tag vs. Time The model of time in Lingua Franca is a bit more sophisticated than we have hinted at. Specifically, a superdense model of…","html":"

    $page-showing-target$

    \n

    Tag vs. Time

    \n

    The model of time in Lingua Franca is a bit more sophisticated than we have hinted at. Specifically, a superdense model of time is used. In particular, instead of a timestamp, LF uses a tag, which consists of a logical time t and a microstep m.

    \n

    A $logical$ $action$ may have a <min_delay> of zero, and the <offset> argument to the schedule() function may be zero. In this case, the call to schedule() appears to be requesting that the action trigger at the current logical time. Here is where superdense time comes in. The action will indeed trigger at the current logical time, but one microstep later. Consider the following example:

    \n

    $start(Microsteps)$

    \n
    target C\nmain reactor {\n  state count: int = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    printf("%d. Logical time is %lld. Microstep is %d.\\n",\n        self->count, lf_tag().time, lf_tag().microstep\n    );\n    if (self->count++ < 5) {\n      lf_schedule(a, 0);\n    }\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  state count: int(1)\n  logical action a\n  reaction(startup, a) -> a {=\n    std::cout << count << " Logical time is " << get_logical_time() << " Microstep: " << get_microstep() <<std::endl;\n    if (count++ < 5) {\n      a.schedule();\n    }\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  state count = 1\n  logical action a\n  reaction(startup, a) {=\n    print(\n        f"{self.count}. Logical time is {lf.tag().time}. "\n        f"Microstep is {lf.tag().microstep}."\n    )\n    if self.count < 5:\n      a.schedule(0)\n    self.count += 1\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  state count: number = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    console.log(`${count}. Logical time is ${util.getCurrentLogicalTime()}. Microstep is ${util.getCurrentTag().microstep}.`)\n    if (count++ < 5) {\n      actions.a.schedule(TimeValue.zero(), null)\n    }\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  state count: u32 = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    let tag = ctx.get_tag();\n    println!(\n        "{}. Logical time is {}. Microstep is {}.",\n        self.count,\n        tag.offset_from_t0.as_nanos(),\n        tag.microstep(),\n    );\n    if self.count < 5 {\n      self.count += 1;\n      ctx.schedule(a, Asap);\n    }\n  =}\n}\n
    \n

    $end(Microsteps)$

    \n\"Lingua\n

    Executing this program will yield something like this:

    \n
    1. Logical time is 1649607749415269000. Microstep is 0.\n2. Logical time is 1649607749415269000. Microstep is 1.\n3. Logical time is 1649607749415269000. Microstep is 2.\n4. Logical time is 1649607749415269000. Microstep is 3.\n5. Logical time is 1649607749415269000. Microstep is 4.
    \n

    Notice that the logical time is not advancing, but the microstep is (the logical time, in this case, gives the number of nanoseconds that have elapsed since January 1, 1970). The general rule is that every call to schedule() advances the tag by at least one microstep.

    \n

    Logical Simultaneity

    \n

    Two events are logically simultaneous only if both the logical time and the microstep are equal. The following example illustrates this:

    \n

    $start(Simultaneous)$

    \n
    target C\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    printf("Time since start: %lld, microstep: %d\\n",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n    if (x->is_present) {\n      printf("  x is present.\\n");\n    }\n    if (y->is_present) {\n      printf("  y is present.\\n");\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    lf_set(d.x, 1);\n    lf_schedule(repeat, 0);\n  =}\n  reaction(repeat) -> d.y {= lf_set(d.y, 1); =}\n}\n
    \n
    target Cpp\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    std::cout << "Time since start: " << get_elapsed_logical_time() << " Current Microstep: " << get_microstep() << std::endl;\n    if (x.is_present()) {\n      std::cout << "x is present" << std::endl;\n    }\n    if (y.is_present()) {\n      std::cout << "y is present" << std::endl;\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x.set(1);\n    repeat.schedule(0ms);\n  =}\n  reaction(repeat) -> d.y {= d.y.set(1); =}\n}\n
    \n
    target Python\nreactor Destination {\n  input x\n  input y\n  reaction(x, y) {=\n    print(\n        f"Time since start: {lf.time.logical_elapsed()}, "\n        f"microstep: {lf.tag().microstep}"\n    )\n    if x.is_present:\n      print("  x is present.")\n    if y.is_present:\n      print("  y is present.")\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x.set(1)\n    repeat.schedule(0)\n  =}\n  reaction(repeat) -> d.y {= d.y.set(1) =}\n}\n
    \n
    target TypeScript\nreactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    console.log(`Time since start: ${util.getElapsedLogicalTime()}, microstep: ${util.getCurrentTag().microstep}`)\n    if (x !== undefined) {\n      console.log("  x is present.")\n    }\n    if (y !== undefined) {\n      console.log("  y is present.")\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x = 1\n    actions.repeat.schedule(0, null)\n  =}\n  reaction(repeat) -> d.y {= d.y = 1 =}\n}\n
    \n
    target Rust\nreactor Destination {\n  input x: u32\n  input y: u32\n  reaction(x, y) {=\n    let tag = ctx.get_tag();\n    println!(\n        "Time since start: {}, microstep: {}",\n        tag.offset_from_t0.as_nanos(),\n        tag.microstep,\n    );\n    if ctx.is_present(x) {\n      println!("  x is present.");\n    }\n    if ctx.is_present(y) {\n      println!("  y is present.");\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    ctx.set(d__x, 1);\n    ctx.schedule(repeat, Asap);\n  =}\n  reaction(repeat) -> d.y {= ctx.set(d__y, 1); =}\n}\n
    \n

    $end(Simultaneous)$

    \n\"Lingua\n

    The Destination reactor has two inputs, x and y, and it reports in a reaction to either input what is the logical time, the microstep, and which input is present. The main reactor reacts to $startup$ by sending data to the x input of Destination. It then schedules a repeat action with an <offset> of zero. The repeat reaction is invoked strictly later, one microstep later. The output printed, therefore, will look like this:

    \n
    Time since start: 0, microstep: 0\n  x is present.\nTime since start: 0, microstep: 1\n  y is present.
    \n

    The reported elapsed logical time has not advanced in the second reaction, but the fact that x is not present in the second reaction proves that the first reaction and the second are not logically simultaneous. The second occurs one microstep later.

    \n

    Alignment of Logical and Physical Times

    \n

    Recall that in Lingua Franca, logical time “chases” physical time, invoking reactions at a physical time close to their logical time. For that purpose, the microstep is ignored.

    ","headings":[{"value":"Tag vs. Time","depth":2},{"value":"Logical Simultaneity","depth":2},{"value":"Alignment of Logical and Physical Times","depth":2}],"frontmatter":{"permalink":"/docs/handbook/superdense-time","title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Actions","oneline":"Actions in Lingua Franca.","permalink":"/docs/handbook/actions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Modal Reactors","oneline":"Modal Reactors","permalink":"/docs/handbook/modal-models"}}}},"pageContext":{"id":"1-superdense-time","slug":"/docs/handbook/superdense-time","repoPath":"/packages/documentation/copy/en/topics/Superdense Time.md","previousID":"ab880406-6c38-59c6-9a2c-a8f736013224","nextID":"ddeb2577-9554-5362-9ed2-abba8f412fc1","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/superdense-time","result":{"data":{"markdownRemark":{"id":"20889c15-4d9e-53ef-b371-5b83af6a6440","excerpt":"$page-showing-target$ Tag vs. Time The model of time in Lingua Franca is a bit more sophisticated than we have hinted at. Specifically, a superdense model of…","html":"

    $page-showing-target$

    \n

    Tag vs. Time

    \n

    The model of time in Lingua Franca is a bit more sophisticated than we have hinted at. Specifically, a superdense model of time is used. In particular, instead of a timestamp, LF uses a tag, which consists of a logical time t and a microstep m.

    \n

    A $logical$ $action$ may have a <min_delay> of zero, and the <offset> argument to the schedule() function may be zero. In this case, the call to schedule() appears to be requesting that the action trigger at the current logical time. Here is where superdense time comes in. The action will indeed trigger at the current logical time, but one microstep later. Consider the following example:

    \n

    $start(Microsteps)$

    \n
    target C\nmain reactor {\n  state count: int = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    printf("%d. Logical time is %lld. Microstep is %d.\\n",\n        self->count, lf_tag().time, lf_tag().microstep\n    );\n    if (self->count++ < 5) {\n      lf_schedule(a, 0);\n    }\n  =}\n}\n
    \n
    target Cpp\nmain reactor {\n  state count: int(1)\n  logical action a\n  reaction(startup, a) -> a {=\n    std::cout << count << " Logical time is " << get_logical_time() << " Microstep: " << get_microstep() <<std::endl;\n    if (count++ < 5) {\n      a.schedule();\n    }\n  =}\n}\n
    \n
    target Python\nmain reactor {\n  state count = 1\n  logical action a\n  reaction(startup, a) {=\n    print(\n        f"{self.count}. Logical time is {lf.tag().time}. "\n        f"Microstep is {lf.tag().microstep}."\n    )\n    if self.count < 5:\n      a.schedule(0)\n    self.count += 1\n  =}\n}\n
    \n
    target TypeScript\nmain reactor {\n  state count: number = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    console.log(`${count}. Logical time is ${util.getCurrentLogicalTime()}. Microstep is ${util.getCurrentTag().microstep}.`)\n    if (count++ < 5) {\n      actions.a.schedule(TimeValue.zero(), null)\n    }\n  =}\n}\n
    \n
    target Rust\nmain reactor {\n  state count: u32 = 1\n  logical action a\n  reaction(startup, a) -> a {=\n    let tag = ctx.get_tag();\n    println!(\n        "{}. Logical time is {}. Microstep is {}.",\n        self.count,\n        tag.offset_from_t0.as_nanos(),\n        tag.microstep(),\n    );\n    if self.count < 5 {\n      self.count += 1;\n      ctx.schedule(a, Asap);\n    }\n  =}\n}\n
    \n

    $end(Microsteps)$

    \n\"Lingua\n

    Executing this program will yield something like this:

    \n
    1. Logical time is 1649607749415269000. Microstep is 0.\n2. Logical time is 1649607749415269000. Microstep is 1.\n3. Logical time is 1649607749415269000. Microstep is 2.\n4. Logical time is 1649607749415269000. Microstep is 3.\n5. Logical time is 1649607749415269000. Microstep is 4.
    \n

    Notice that the logical time is not advancing, but the microstep is (the logical time, in this case, gives the number of nanoseconds that have elapsed since January 1, 1970). The general rule is that every call to schedule() advances the tag by at least one microstep.

    \n

    Logical Simultaneity

    \n

    Two events are logically simultaneous only if both the logical time and the microstep are equal. The following example illustrates this:

    \n

    $start(Simultaneous)$

    \n
    target C\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    printf("Time since start: %lld, microstep: %d\\n",\n        lf_time_logical_elapsed(), lf_tag().microstep\n    );\n    if (x->is_present) {\n      printf("  x is present.\\n");\n    }\n    if (y->is_present) {\n      printf("  y is present.\\n");\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    lf_set(d.x, 1);\n    lf_schedule(repeat, 0);\n  =}\n  reaction(repeat) -> d.y {= lf_set(d.y, 1); =}\n}\n
    \n
    target Cpp\nreactor Destination {\n  input x: int\n  input y: int\n  reaction(x, y) {=\n    std::cout << "Time since start: " << get_elapsed_logical_time() << " Current Microstep: " << get_microstep() << std::endl;\n    if (x.is_present()) {\n      std::cout << "x is present" << std::endl;\n    }\n    if (y.is_present()) {\n      std::cout << "y is present" << std::endl;\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x.set(1);\n    repeat.schedule(0ms);\n  =}\n  reaction(repeat) -> d.y {= d.y.set(1); =}\n}\n
    \n
    target Python\nreactor Destination {\n  input x\n  input y\n  reaction(x, y) {=\n    print(\n        f"Time since start: {lf.time.logical_elapsed()}, "\n        f"microstep: {lf.tag().microstep}"\n    )\n    if x.is_present:\n      print("  x is present.")\n    if y.is_present:\n      print("  y is present.")\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x.set(1)\n    repeat.schedule(0)\n  =}\n  reaction(repeat) -> d.y {= d.y.set(1) =}\n}\n
    \n
    target TypeScript\nreactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    console.log(`Time since start: ${util.getElapsedLogicalTime()}, microstep: ${util.getCurrentTag().microstep}`)\n    if (x !== undefined) {\n      console.log("  x is present.")\n    }\n    if (y !== undefined) {\n      console.log("  y is present.")\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    d.x = 1\n    actions.repeat.schedule(0, null)\n  =}\n  reaction(repeat) -> d.y {= d.y = 1 =}\n}\n
    \n
    target Rust\nreactor Destination {\n  input x: u32\n  input y: u32\n  reaction(x, y) {=\n    let tag = ctx.get_tag();\n    println!(\n        "Time since start: {}, microstep: {}",\n        tag.offset_from_t0.as_nanos(),\n        tag.microstep,\n    );\n    if ctx.is_present(x) {\n      println!("  x is present.");\n    }\n    if ctx.is_present(y) {\n      println!("  y is present.");\n    }\n  =}\n}\nmain reactor {\n  logical action repeat\n  d = new Destination()\n  reaction(startup) -> d.x, repeat {=\n    ctx.set(d__x, 1);\n    ctx.schedule(repeat, Asap);\n  =}\n  reaction(repeat) -> d.y {= ctx.set(d__y, 1); =}\n}\n
    \n

    $end(Simultaneous)$

    \n\"Lingua\n

    The Destination reactor has two inputs, x and y, and it reports in a reaction to either input what is the logical time, the microstep, and which input is present. The main reactor reacts to $startup$ by sending data to the x input of Destination. It then schedules a repeat action with an <offset> of zero. The repeat reaction is invoked strictly later, one microstep later. The output printed, therefore, will look like this:

    \n
    Time since start: 0, microstep: 0\n  x is present.\nTime since start: 0, microstep: 1\n  y is present.
    \n

    The reported elapsed logical time has not advanced in the second reaction, but the fact that x is not present in the second reaction proves that the first reaction and the second are not logically simultaneous. The second occurs one microstep later.

    \n

    Alignment of Logical and Physical Times

    \n

    Recall that in Lingua Franca, logical time “chases” physical time, invoking reactions at a physical time close to their logical time. For that purpose, the microstep is ignored.

    ","headings":[{"value":"Tag vs. Time","depth":2},{"value":"Logical Simultaneity","depth":2},{"value":"Alignment of Logical and Physical Times","depth":2}],"frontmatter":{"permalink":"/docs/handbook/superdense-time","title":"Superdense Time","oneline":"Superdense time in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Actions","oneline":"Actions in Lingua Franca.","permalink":"/docs/handbook/actions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Modal Reactors","oneline":"Modal Reactors","permalink":"/docs/handbook/modal-models"}}}},"pageContext":{"id":"1-superdense-time","slug":"/docs/handbook/superdense-time","repoPath":"/packages/documentation/copy/en/topics/Superdense Time.md","previousID":"ab880406-6c38-59c6-9a2c-a8f736013224","nextID":"ddeb2577-9554-5362-9ed2-abba8f412fc1","lang":"en","modifiedTime":"2023-11-08T00:42:45.532Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/target-declaration/page-data.json b/page-data/docs/handbook/target-declaration/page-data.json index 8285ac8fa..d63e3a657 100644 --- a/page-data/docs/handbook/target-declaration/page-data.json +++ b/page-data/docs/handbook/target-declaration/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/target-declaration","result":{"data":{"markdownRemark":{"id":"e4812e4e-df22-54fb-9a27-a82edb11ac44","excerpt":"$page-showing-target$ Every Lingua Franca program begins with a statement of this form: The gives the name of some Lingua Franca target language, which…","html":"

    $page-showing-target$

    \n

    Every Lingua Franca program begins with a statement of this form:

    \n
        target <name> <parameters>\n
    \n

    The <name> gives the name of some Lingua Franca target language, which is the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca compiler. The target languages currently supported are C, C++, Python, TypeScript, and Rust.

    \n

    Summary of Parameters

    \n

    A target specification may have optional parameters, the names and values of which depend on which specific target you are using. Each parameter is a key-value pair, where the supported keys are a subset of the following:

    \n
      \n
    • auth: A boolean specifying to apply authorization between RTI and federates when federated execution.
    • \n
    • build: A command to execute after code generation instead of the default compile command.
    • \n
    • build-type: One of Release (the default), Debug, RelWithDebInfo and MinSizeRel.
    • \n
    • cargo-dependencies: (Rust only) list of dependencies to include in the generated Cargo.toml file.
    • \n
    • cargo-features: (Rust only) List of string names of features to include.
    • \n
    • cmake: Whether to use cmake for building.
    • \n
    • cmake-include: List of paths to cmake files to guide compilation.
    • \n
    • compiler: A string giving the name of the target language compiler to use.
    • \n
    • docker: A boolean to generate a Dockerfile.
    • \n
    • external-runtime-path: Specify a pre-compiled external runtime library located to link to instead of the default.
    • \n
    • export-dependency-graph: To export the reaction dependency graph as a dot graph (for debugging).
    • \n
    • fast: A boolean specifying to execute as fast as possible without waiting for physical time to match logical time.
    • \n
    • files: An array of paths to files or directories to be copied to the directory that contains the generated sources.
    • \n
    • flags: An arrays of strings giving options to be passed to the target compiler.
    • \n
    • logging: An indicator of how much information to print when executing the program.
    • \n
    • no-compile: If true, then do not invoke a target language compiler. Just generate code.
    • \n
    • no-runtime-validation: If true, disable runtime validation.
    • \n
    • protobufs: An array of .proto files that are to be compiled and included in the generated code.
    • \n
    • runtime-version: Specify which version of the runtime system to use.
    • \n
    • rust-include: (Rust only) A set of Rust modules in the generated project.
    • \n
    • scheduler: (C only) Specification of the scheduler to us.
    • \n
    • single-file-project: (Rust only) If true, enables single-file project layout.
    • \n
    • threading: Whether to use multiple threads.
    • \n
    • timeout: A time value (with units) specifying the logical stop time of execution. See Termination.
    • \n
    • workers: If using multiple threads, how many worker threads to create.
    • \n
    \n

    Not all targets support all target parameters. The full set of target parameters supported by the $target-language$ target is:

    \n
    target C {\n    auth: <true or false>\n    build: <string>,\n    build-type: <Release, Debug, RelWithDebInfo, or MinSizeRel>,\n    cmake: <true or false>,\n    cmake-include: <string or list of strings>,\n    compiler: <string>,\n    docker: <true or false>,\n    fast: <true or false>,\n    files: <string or list of strings>,\n    flags: <string or list of strings>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    protobufs: <string or list of strings>,\n    threading: <true or false>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target Cpp {\n    build-type: <Release, Debug, RelWithDebInfo, or MinSizeRel>,\n    cmake-include: <string or list of strings>,\n    external-runtime-path: <string>,\n    export-dependency-graph <true or false>,\n    fast: <true or false>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    no-runtime-validation: <true or false>,\n    runtime-version: <string>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target Python {\n    docker: <true or false>,\n    fast: <true or false>,\n    files: <string or list of strings>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    protobufs: <string or list of strings>,\n    threading: <true or false>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target TypeScript {\n    docker: <true or false>,\n    fast: <true or false>,\n    logging: <ERROR, WARN, INFO, LOG, or DEBUG>,\n    timeout: <time>,\n};\n
    \n
    target Rust {\n    build-type: <Debug, Release, RelWithDebInfo, or MinSizeRel>,\n    cargo-features: <array of strings>,\n    cargo-dependencies: <list of key-value pairs>,\n    export-dependency-graph: <true or false>,\n    rust-include: <array of strings>,\n    single-file-project: <true or false>,\n    timeout: <time value>,\n}\n
    \n
    \n

    For example:

    \n
    target C {\n    cmake: false,\n    compiler: "cc",\n    flags: "-O3",\n    fast: true,\n    logging: log,\n    timeout: 1 secs,\n};\n
    \n

    This specifies to use compiler cc instead of the default gcc, to use optimization level 3, to execute as fast as possible, and to exit execution when logical time has advanced to 10 seconds. Note that all events at logical time 10 seconds greater than the starting logical time will be executed.

    \n
    \n

    The comma on the last parameter is optional, as is the semicolon on the last line.\nA target may support overriding the target parameters on the command line when invoking the compiled program.

    \n

    auth

    \n
    \n

    The $target-language$ target does not currently support the auth target option.

    \n
    \n
    \n

    The detailed documentation is here.

    \n
    \n

    build

    \n
    \n

    The $target-language$ target does not currently support the build target option.

    \n
    \n
    \n

    A command to execute after code generation instead of the default compile command. This is either a single string or an array of strings. The specified command(s) will be executed an environment that has the following environment variables defined:

    \n
      \n
    • LF_CURRENT_WORKING_DIRECTORY: The directory in which the command is invoked.
    • \n
    • LF_SOURCE_DIRECTORY: The directory containing the .lf file being compiled.
    • \n
    • LF_PACKAGE_DIRECTORY: The directory for the root of the project or package (normally the directory above the src directory).
    • \n
    • LF_SOURCE_GEN_DIRECTORY: The directory in which generated files are placed.
    • \n
    • LF_BIN_DIRECTORY: The directory into which to put binaries.
    • \n
    \n

    The command will be executed in the same directory as the .lf file being compiled. For example, if you specify

    \n
    target C {\n    build: "./compile.sh Foo"\n}\n
    \n

    then instead of invoking the C compiler after generating code, the code generator will invoke your compile.sh script, which could look something like this:

    \n
    #!/bin/bash\n# Build the generated code.\ncd ${LF_SOURCE_GEN_DIRECTORY}\ncmake .\nmake\n\n# Move the executable to the bin directory.\nmv $1 ${LF_BIN_DIRECTORY}\n\n# Invoke the executable.\n${LF_BIN_DIRECTORY}/$1\n\n# Plot the results, which have appeared in the src-gen directory.\ngnuplot ${LF_SOURCE_DIRECTORY}/$1.gnuplot\nopen $1.pdf\n
    \n

    The first few lines of this script do the same thing that is normally done when there is no build option in the target. Specifically, they use cmake to create a makefile, invoke make, and then move the executable to the bin directory. The next line, however, gives new functionality. It executes the compiled code! The final two lines assume that the program has produced a file with data to be plotted and use gnuplot to plot the data. This requires, of course, that you have gnuplot installed, and that there is a file called Foo.gnuplot in the same directory as Foo.lf. The file Foo.gnuplot contains the commands to plot the data, and might look something like the following:

    \n
    set title 'My Title'\nset xrange [0:3]\nset yrange [-2:2]\nset xlabel "Time (seconds)"\nset terminal pdf size 5, 3.5\nset output 'Foo.pdf'\nplot 'mydata1.data' using 1:2 with lines, \\\n     'mydata2.data' using 1:2 with lines\n
    \n

    This assumes that your program has written two files, mydata1.data and mydata2.data containing two columns, time and value.

    \n
    \n

    build-type

    \n
    \n

    The $target-language$ target does not currently support the build-type target option.

    \n
    \n
    \n

    This parameter works with cargo to specify how to compile the code. The following options are supported:

    \n
      \n
    • Release: Optimization is turned on and debug information is missing.
    • \n
    • Debug: Debug information is included in the executable.
    • \n
    • RelWithDebInfo: Optimization with debug information.
    • \n
    • MinSizeRel: Optimize for smallest size.
    • \n
    \n

    This defaults to Release.

    \n
    \n
    \n

    This parameter works with cmake to specify how to compile the code. The following options are supported:

    \n
      \n
    • Release: Optimization is turned on and debug information is missing.
    • \n
    • Debug: Debug information is included in the executable.
    • \n
    • RelWithDebInfo: Optimization with debug information.
    • \n
    • MinSizeRel: Optimize for smallest size.
    • \n
    \n

    This defaults to Release.

    \n
    \n
    \n

    cargo-dependencies

    \n

    This is a list of dependencies to include in the generated Cargo.toml file. The value of this parameter is a map of package name to dependency-spec.

    \n

    Here is an example for defining dependencies:

    \n
    target Rust {\n    cargo-dependencies: {\n        fxhash: {\n            version: "0.2.1",\n        },\n        rand: {\n            version: "0.8",\n            features: ["small_rng"],\n        },\n    }\n};\n
    \n

    cargo-features

    \n

    This is a list of features of the generated crate. Supported are:

    \n\n
    \n

    cmake

    \n
    \n

    The $target-language$ target does not support the cmake target option.

    \n
    \n
    \n

    The $target-language$ target does not support the cmake target option because it always uses cmake.

    \n
    \n
    \n
    target C {\n    cmake: <true or false>\n};\n
    \n

    This will enable or disable the CMake-based build system (the default is true). Enabling the CMake build system will result in a CMakeLists.txt being generated in the src-gen directory. This CMakeLists.txt is then used when cmake is invoked by the LF runtime (either the lfc or the IDE). Alternatively, the generated program can be built manually. To do so, in the src-gen/ProgramName directory, run:

    \n
    mkdir build && cd build\ncmake ../\nmake\n
    \n

    If cmake is disabled, gcc is directly invoked after code generation by default. In this case, additional target properties, such as compiler and flags can be used to gain finer control over the compilation process.

    \n
    \n

    cmake-include

    \n
    \n

    The $target-language$ target does not support the cmake-include target option.

    \n
    \n
    \n
    target C {\n    cmake-include: ["relative/path/to/foo.txt", "relative/path/to/bar.txt", ...]\n};\n
    \n
    target Cpp {\n    cmake-include: ["relative/path/to/foo.txt", "relative/path/to/bar.txt", ...]\n};\n
    \n

    This will optionally append additional custom CMake instructions to the generated CMakeLists.txt, drawing these instructions from the specified text files (e.g, foo.txt). The specified files are resolved using the same file search algorithm as used for the files target parameter. Those files will be copied into the src-gen directory that contains the generated sources. This is done to make the generated code more portable (a feature that is useful in federated execution).

    \n

    The cmake-include target property can be used, for example, to add dependencies on various packages (e.g., by using the find_package and target_link_libraries commands).

    \n

    A CMake variable called ${LF_MAIN_TARGET} can be used in the included text file(s) for convenience. This variable will contain the name of the CMake target (i.e., the name of the main reactor). For example, a foo.txt file can contain:

    \n
    find_package(m REQUIRED) # Finds the m library\n\ntarget_link_libraries( ${LF_MAIN_TARGET} m ) # Links the m library\n
    \n

    foo.txt can then be included by specifying it as an argument to cmake-include.

    \n

    Note: For a general tutorial on finding packages in CMake, see this external documentation entry. For a list of CMake find modules, see this.

    \n

    The cmake-include parameter works in conjunction with the $import$ statement. If any imported .lf file has cmake-include among its target properties, the specified text files will be appended to the current list of cmake-includes. These files will be resolved relative to the imported .lf file using the same search procedure as for the files parameter. This helps resolve dependencies in imported reactors automatically and makes the code more modular.

    \n
    \n
    \n

    CMakeInclude.lf is an example that uses this feature. A more sophisticated example of the usage of this target parameter can be found in Rhythm.lf. A distributed version can be found in DistributedCMakeInclude.lf is a test that uses this feature.

    \n

    Note: For federated execution, both cmake-include and files are kept separate for each federate as much as possible. This means that if one federate is imported, or uses an imported reactor that other federates don’t use, it will only have access to cmake-includes and files defined in the main .lf file, plus the selectively imported .lf files. DistributedCMakeIncludeSeparateCompile.lf is a test that demonstrates this feature.

    \n
    \n
    \n

    See AsyncCallback.lf for an example.

    \n
    \n

    compiler

    \n
    \n

    The $target-language$ target does not support the compiler target option.

    \n
    \n
    \n

    This parameter is a string giving the name of the C compiler to use.\nIt is used only when cmake is set to false. For example:

    \n
    target C {\n    cmake: false,\n    compiler: "cc",\n};\n
    \n

    The compiler option here specifies to use cc rather than gcc.

    \n
    \n
    \n

    This parameter is a string giving the name of the C++ compiler to use. Normally\nCMake selects the best compiler for your system, but you can use this parameter\nto point it to your preferred C++ compiler.

    \n
    \n

    docker

    \n
    \n

    This option takes a boolean argument (default is false).

    \n

    If true, a docker file will be generated in the unfederated case.

    \n

    In the federated case, a docker file for each federate will be generated. A docker-compose file will also be generated for the top-level federated reactor.

    \n
    \n
    \n

    The $target-language$ target does not support the docker target option.

    \n
    \n

    external-runtime-path

    \n
    \n

    The $target-language$ target does not support the external-runtime-path target option.

    \n
    \n
    \n

    This option takes a string argument given a path to a pre-compiled external runtime library to use instead of the default one.

    \n
    \n
    \n

    This option takes a path as string argument to a folder containing an alternative runtime crate to use instead of the default one.

    \n
    \n

    export-dependency-graph

    \n
    \n

    The $target-language$ target does not support the export-dependency-graph target option.

    \n
    \n
    \n

    This parameter takes arguments true or false to specify whether the compiled binary will export its internal dependency graph as a dot graph when executed. This is a debugging utility.

    \n
    \n

    If a CLI is generated, the target property is ignored, and the user should instead use the --export-graph flag of the generated program.

    \n
    \n
    \n

    fast

    \n

    By default, the execution of a Lingua Franca program is slowed down, if necessary, so that logical time does not elapse faster than physical time. If you wish to execute the program as fast as possible without this constraint, then specify the fast target parameter with value true.

    \n

    files

    \n
    \n

    The $target-language$ target does not support the files option.

    \n
    \n
    \n

    The files target parameter specifies array of files or directories to be copied to the directory that contains the generated sources.

    \n
    target C {\n    files: ["file1", "file2", ...]\n}\n
    \n
    target Python {\n    files: ["file1", "file2", ...]\n}\n
    \n

    The lookup procedure for these files and directories is as follows:

    \n

    1- Search in the directory containing the .lf file that has the target directive.

    \n

    2- If not found, search in LF_CLASSPATH.

    \n

    3- If still not found, search in CLASSPATH.

    \n

    4- If still not found, search for the file as a resource. Specifically, if a file begins with a forward slash /, then the path is assumed to be relative to the root directory of the Lingua Franca source tree.

    \n
    \n

    For example, if you wish to use audio on a Mac, you can specify:

    \n
    target C {\n    files: ["/lib/C/util/audio_loop_mac.c", "/lib/C/util/audio_loop.h"]\n}\n
    \n

    Your preamble code can then include these files, for example:

    \n
    preamble {=\n    #include "audio_loop_mac.c"\n=}\n
    \n

    Your reactions can then invoke functions defined in that .c file.

    \n

    Sometimes, you will need access to these files from target code in a reaction. For the C target (at least), the generated program will contain a line like this:

    \n
        #define TARGET_FILES_DIRECTORY "path"\n
    \n

    where path is the full path to the directory containing these files. This can be used in reactions, for example, to read those files.

    \n
    \n

    Moreover, the files target specification works in conjunction with the $import$ statement. If a .lf file is imported and has designated supporting files using the files target parameter, those files will be resolved relative to that .lf file and copied to the directory that contains the generated sources. This is done to make code that imports other .lf files more modular. Rhythm.lf is an example that demonstrates most of these features.

    \n
    \n

    flags

    \n
    \n

    The $target-language$ target does not support the flags parameter.

    \n
    \n
    \n

    This parameter is a list of strings giving additional arguments to pass to the target language compiler.\nIt is used only when cmake is set to false. For example:

    \n
    target C {\n    cmake: false,\n    flags: ["-g", "-I/usr/local/include", "-L/usr/local/lib", "-lpaho-mqtt3c"],\n};\n
    \n

    The flags option specifies to include debug information in the compiled code (-g); a directory to search for include files (-I/usr/local/include); a directory to search for library files (-L/usr/local/lib); a library to link with (-lpaho-mqtt3c, which will link with file libpaho-mqtt3c.so).

    \n

    Note: Using the flags standard parameter when cmake is enabled is strongly discouraged, although supported. Flags are compiler-specific, and thus interfere with CMake’s ability to find the most suitable compiler for each platform. In a similar fashion, we recommend against the use of the compiler standard parameter for the same reason. A better solution is to provide a cmake-include file, as described next.

    \n
    \n

    logging

    \n
    \n

    The $target-language$ target does not support the logging parameter.

    \n
    \n
    \n

    By default, when executing a generated Lingua Franca program, error messages, warnings, and informational messages are printed to standard out. You can get additional information printed by setting the logging parameter to LOG or DEBUG (or log or debug). The latter is more verbose. If you set the logging parameter to warn, then warnings and errors will be printed, but informational messages will not (e.g. message produced using the info_print utility function). If you set logging to error, then warning messages will also not be printed.

    \n

    The C and Python targets also support tracing, which outputs binary traces of an execution rather than human-readable text and is designed to have minimal impact on performance.

    \n
    \n
    \n

    The logging option is one of error, warn, info, log or debug. It specifies the level of diagnostic messages about execution to print to the console. A message will print if this parameter is greater than or equal to the level of the message, where error < warn < info < log < debug. The default value is info, which means that messages log or debug messages will not print.

    \n
    \n
    \n

    The logging option is one of ERROR, WARN, INFO, LOG or DEBUG. It specifies the level of diagnostic messages about execution to print to the console. A message will print if this parameter is greater than or equal to the level of the message, where ERROR < WARN < INFO < LOG < DEBUG. The default value is INFO, which means that messages tagged LOG and DEBUG will not print. Internally this is handled by the ulog module.

    \n
    \n

    no-compile

    \n
    \n

    The $target-language$ target does not support the no-compile target option.

    \n
    \n
    \n

    If true, then do not invoke a target language compiler nor cmake. Just generate code.

    \n
    \n

    no-runtime-validation

    \n
    \n

    The $target-language$ target does not support the no-runtime-validation target option.

    \n
    \n
    \n

    This parameter takes value true or false (the default). If this is set to true, then all runtime checks in reactor-cpp will be disabled. This brings a slight performance boost but should be used with care and only on tested programs.

    \n
    \n

    protobufs

    \n
    \n

    The $target-language$ target does not support the protobufs target option.

    \n
    \n
    \n

    Protobufs is a serialization protocol by which data in a target language can be copied over the network to a remote location. The protobufs target parameter gives an array of .proto files that are to be compiled and included in the generated code. For an example, see\nPersonProtocolBuffers.lf\nPersonProtocolBuffers.lf.

    \n
    \n

    runtime-version

    \n
    \n

    The $target-language$ target does not support the runtime-version target option.

    \n
    \n
    \n

    This argument takes a string (with quotation marks) containing any tag, branch name, or git hash in the reactor-cpp repository. This will specify the version of the runtime library that the compiled binary will link against.

    \n
    \n
    \n

    rust-include

    \n

    This specifies a set of Rust modules in the generated project. See Linking support files.

    \n

    scheduler

    \n

    This specifies the scheduler to use. SeeTarget Language Details.

    \n

    single-file-project

    \n

    If true, enables single-file project layout.

    \n
    \n

    threading

    \n
    \n

    The $target-language$ target does not support the threading target option.

    \n
    \n
    \n

    If threading is disabled (by setting threading to false), then no thread library is used, and the lf_schedule() function is not thread safe. This setting is incompatible with asynchronously scheduling any physical actions and hence this parameter will be ignored for programs that have physical actions.\nSee workers.

    \n
    \n
    \n

    The Python target uses the single threaded C runtime by default but will switch to the multithreaded C runtime if a physical action is detected. This target property can be used to override this behavior.

    \n
    \n
    \n

    Boolean flag (either true (default) or false) that controls if the project is to be compiled with support for multi-threading.

    \n

    See workers.

    \n
    \n

    timeout

    \n

    A time value (with units) specifying the logical stop time of execution. See Termination.

    \n

    workers

    \n
    \n

    The $target-language$ target does not support the workers target option.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. If threading is turned on (the default, see threading), then the generated code will use a target platform thread library and generate multi-threaded code. This can transparently execute reactions that have no dependence on one another in parallel on multiple cores. By default, threading is turned on, and the workers property is set to 0, which means that the number of workers is determined by the runtime system. Typically, it will be set to the number of cores on the machine running the code. To use a different number of worker threads, give a positive integer for this target parameter.

    \n

    With value 0, the runtime engine is free to choose the number of worker threads to use. Typically, this will equal the number of hardware threads on the machine on which the Lingua Franca code generator is run.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. With value 0 (the default), the runtime engine is free to choose the number of worker threads to use. In the $target-language$ target, the runtime system will determine the number of hardware threads on the machine on which the program is run and set the number of worker threads equal to that number.

    \n

    If the workers property is set to 1, the scheduler will not create any worker threads and instead inline the execution of reactions. This is an optimization and avoids any unnecessary synchronization. Note that, in contrast to the C target, the single threaded implementation is still thread safe and asynchronous reaction scheduling is supported.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. Note, however, that the Python core is unable to execute safely in parallel on multiple cores. As a consequence, at execution time, each reaction invocation will acquire a mutual exclusion lock before executing. Hence, there is little point in setting this to any number greater than 1.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. With value 0 (the default), the runtime engine is free to choose the number of worker threads to use and the number of worker threads may vary over time.

    \n
    \n

    Command-Line Arguments

    \n
    \n

    The generated executable may feature a command-line interface (CLI), if it uses the cargo-features: [\"cli\"] target property. When that feature is enabled:

    \n
      \n
    • some target properties become settable at runtime:\n
        \n
      • --timeout <time value>: override the default timeout mentioned as a target property. The syntax for times is just like the LF one (e.g. 1msec, \"2 seconds\").
      • \n
      • --workers <number>: override the default worker count mentioned as a target property. This option is ignored unless the runtime crate has been built with the feature parallel-runtime.
      • \n
      • --export-graph: export the dependency graph (corresponds to export-dependency-graph target property). This is a flag, i.e., absent means false, present means true. This means the value of the target property is ignored and not used as default.
      • \n
      • --log-level: corresponds to the logging target property, but note that the levels have different meanings, and the target property is ignored. See Logging levels.
      • \n
      \n
    • \n
    • parameters of the main reactor are translated to CLI parameters.\n
        \n
      • Each LF parameter named param corresponds to a CLI parameter named --main-param. Underscores in the LF parameter name are replaced by hyphens.
      • \n
      • The type of each parameters must implement the trait FromStr.
      • \n
      \n
    • \n
    \n

    When the cli feature is disabled, the parameters of the main reactor will each assume their default value.

    \n
    \n
    \n

    The generated C program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast [true | false]: Specifies whether to wait for physical time to match logical time. The default is false. If this is true, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout <duration> <units>: Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those.
    • \n
    • -w, --workers <n>: Executed using worker threads if possible. This option is ignored in the single-threaded version. That is, it is ignored if a threading option was given in the target properties with value false.
    • \n
    • -i, --id <n>: The ID of the federation that this reactor will join.
    • \n
    \n

    Any other command-line arguments result in printing the above information.

    \n
    \n
    \n

    The generated C++ program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast: If set, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout '<duration> <units>': Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those.
    • \n
    • -w, --workers <n>: Use n worker threads for executing reactions.
    • \n
    • --help: Print the above information.
    • \n
    \n

    If the main reactor declares parameters, these parameters will appear as additional CLI options that can be specified when invoking the binary (see Using Parameters).

    \n
    \n
    \n

    The Python target does not currently support any command-line arguments. You must specify properties as target parameters.

    \n
    \n
    \n

    In the TypeScript target, the generated JavaScript program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast [true | false]: Specifies whether to wait for physical time to match logical time. The default is false. If this is true, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout '<duration> <units>': Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those. For the duration and units of a timeout argument to be parsed correctly as a single value, these should be specified in quotes with no leading or trailing space (e.g. ‘5 sec’).
    • \n
    • -k, --keepalive [true | false]: Specifies whether to stop execution if there are no events to process. This defaults to false, meaning that the program will stop executing when there are no more events on the event queue. If you set this to true, then the program will keep executing until either the timeout logical time is reached or the program is externally killed. If you have physical actions, it usually makes sense to set this to true.
    • \n
    • -l, --logging [ERROR | WARN | INFO | LOG | DEBUG]: The level of logging messages from the reactor-ts runtime to to print to the console. Messages tagged with a given type (error, warn, etc.) will print if this argument is greater than or equal to the level of the message (ERROR < WARN < INFO < LOG < DEBUG).
    • \n
    • -h, --help: Print this usage guide. The program will not execute if this flag is present.
    • \n
    \n

    If provided, a command line argument will override whatever value the corresponding target property had specified in the source .lf file.

    \n

    Command line options are parsed by the command-line-arguments module with these rules. For example

    \n
    node <LF_file_name>/dist/<LF_file_name>.js -f false --keepalive=true -o '4 sec' -l INFO\n
    \n

    is a valid setting.

    \n

    Any errors in command-line arguments result in printing the above information. The program will not execute if there is a parsing error for command-line arguments.

    \n

    Custom Command-Line Arguments

    \n

    User-defined command-line arguments may be created by giving the main reactor parameters. Assigning the main reactor a parameter of type string, number, boolean, or time will add an argument with corresponding name and type to the generated program’s command-line-interface. Custom arguments will also appear in the generated program’s usage guide (from the --help option). If the generated program is executed with a value specified for a custom command-line argument, that value will override the default value for the corresponding parameter. Arguments typed string, number, and boolean are parsed in the expected way, but time arguments must be specified on the command line like the --timeout property as '<duration> <units>' (in quotes).

    \n

    Note: Custom arguments may not have the same names as standard arguments like timeout or keepalive.

    \n

    For example this reactor has a custom command line argument named customArg of type number and default value 2:

    \n
    target TypeScript;\nmain reactor clArg(customArg:number(2)) {\n    reaction (startup) {=\n        console.log(customArg);\n    =}\n}\n
    \n

    If this reactor is compiled from the file simpleCLArgs.lf, executing

    \n
    node simpleCLArgs/dist/simpleCLArgs.js\n
    \n

    outputs the default value 2. But running

    \n
    node simpleCLArgs/dist/simpleCLArgs.js --customArg=42\n
    \n

    outputs 42. Additionally, we can view documentation for the custom command line argument with the --help command.

    \n
    node simpleCLArgs/dist/simpleCLArgs.js -h\n
    \n

    The program will generate the standard usage guide, but also

    \n
    --customArg '<duration> <units>'                    Custom argument. Refer to\n                                                      <path>/simpleCLArgs.lf\n                                                      for documentation.
    \n

    Additional types for Custom Command-Line Arguments

    \n

    Main reactor parameters that are not typed string, number, boolean, or time will not create custom command-line arguments. However, that doesn’t mean it is impossible to obtain other types from the command line, just use a string and specify how the parsing is done yourself. See below for an example of a reactor that parses a custom command-line argument of type string into a state variable of type Array<number> using JSON.parse and a user-defined type guard.

    \n
    target TypeScript;\nmain reactor customType(arrayArg:string("")) {\n    preamble {=\n        function isArrayOfNumbers(x: any): x is Array<number> {\n            for (let item of x) {\n                if (typeof item !== "number") {\n                    return false;\n                }\n            }\n            return true;\n        }\n    =}\n    state foo:{=Array<number>=}({=[]=});\n    reaction (startup) {=\n        let parsedArgument = JSON.parse(customType);\n        if (isArrayOfNumbers(parsedArgument)) {\n            foo = parsedArgument;\n            }\n        else {\n            throw new Error("Custom command line argument is not an array of numbers.");\n        }\n        console.log(foo);\n    =}\n}\n
    \n
    ","headings":[{"value":"Summary of Parameters","depth":1},{"value":"auth","depth":2},{"value":"build","depth":2},{"value":"build-type","depth":2},{"value":"cargo-dependencies","depth":2},{"value":"cargo-features","depth":2},{"value":"cmake","depth":2},{"value":"cmake-include","depth":2},{"value":"compiler","depth":2},{"value":"docker","depth":2},{"value":"external-runtime-path","depth":2},{"value":"export-dependency-graph","depth":2},{"value":"fast","depth":2},{"value":"files","depth":2},{"value":"flags","depth":2},{"value":"logging","depth":2},{"value":"no-compile","depth":2},{"value":"no-runtime-validation","depth":2},{"value":"protobufs","depth":2},{"value":"runtime-version","depth":2},{"value":"rust-include","depth":2},{"value":"scheduler","depth":2},{"value":"single-file-project","depth":2},{"value":"threading","depth":2},{"value":"timeout","depth":2},{"value":"workers","depth":2},{"value":"Command-Line Arguments","depth":1},{"value":"Custom Command-Line Arguments","depth":3},{"value":"Additional types for Custom Command-Line Arguments","depth":3}],"frontmatter":{"permalink":"/docs/handbook/target-declaration","title":"Target Declaration","oneline":"The target declaration and its parameters in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Target Language Details","oneline":"Detailed reference for each target langauge.","permalink":"/docs/handbook/target-language-details"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Tracing","oneline":"Tracing (preliminary)","permalink":"/docs/handbook/tracing"}}}},"pageContext":{"id":"3-target-declaration","slug":"/docs/handbook/target-declaration","repoPath":"/packages/documentation/copy/en/reference/Target Declaration.md","previousID":"15e2a8e5-dd68-55e9-839f-18c6d342e73c","nextID":"3f4000b5-1133-5c34-807d-29c05884f149","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/target-declaration","result":{"data":{"markdownRemark":{"id":"e4812e4e-df22-54fb-9a27-a82edb11ac44","excerpt":"$page-showing-target$ Every Lingua Franca program begins with a statement of this form: The gives the name of some Lingua Franca target language, which…","html":"

    $page-showing-target$

    \n

    Every Lingua Franca program begins with a statement of this form:

    \n
        target <name> <parameters>\n
    \n

    The <name> gives the name of some Lingua Franca target language, which is the language in which reactions are written. This is also the language of the program(s) generated by the Lingua Franca compiler. The target languages currently supported are C, C++, Python, TypeScript, and Rust.

    \n

    Summary of Parameters

    \n

    A target specification may have optional parameters, the names and values of which depend on which specific target you are using. Each parameter is a key-value pair, where the supported keys are a subset of the following:

    \n
      \n
    • auth: A boolean specifying to apply authorization between RTI and federates when federated execution.
    • \n
    • build: A command to execute after code generation instead of the default compile command.
    • \n
    • build-type: One of Release (the default), Debug, RelWithDebInfo and MinSizeRel.
    • \n
    • cargo-dependencies: (Rust only) list of dependencies to include in the generated Cargo.toml file.
    • \n
    • cargo-features: (Rust only) List of string names of features to include.
    • \n
    • cmake: Whether to use cmake for building.
    • \n
    • cmake-include: List of paths to cmake files to guide compilation.
    • \n
    • compiler: A string giving the name of the target language compiler to use.
    • \n
    • docker: A boolean to generate a Dockerfile.
    • \n
    • external-runtime-path: Specify a pre-compiled external runtime library located to link to instead of the default.
    • \n
    • export-dependency-graph: To export the reaction dependency graph as a dot graph (for debugging).
    • \n
    • fast: A boolean specifying to execute as fast as possible without waiting for physical time to match logical time.
    • \n
    • files: An array of paths to files or directories to be copied to the directory that contains the generated sources.
    • \n
    • flags: An arrays of strings giving options to be passed to the target compiler.
    • \n
    • logging: An indicator of how much information to print when executing the program.
    • \n
    • no-compile: If true, then do not invoke a target language compiler. Just generate code.
    • \n
    • no-runtime-validation: If true, disable runtime validation.
    • \n
    • protobufs: An array of .proto files that are to be compiled and included in the generated code.
    • \n
    • runtime-version: Specify which version of the runtime system to use.
    • \n
    • rust-include: (Rust only) A set of Rust modules in the generated project.
    • \n
    • scheduler: (C only) Specification of the scheduler to us.
    • \n
    • single-file-project: (Rust only) If true, enables single-file project layout.
    • \n
    • threading: Whether to use multiple threads.
    • \n
    • timeout: A time value (with units) specifying the logical stop time of execution. See Termination.
    • \n
    • workers: If using multiple threads, how many worker threads to create.
    • \n
    \n

    Not all targets support all target parameters. The full set of target parameters supported by the $target-language$ target is:

    \n
    target C {\n    auth: <true or false>\n    build: <string>,\n    build-type: <Release, Debug, RelWithDebInfo, or MinSizeRel>,\n    cmake: <true or false>,\n    cmake-include: <string or list of strings>,\n    compiler: <string>,\n    docker: <true or false>,\n    fast: <true or false>,\n    files: <string or list of strings>,\n    flags: <string or list of strings>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    protobufs: <string or list of strings>,\n    threading: <true or false>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target Cpp {\n    build-type: <Release, Debug, RelWithDebInfo, or MinSizeRel>,\n    cmake-include: <string or list of strings>,\n    external-runtime-path: <string>,\n    export-dependency-graph <true or false>,\n    fast: <true or false>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    no-runtime-validation: <true or false>,\n    runtime-version: <string>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target Python {\n    docker: <true or false>,\n    fast: <true or false>,\n    files: <string or list of strings>,\n    logging: <error, warning, info, log, debug>,\n    no-compile: <true or false>,\n    protobufs: <string or list of strings>,\n    threading: <true or false>,\n    timeout: <time>,\n    workers: <non-negative integer>,\n};\n
    \n
    target TypeScript {\n    docker: <true or false>,\n    fast: <true or false>,\n    logging: <ERROR, WARN, INFO, LOG, or DEBUG>,\n    timeout: <time>,\n};\n
    \n
    target Rust {\n    build-type: <Debug, Release, RelWithDebInfo, or MinSizeRel>,\n    cargo-features: <array of strings>,\n    cargo-dependencies: <list of key-value pairs>,\n    export-dependency-graph: <true or false>,\n    rust-include: <array of strings>,\n    single-file-project: <true or false>,\n    timeout: <time value>,\n}\n
    \n
    \n

    For example:

    \n
    target C {\n    cmake: false,\n    compiler: "cc",\n    flags: "-O3",\n    fast: true,\n    logging: log,\n    timeout: 1 secs,\n};\n
    \n

    This specifies to use compiler cc instead of the default gcc, to use optimization level 3, to execute as fast as possible, and to exit execution when logical time has advanced to 10 seconds. Note that all events at logical time 10 seconds greater than the starting logical time will be executed.

    \n
    \n

    The comma on the last parameter is optional, as is the semicolon on the last line.\nA target may support overriding the target parameters on the command line when invoking the compiled program.

    \n

    auth

    \n
    \n

    The $target-language$ target does not currently support the auth target option.

    \n
    \n
    \n

    The detailed documentation is here.

    \n
    \n

    build

    \n
    \n

    The $target-language$ target does not currently support the build target option.

    \n
    \n
    \n

    A command to execute after code generation instead of the default compile command. This is either a single string or an array of strings. The specified command(s) will be executed an environment that has the following environment variables defined:

    \n
      \n
    • LF_CURRENT_WORKING_DIRECTORY: The directory in which the command is invoked.
    • \n
    • LF_SOURCE_DIRECTORY: The directory containing the .lf file being compiled.
    • \n
    • LF_PACKAGE_DIRECTORY: The directory for the root of the project or package (normally the directory above the src directory).
    • \n
    • LF_SOURCE_GEN_DIRECTORY: The directory in which generated files are placed.
    • \n
    • LF_BIN_DIRECTORY: The directory into which to put binaries.
    • \n
    \n

    The command will be executed in the same directory as the .lf file being compiled. For example, if you specify

    \n
    target C {\n    build: "./compile.sh Foo"\n}\n
    \n

    then instead of invoking the C compiler after generating code, the code generator will invoke your compile.sh script, which could look something like this:

    \n
    #!/bin/bash\n# Build the generated code.\ncd ${LF_SOURCE_GEN_DIRECTORY}\ncmake .\nmake\n\n# Move the executable to the bin directory.\nmv $1 ${LF_BIN_DIRECTORY}\n\n# Invoke the executable.\n${LF_BIN_DIRECTORY}/$1\n\n# Plot the results, which have appeared in the src-gen directory.\ngnuplot ${LF_SOURCE_DIRECTORY}/$1.gnuplot\nopen $1.pdf\n
    \n

    The first few lines of this script do the same thing that is normally done when there is no build option in the target. Specifically, they use cmake to create a makefile, invoke make, and then move the executable to the bin directory. The next line, however, gives new functionality. It executes the compiled code! The final two lines assume that the program has produced a file with data to be plotted and use gnuplot to plot the data. This requires, of course, that you have gnuplot installed, and that there is a file called Foo.gnuplot in the same directory as Foo.lf. The file Foo.gnuplot contains the commands to plot the data, and might look something like the following:

    \n
    set title 'My Title'\nset xrange [0:3]\nset yrange [-2:2]\nset xlabel "Time (seconds)"\nset terminal pdf size 5, 3.5\nset output 'Foo.pdf'\nplot 'mydata1.data' using 1:2 with lines, \\\n     'mydata2.data' using 1:2 with lines\n
    \n

    This assumes that your program has written two files, mydata1.data and mydata2.data containing two columns, time and value.

    \n
    \n

    build-type

    \n
    \n

    The $target-language$ target does not currently support the build-type target option.

    \n
    \n
    \n

    This parameter works with cargo to specify how to compile the code. The following options are supported:

    \n
      \n
    • Release: Optimization is turned on and debug information is missing.
    • \n
    • Debug: Debug information is included in the executable.
    • \n
    • RelWithDebInfo: Optimization with debug information.
    • \n
    • MinSizeRel: Optimize for smallest size.
    • \n
    \n

    This defaults to Release.

    \n
    \n
    \n

    This parameter works with cmake to specify how to compile the code. The following options are supported:

    \n
      \n
    • Release: Optimization is turned on and debug information is missing.
    • \n
    • Debug: Debug information is included in the executable.
    • \n
    • RelWithDebInfo: Optimization with debug information.
    • \n
    • MinSizeRel: Optimize for smallest size.
    • \n
    \n

    This defaults to Release.

    \n
    \n
    \n

    cargo-dependencies

    \n

    This is a list of dependencies to include in the generated Cargo.toml file. The value of this parameter is a map of package name to dependency-spec.

    \n

    Here is an example for defining dependencies:

    \n
    target Rust {\n    cargo-dependencies: {\n        fxhash: {\n            version: "0.2.1",\n        },\n        rand: {\n            version: "0.8",\n            features: ["small_rng"],\n        },\n    }\n};\n
    \n

    cargo-features

    \n

    This is a list of features of the generated crate. Supported are:

    \n\n
    \n

    cmake

    \n
    \n

    The $target-language$ target does not support the cmake target option.

    \n
    \n
    \n

    The $target-language$ target does not support the cmake target option because it always uses cmake.

    \n
    \n
    \n
    target C {\n    cmake: <true or false>\n};\n
    \n

    This will enable or disable the CMake-based build system (the default is true). Enabling the CMake build system will result in a CMakeLists.txt being generated in the src-gen directory. This CMakeLists.txt is then used when cmake is invoked by the LF runtime (either the lfc or the IDE). Alternatively, the generated program can be built manually. To do so, in the src-gen/ProgramName directory, run:

    \n
    mkdir build && cd build\ncmake ../\nmake\n
    \n

    If cmake is disabled, gcc is directly invoked after code generation by default. In this case, additional target properties, such as compiler and flags can be used to gain finer control over the compilation process.

    \n
    \n

    cmake-include

    \n
    \n

    The $target-language$ target does not support the cmake-include target option.

    \n
    \n
    \n
    target C {\n    cmake-include: ["relative/path/to/foo.txt", "relative/path/to/bar.txt", ...]\n};\n
    \n
    target Cpp {\n    cmake-include: ["relative/path/to/foo.txt", "relative/path/to/bar.txt", ...]\n};\n
    \n

    This will optionally append additional custom CMake instructions to the generated CMakeLists.txt, drawing these instructions from the specified text files (e.g, foo.txt). The specified files are resolved using the same file search algorithm as used for the files target parameter. Those files will be copied into the src-gen directory that contains the generated sources. This is done to make the generated code more portable (a feature that is useful in federated execution).

    \n

    The cmake-include target property can be used, for example, to add dependencies on various packages (e.g., by using the find_package and target_link_libraries commands).

    \n

    A CMake variable called ${LF_MAIN_TARGET} can be used in the included text file(s) for convenience. This variable will contain the name of the CMake target (i.e., the name of the main reactor). For example, a foo.txt file can contain:

    \n
    find_package(m REQUIRED) # Finds the m library\n\ntarget_link_libraries( ${LF_MAIN_TARGET} m ) # Links the m library\n
    \n

    foo.txt can then be included by specifying it as an argument to cmake-include.

    \n

    Note: For a general tutorial on finding packages in CMake, see this external documentation entry. For a list of CMake find modules, see this.

    \n

    The cmake-include parameter works in conjunction with the $import$ statement. If any imported .lf file has cmake-include among its target properties, the specified text files will be appended to the current list of cmake-includes. These files will be resolved relative to the imported .lf file using the same search procedure as for the files parameter. This helps resolve dependencies in imported reactors automatically and makes the code more modular.

    \n
    \n
    \n

    CMakeInclude.lf is an example that uses this feature. A more sophisticated example of the usage of this target parameter can be found in Rhythm.lf. A distributed version can be found in DistributedCMakeInclude.lf is a test that uses this feature.

    \n

    Note: For federated execution, both cmake-include and files are kept separate for each federate as much as possible. This means that if one federate is imported, or uses an imported reactor that other federates don’t use, it will only have access to cmake-includes and files defined in the main .lf file, plus the selectively imported .lf files. DistributedCMakeIncludeSeparateCompile.lf is a test that demonstrates this feature.

    \n
    \n
    \n

    See AsyncCallback.lf for an example.

    \n
    \n

    compiler

    \n
    \n

    The $target-language$ target does not support the compiler target option.

    \n
    \n
    \n

    This parameter is a string giving the name of the C compiler to use.\nIt is used only when cmake is set to false. For example:

    \n
    target C {\n    cmake: false,\n    compiler: "cc",\n};\n
    \n

    The compiler option here specifies to use cc rather than gcc.

    \n
    \n
    \n

    This parameter is a string giving the name of the C++ compiler to use. Normally\nCMake selects the best compiler for your system, but you can use this parameter\nto point it to your preferred C++ compiler.

    \n
    \n

    docker

    \n
    \n

    This option takes a boolean argument (default is false).

    \n

    If true, a docker file will be generated in the unfederated case.

    \n

    In the federated case, a docker file for each federate will be generated. A docker-compose file will also be generated for the top-level federated reactor.

    \n
    \n
    \n

    The $target-language$ target does not support the docker target option.

    \n
    \n

    external-runtime-path

    \n
    \n

    The $target-language$ target does not support the external-runtime-path target option.

    \n
    \n
    \n

    This option takes a string argument given a path to a pre-compiled external runtime library to use instead of the default one.

    \n
    \n
    \n

    This option takes a path as string argument to a folder containing an alternative runtime crate to use instead of the default one.

    \n
    \n

    export-dependency-graph

    \n
    \n

    The $target-language$ target does not support the export-dependency-graph target option.

    \n
    \n
    \n

    This parameter takes arguments true or false to specify whether the compiled binary will export its internal dependency graph as a dot graph when executed. This is a debugging utility.

    \n
    \n

    If a CLI is generated, the target property is ignored, and the user should instead use the --export-graph flag of the generated program.

    \n
    \n
    \n

    fast

    \n

    By default, the execution of a Lingua Franca program is slowed down, if necessary, so that logical time does not elapse faster than physical time. If you wish to execute the program as fast as possible without this constraint, then specify the fast target parameter with value true.

    \n

    files

    \n
    \n

    The $target-language$ target does not support the files option.

    \n
    \n
    \n

    The files target parameter specifies array of files or directories to be copied to the directory that contains the generated sources.

    \n
    target C {\n    files: ["file1", "file2", ...]\n}\n
    \n
    target Python {\n    files: ["file1", "file2", ...]\n}\n
    \n

    The lookup procedure for these files and directories is as follows:

    \n

    1- Search in the directory containing the .lf file that has the target directive.

    \n

    2- If not found, search in LF_CLASSPATH.

    \n

    3- If still not found, search in CLASSPATH.

    \n

    4- If still not found, search for the file as a resource. Specifically, if a file begins with a forward slash /, then the path is assumed to be relative to the root directory of the Lingua Franca source tree.

    \n
    \n

    For example, if you wish to use audio on a Mac, you can specify:

    \n
    target C {\n    files: ["/lib/C/util/audio_loop_mac.c", "/lib/C/util/audio_loop.h"]\n}\n
    \n

    Your preamble code can then include these files, for example:

    \n
    preamble {=\n    #include "audio_loop_mac.c"\n=}\n
    \n

    Your reactions can then invoke functions defined in that .c file.

    \n

    Sometimes, you will need access to these files from target code in a reaction. For the C target (at least), the generated program will contain a line like this:

    \n
        #define TARGET_FILES_DIRECTORY "path"\n
    \n

    where path is the full path to the directory containing these files. This can be used in reactions, for example, to read those files.

    \n
    \n

    Moreover, the files target specification works in conjunction with the $import$ statement. If a .lf file is imported and has designated supporting files using the files target parameter, those files will be resolved relative to that .lf file and copied to the directory that contains the generated sources. This is done to make code that imports other .lf files more modular. Rhythm.lf is an example that demonstrates most of these features.

    \n
    \n

    flags

    \n
    \n

    The $target-language$ target does not support the flags parameter.

    \n
    \n
    \n

    This parameter is a list of strings giving additional arguments to pass to the target language compiler.\nIt is used only when cmake is set to false. For example:

    \n
    target C {\n    cmake: false,\n    flags: ["-g", "-I/usr/local/include", "-L/usr/local/lib", "-lpaho-mqtt3c"],\n};\n
    \n

    The flags option specifies to include debug information in the compiled code (-g); a directory to search for include files (-I/usr/local/include); a directory to search for library files (-L/usr/local/lib); a library to link with (-lpaho-mqtt3c, which will link with file libpaho-mqtt3c.so).

    \n

    Note: Using the flags standard parameter when cmake is enabled is strongly discouraged, although supported. Flags are compiler-specific, and thus interfere with CMake’s ability to find the most suitable compiler for each platform. In a similar fashion, we recommend against the use of the compiler standard parameter for the same reason. A better solution is to provide a cmake-include file, as described next.

    \n
    \n

    logging

    \n
    \n

    The $target-language$ target does not support the logging parameter.

    \n
    \n
    \n

    By default, when executing a generated Lingua Franca program, error messages, warnings, and informational messages are printed to standard out. You can get additional information printed by setting the logging parameter to LOG or DEBUG (or log or debug). The latter is more verbose. If you set the logging parameter to warn, then warnings and errors will be printed, but informational messages will not (e.g. message produced using the info_print utility function). If you set logging to error, then warning messages will also not be printed.

    \n

    The C and Python targets also support tracing, which outputs binary traces of an execution rather than human-readable text and is designed to have minimal impact on performance.

    \n
    \n
    \n

    The logging option is one of error, warn, info, log or debug. It specifies the level of diagnostic messages about execution to print to the console. A message will print if this parameter is greater than or equal to the level of the message, where error < warn < info < log < debug. The default value is info, which means that messages log or debug messages will not print.

    \n
    \n
    \n

    The logging option is one of ERROR, WARN, INFO, LOG or DEBUG. It specifies the level of diagnostic messages about execution to print to the console. A message will print if this parameter is greater than or equal to the level of the message, where ERROR < WARN < INFO < LOG < DEBUG. The default value is INFO, which means that messages tagged LOG and DEBUG will not print. Internally this is handled by the ulog module.

    \n
    \n

    no-compile

    \n
    \n

    The $target-language$ target does not support the no-compile target option.

    \n
    \n
    \n

    If true, then do not invoke a target language compiler nor cmake. Just generate code.

    \n
    \n

    no-runtime-validation

    \n
    \n

    The $target-language$ target does not support the no-runtime-validation target option.

    \n
    \n
    \n

    This parameter takes value true or false (the default). If this is set to true, then all runtime checks in reactor-cpp will be disabled. This brings a slight performance boost but should be used with care and only on tested programs.

    \n
    \n

    protobufs

    \n
    \n

    The $target-language$ target does not support the protobufs target option.

    \n
    \n
    \n

    Protobufs is a serialization protocol by which data in a target language can be copied over the network to a remote location. The protobufs target parameter gives an array of .proto files that are to be compiled and included in the generated code. For an example, see\nPersonProtocolBuffers.lf\nPersonProtocolBuffers.lf.

    \n
    \n

    runtime-version

    \n
    \n

    The $target-language$ target does not support the runtime-version target option.

    \n
    \n
    \n

    This argument takes a string (with quotation marks) containing any tag, branch name, or git hash in the reactor-cpp repository. This will specify the version of the runtime library that the compiled binary will link against.

    \n
    \n
    \n

    rust-include

    \n

    This specifies a set of Rust modules in the generated project. See Linking support files.

    \n

    scheduler

    \n

    This specifies the scheduler to use. SeeTarget Language Details.

    \n

    single-file-project

    \n

    If true, enables single-file project layout.

    \n
    \n

    threading

    \n
    \n

    The $target-language$ target does not support the threading target option.

    \n
    \n
    \n

    If threading is disabled (by setting threading to false), then no thread library is used, and the lf_schedule() function is not thread safe. This setting is incompatible with asynchronously scheduling any physical actions and hence this parameter will be ignored for programs that have physical actions.\nSee workers.

    \n
    \n
    \n

    The Python target uses the single threaded C runtime by default but will switch to the multithreaded C runtime if a physical action is detected. This target property can be used to override this behavior.

    \n
    \n
    \n

    Boolean flag (either true (default) or false) that controls if the project is to be compiled with support for multi-threading.

    \n

    See workers.

    \n
    \n

    timeout

    \n

    A time value (with units) specifying the logical stop time of execution. See Termination.

    \n

    workers

    \n
    \n

    The $target-language$ target does not support the workers target option.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. If threading is turned on (the default, see threading), then the generated code will use a target platform thread library and generate multi-threaded code. This can transparently execute reactions that have no dependence on one another in parallel on multiple cores. By default, threading is turned on, and the workers property is set to 0, which means that the number of workers is determined by the runtime system. Typically, it will be set to the number of cores on the machine running the code. To use a different number of worker threads, give a positive integer for this target parameter.

    \n

    With value 0, the runtime engine is free to choose the number of worker threads to use. Typically, this will equal the number of hardware threads on the machine on which the Lingua Franca code generator is run.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. With value 0 (the default), the runtime engine is free to choose the number of worker threads to use. In the $target-language$ target, the runtime system will determine the number of hardware threads on the machine on which the program is run and set the number of worker threads equal to that number.

    \n

    If the workers property is set to 1, the scheduler will not create any worker threads and instead inline the execution of reactions. This is an optimization and avoids any unnecessary synchronization. Note that, in contrast to the C target, the single threaded implementation is still thread safe and asynchronous reaction scheduling is supported.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. Note, however, that the Python core is unable to execute safely in parallel on multiple cores. As a consequence, at execution time, each reaction invocation will acquire a mutual exclusion lock before executing. Hence, there is little point in setting this to any number greater than 1.

    \n
    \n
    \n

    This parameter takes a non-negative integer and specifies the number of worker threads to execute the generated program. With value 0 (the default), the runtime engine is free to choose the number of worker threads to use and the number of worker threads may vary over time.

    \n
    \n

    Command-Line Arguments

    \n
    \n

    The generated executable may feature a command-line interface (CLI), if it uses the cargo-features: [\"cli\"] target property. When that feature is enabled:

    \n
      \n
    • some target properties become settable at runtime:\n
        \n
      • --timeout <time value>: override the default timeout mentioned as a target property. The syntax for times is just like the LF one (e.g. 1msec, \"2 seconds\").
      • \n
      • --workers <number>: override the default worker count mentioned as a target property. This option is ignored unless the runtime crate has been built with the feature parallel-runtime.
      • \n
      • --export-graph: export the dependency graph (corresponds to export-dependency-graph target property). This is a flag, i.e., absent means false, present means true. This means the value of the target property is ignored and not used as default.
      • \n
      • --log-level: corresponds to the logging target property, but note that the levels have different meanings, and the target property is ignored. See Logging levels.
      • \n
      \n
    • \n
    • parameters of the main reactor are translated to CLI parameters.\n
        \n
      • Each LF parameter named param corresponds to a CLI parameter named --main-param. Underscores in the LF parameter name are replaced by hyphens.
      • \n
      • The type of each parameters must implement the trait FromStr.
      • \n
      \n
    • \n
    \n

    When the cli feature is disabled, the parameters of the main reactor will each assume their default value.

    \n
    \n
    \n

    The generated C program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast [true | false]: Specifies whether to wait for physical time to match logical time. The default is false. If this is true, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout <duration> <units>: Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those.
    • \n
    • -w, --workers <n>: Executed using worker threads if possible. This option is ignored in the single-threaded version. That is, it is ignored if a threading option was given in the target properties with value false.
    • \n
    • -i, --id <n>: The ID of the federation that this reactor will join.
    • \n
    \n

    Any other command-line arguments result in printing the above information.

    \n
    \n
    \n

    The generated C++ program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast: If set, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout '<duration> <units>': Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those.
    • \n
    • -w, --workers <n>: Use n worker threads for executing reactions.
    • \n
    • --help: Print the above information.
    • \n
    \n

    If the main reactor declares parameters, these parameters will appear as additional CLI options that can be specified when invoking the binary (see Using Parameters).

    \n
    \n
    \n

    The Python target does not currently support any command-line arguments. You must specify properties as target parameters.

    \n
    \n
    \n

    In the TypeScript target, the generated JavaScript program understands the following command-line arguments, each of which has a short form (one character) and a long form:

    \n
      \n
    • -f, --fast [true | false]: Specifies whether to wait for physical time to match logical time. The default is false. If this is true, then the program will execute as fast as possible, letting logical time advance faster than physical time.
    • \n
    • -o, --timeout '<duration> <units>': Stop execution when logical time has advanced by the specified duration. The units can be any of nsec, usec, msec, sec, minute, hour, day, week, or the plurals of those. For the duration and units of a timeout argument to be parsed correctly as a single value, these should be specified in quotes with no leading or trailing space (e.g. ‘5 sec’).
    • \n
    • -k, --keepalive [true | false]: Specifies whether to stop execution if there are no events to process. This defaults to false, meaning that the program will stop executing when there are no more events on the event queue. If you set this to true, then the program will keep executing until either the timeout logical time is reached or the program is externally killed. If you have physical actions, it usually makes sense to set this to true.
    • \n
    • -l, --logging [ERROR | WARN | INFO | LOG | DEBUG]: The level of logging messages from the reactor-ts runtime to to print to the console. Messages tagged with a given type (error, warn, etc.) will print if this argument is greater than or equal to the level of the message (ERROR < WARN < INFO < LOG < DEBUG).
    • \n
    • -h, --help: Print this usage guide. The program will not execute if this flag is present.
    • \n
    \n

    If provided, a command line argument will override whatever value the corresponding target property had specified in the source .lf file.

    \n

    Command line options are parsed by the command-line-arguments module with these rules. For example

    \n
    node <LF_file_name>/dist/<LF_file_name>.js -f false --keepalive=true -o '4 sec' -l INFO\n
    \n

    is a valid setting.

    \n

    Any errors in command-line arguments result in printing the above information. The program will not execute if there is a parsing error for command-line arguments.

    \n

    Custom Command-Line Arguments

    \n

    User-defined command-line arguments may be created by giving the main reactor parameters. Assigning the main reactor a parameter of type string, number, boolean, or time will add an argument with corresponding name and type to the generated program’s command-line-interface. Custom arguments will also appear in the generated program’s usage guide (from the --help option). If the generated program is executed with a value specified for a custom command-line argument, that value will override the default value for the corresponding parameter. Arguments typed string, number, and boolean are parsed in the expected way, but time arguments must be specified on the command line like the --timeout property as '<duration> <units>' (in quotes).

    \n

    Note: Custom arguments may not have the same names as standard arguments like timeout or keepalive.

    \n

    For example this reactor has a custom command line argument named customArg of type number and default value 2:

    \n
    target TypeScript;\nmain reactor clArg(customArg:number(2)) {\n    reaction (startup) {=\n        console.log(customArg);\n    =}\n}\n
    \n

    If this reactor is compiled from the file simpleCLArgs.lf, executing

    \n
    node simpleCLArgs/dist/simpleCLArgs.js\n
    \n

    outputs the default value 2. But running

    \n
    node simpleCLArgs/dist/simpleCLArgs.js --customArg=42\n
    \n

    outputs 42. Additionally, we can view documentation for the custom command line argument with the --help command.

    \n
    node simpleCLArgs/dist/simpleCLArgs.js -h\n
    \n

    The program will generate the standard usage guide, but also

    \n
    --customArg '<duration> <units>'                    Custom argument. Refer to\n                                                      <path>/simpleCLArgs.lf\n                                                      for documentation.
    \n

    Additional types for Custom Command-Line Arguments

    \n

    Main reactor parameters that are not typed string, number, boolean, or time will not create custom command-line arguments. However, that doesn’t mean it is impossible to obtain other types from the command line, just use a string and specify how the parsing is done yourself. See below for an example of a reactor that parses a custom command-line argument of type string into a state variable of type Array<number> using JSON.parse and a user-defined type guard.

    \n
    target TypeScript;\nmain reactor customType(arrayArg:string("")) {\n    preamble {=\n        function isArrayOfNumbers(x: any): x is Array<number> {\n            for (let item of x) {\n                if (typeof item !== "number") {\n                    return false;\n                }\n            }\n            return true;\n        }\n    =}\n    state foo:{=Array<number>=}({=[]=});\n    reaction (startup) {=\n        let parsedArgument = JSON.parse(customType);\n        if (isArrayOfNumbers(parsedArgument)) {\n            foo = parsedArgument;\n            }\n        else {\n            throw new Error("Custom command line argument is not an array of numbers.");\n        }\n        console.log(foo);\n    =}\n}\n
    \n
    ","headings":[{"value":"Summary of Parameters","depth":1},{"value":"auth","depth":2},{"value":"build","depth":2},{"value":"build-type","depth":2},{"value":"cargo-dependencies","depth":2},{"value":"cargo-features","depth":2},{"value":"cmake","depth":2},{"value":"cmake-include","depth":2},{"value":"compiler","depth":2},{"value":"docker","depth":2},{"value":"external-runtime-path","depth":2},{"value":"export-dependency-graph","depth":2},{"value":"fast","depth":2},{"value":"files","depth":2},{"value":"flags","depth":2},{"value":"logging","depth":2},{"value":"no-compile","depth":2},{"value":"no-runtime-validation","depth":2},{"value":"protobufs","depth":2},{"value":"runtime-version","depth":2},{"value":"rust-include","depth":2},{"value":"scheduler","depth":2},{"value":"single-file-project","depth":2},{"value":"threading","depth":2},{"value":"timeout","depth":2},{"value":"workers","depth":2},{"value":"Command-Line Arguments","depth":1},{"value":"Custom Command-Line Arguments","depth":3},{"value":"Additional types for Custom Command-Line Arguments","depth":3}],"frontmatter":{"permalink":"/docs/handbook/target-declaration","title":"Target Declaration","oneline":"The target declaration and its parameters in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Target Language Details","oneline":"Detailed reference for each target langauge.","permalink":"/docs/handbook/target-language-details"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Tracing","oneline":"Tracing (preliminary)","permalink":"/docs/handbook/tracing"}}}},"pageContext":{"id":"3-target-declaration","slug":"/docs/handbook/target-declaration","repoPath":"/packages/documentation/copy/en/reference/Target Declaration.md","previousID":"15e2a8e5-dd68-55e9-839f-18c6d342e73c","nextID":"3f4000b5-1133-5c34-807d-29c05884f149","lang":"en","modifiedTime":"2023-11-08T00:42:45.524Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/target-language-details/page-data.json b/page-data/docs/handbook/target-language-details/page-data.json index 46ce58430..6d3b1ea66 100644 --- a/page-data/docs/handbook/target-language-details/page-data.json +++ b/page-data/docs/handbook/target-language-details/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/target-language-details","result":{"data":{"markdownRemark":{"id":"3af06b85-61a2-5d81-a7ad-232624b3eaf6","excerpt":"$page-showing-target$ Overview In the C reactor target for Lingua Franca, reactions are written in C and the code generator generates one or more standalone C…","html":"

    $page-showing-target$

    \n

    Overview

    \n
    \n

    In the C reactor target for Lingua Franca, reactions are written in C and the code generator generates one or more standalone C programs that can be compiled and run on several platforms. It has been tested on macOS, Linux, Windows, and at least one bare-iron embedded platform. The single-threaded version (which you get by setting the threading target parameter to false) is the most portable, requiring only a handful of common C libraries (see Included Libraries below). The multithreaded version requires a small subset of the POSIX thread library (pthreads) and transparently executes in parallel on a multicore machine while preserving the deterministic semantics of Lingua Franca.

    \n

    Note that C is not a safe language. There are many ways that a programmer can circumvent the semantics of Lingua Franca and introduce nondeterminism and illegal memory accesses. For example, it is easy for a programmer to mistakenly send a message that is a pointer to data on the stack. The destination reactors will very likely read invalid data. It is also easy to create memory leaks, where memory is allocated and never freed. Here, we provide some guidelines for a style for writing reactors that will be safe.

    \n

    NOTE: If you intend to use C++ code or import C++ libraries in the C target, we provide a special CCpp target that automatically uses a C++ compiler by default. Alternatively, you might want to use the Cpp target.

    \n
    \n
    \n

    In the C++ reactor target for Lingua Franca, reactions are written in C++ and the code generator generates a standalone C++ program that can be compiled and run on all major platforms. Our continuous integration ensures compatibility with Windows, macOS, and Linux.\nThe C++ target solely depends on a working C++ build system including a recent C++ compiler (supporting C++17) and CMake (>= 3.5). It relies on the reactor-cpp runtime, which is automatically fetched and compiled in the background by the Lingua Franca compiler.

    \n

    Note that C++ is not a safe language. There are many ways that a programmer can circumvent the semantics of Lingua Franca and introduce nondeterminism and illegal memory accesses. For example, it is easy for a programmer to mistakenly send a message that is a pointer to data on the stack. The destination reactors will very likely read invalid data. It is also easy to create memory leaks, where memory is allocated and never freed. Note, however, that the C++ reactor library is designed to prevent common errors and to encourage a safe modern C++ style. Here, we introduce the specifics of writing Reactor programs in C++ and present some guidelines for a style that will be safe.

    \n
    \n
    \n

    In the Python reactor target for Lingua Franca, reactions are written in Python. The user-written reactors are then generated into a Python 3 script that can be executed on several platforms. The Python target has been tested on Linux, macOS, and Windows. To facilitate efficient and fast execution of Python code, the generated program relies on a C extension to facilitate Lingua Franca API such as set and schedule. To learn more about the structure of the generated Python program see Implementation Details.

    \n

    Python reactors can bring the vast library of scientific modules that exist for Python into a Lingua Franca program. Moreover, since the Python reactor target is based on a fast and efficient C runtime library, Lingua Franca programs can execute much faster than native equivalent Python programs in many cases. Finally, interoperability with C reactors is planned for the future.

    \n

    In comparison to the C target, the Python target can be up to an order of magnitude slower. However, depending on the type of application and the implementation optimizations in Python, you can achieve an on-par performance to the C target in many applications.

    \n

    NOTE: A Python C\nextension is\ngenerated for each Lingua Franca program (see Implementation\nDetails). This extension module will\nhave the name LinguaFranca[your_LF_program_name].

    \n
    \n
    \n

    In the TypeScript reactor target for Lingua Franca, reactions are written in TypeScript and the code generator generates a standalone TypeScript program that can be compiled to JavaScript and run on Node.js.

    \n

    TypeScript reactors bring the strengths of TypeScript and Node.js to Lingua Franca programming. The TypeScript language and its associated tools enable static type checking for both reaction code and Lingua Franca elements like ports and actions. The Node.js JavaScript runtime provides an execution environment for asynchronous network applications. With Node.js comes Node Package Manager (npm) and its large library of supporting modules.

    \n

    In terms of raw performance on CPU intensive operations, TypeScript reactors are about two orders of magnitude slower than C reactors. But excelling at CPU intensive operations isn’t really the point of Node.js (or by extension TypeScript reactors). Node.js is about achieving high throughput on network applications by efficiently handling asynchronous I/O operations. Keep this in mind when choosing the right Lingua Franca target for your application.

    \n
    \n
    \n

    Important: The Rust target is still quite preliminary. This is early WIP documentation to let you try it out if you’re curious

    \n

    In the Rust reactor target for Lingua Franca, reactions are written in Rust and the code generator generates a standalone Rust program that can be compiled and run on platforms supported by rustc. The program depends on a runtime library distributed as the crate reactor_rt, and depends on the Rust standard library.

    \n

    Documentation for the runtime API is available here: https://lf-lang.github.io/reactor-rust/

    \n

    LF-Rust generates a Cargo project per compiled main reactor. This specification assumes in some places that the user is somewhat familiar with how Cargo works.\nIf you’re not, here’s a primer:

    \n
      \n
    • a Rust project (and its library artifact) are called a crate.
    • \n
    • Cargo is the Rust package manager and build tool. LF/Rust uses Cargo to build the generated project.
    • \n
    • Rust has extensive support for conditional compilation. Cargo features are commonly used to enable or disable the compilation of parts of a crate. A feature may also pull in additional dependencies. Cargo features only influence the compilation process; if you don’t mention the correct feature flags at compilation time, those features cannot be made available at runtime. The Rust reactor runtime crate uses Cargo features to conditionally enable some features, e.g., command-line argument parsing.
    • \n
    \n
    \n

    Requirements

    \n
    \n

    The following tools are required in order to compile the generated C source code:

    \n
      \n
    • A C compiler such as gcc
    • \n
    • A recent version of cmake (at least 3.5)
    • \n
    \n
    \n
    \n

    The following tools are required in order to compile the generated C++ source code:

    \n
      \n
    • A recent C++ compiler supporting C++17
    • \n
    • A recent version of cmake (at least 3.5)
    • \n
    \n
    \n
    \n

    To use this target, install Python 3 on your machine. See downloading Python.

    \n

    NOTE: The Python target requires a C implementation of Python (nicknamed CPython). This is what you will get if you use the above link, or with most of the alternative Python installations such as Anaconda. See Python download alternatives for more details.

    \n

    The Python reactor target relies on setuptools to be able to compile a Python\nC extension for each LF\nprogram.

    \n\n

    To install setuptools using pip3, do this:

    \n
    pip3 install setuptools\n
    \n
    \n
    \n

    First, make sure Node.js is installed on your machine. You can download Node.js here. The npm package manager comes along with Node.

    \n

    After installing Node, you may optionally install the TypeScript compiler.

    \n
    npm install -g typescript\n
    \n

    TypeScript reactor projects are created with a local copy of the TypeScript compiler, but having the TypeScript compiler globally installed can be useful for debugging type errors and type checking on the command line.

    \n
    \n
    \n

    In order to compile the generated Rust source code, you need a recent version of Cargo, the Rust package manager. See How to Install Rust and Cargo if you don’t have them on your system.

    \n

    You can use a development version of the runtime library by setting the LFC option --external-runtime-path to the root directory of the runtime library crate sources. If this variable is mentioned, LFC will ask Cargo to fetch the runtime library from there.

    \n
    \n

    Limitations

    \n
    \n
      \n
    • The C target does make any distinction between $private$ and $public$ $preamble$.
    • \n
    \n
    \n
    \n

    The C++ target does not yet implement:

    \n\n
    \n
    \n
      \n
    • The Lingua Franca lexer does not support single-quoted strings in Python. This limitation also applies to target property values. You must use double quotes.
    • \n
    \n
    \n
    \n
      \n
    • \n

      The $federated$ implementation in the TypeScript target is still quite preliminary.

      \n
    • \n
    • \n

      The TypeScript target does not yet implement methods.

      \n
    • \n
    • \n

      The TypeScript target does not yet implement modal reactors

      \n
    • \n
    \n
    \n
    \n

    The Rust target does not yet implement:

    \n\n
    \n

    The Target Specification

    \n
    \n

    To have Lingua Franca generate C code, start your .lf file with one of the following target specifications:

    \n
      target C <options>\n  target CCpp <options>\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in C, you may want to include the final semicolon for uniformity.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n

    The second form, CCpp, is used when you wish to use a C++ compiler to compile\nthe generated code, thereby allowing your C reactors to call C++ code.

    \n\n

    Here is a minimal example of a program written in the CCpp target, taken from HelloWorldCCPP.lf:

    \n
    target CCpp\nreactor HelloWorld {\n  preamble {=\n    #include <iostream> // Note that no C++ header will be included by default.\n  =}\n  reaction(startup) {=\n    std::cout << "Hello World." << std::endl;\n  =}\n}\nmain reactor {\n  a = new HelloWorld()\n}\n
    \n

    Note: Unless some feature in the C target is needed, we recommend using the Cpp target that uses a runtime that is written natively in C++.

    \n

    Note: A .lf file that uses the CCpp target cannot and should not be imported to a .lf file that uses the C target. Although these two targets use essentially the same runtime, such a scenario can cause unintended compile errors.

    \n
    \n
    \n

    To have Lingua Franca generate C++ code, start your .lf file with the following target specification:

    \n
        target Cpp\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in C++, you may want to include the final semicolon for uniformity.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n
    \n
    \n

    To have Lingua Franca generate Python code, start your .lf file with the following target specification:

    \n
        target Python\n
    \n

    Note that for all LF statements, a final semicolon is optional.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n
    \n
    \n

    To have Lingua Franca generate TypeScript code, start your .lf file with the following target specification:

    \n
        target TypeScript\n
    \n

    Note that for all LF statements, a final semicolon is optional.

    \n

    The supported target parameters and command-line options are documented in the Target Declaration documentation.

    \n
    \n
    \n

    To have Lingua Franca generate Rust code, start your .lf file with the following target specification:

    \n
        target Rust\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in Rust, you may want to include the final semicolon for uniformity.

    \n
    \n

    Parameters and State Variables

    \n
    \n

    Reactor parameters and state variables are referenced in the C code using the\nself struct. The following\nStride\nexample modifies the Count reactor in State\nDeclaration to\ninclude both a parameter and a state variable:

    \n
    reactor Count(stride: int = 1) {\n  state count: int = 1\n  output y: int\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    lf_set(y, self->count);\n    self->count += self->stride;\n  =}\n}\n
    \n

    This defines a stride parameter with type int and initial value 1 and\na count state variable with the same type and initial value.\nThese are referenced in the reaction with the syntax self->stride and self->count respectively.

    \n

    The self Struct:\nThe code generator synthesizes a struct type in C for each reactor class and a constructor that creates an instance of this struct. By convention, these instances are called self and are visible within each reactor body. The self struct contains the parameters, state variables, and values associated with actions and ports of the reactor. Parameters and state variables are accessed directly on the self struct, whereas ports and actions are directly in scope by name, as we will see below. Let’s begin with parameters.

    \n

    It may be tempting to declare state variables in the $preamble$, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    int count = 0;\n  =}\n  output y: int\n  timer t(0, 100 msec)\n\n  reaction(t) -> y {=\n    lf_set(y, count++);\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance of the reactor, those instances will share the same variable count. Hence, don’t do this! Sharing variables across instances of reactors violates a basic principle, which is that reactors communicate only by sending messages to one another. Sharing variables will make your program nondeterministic. If you have multiple instances of the above FlawedCount reactor, the outputs produced by each instance will not be predictable, and in a multithreaded implementation, will also not be repeatable.

    \n

    Array Values for Parameters

    \n

    Parameters and state variables can have array values, though some care is needed. The ArrayAsParameter example outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence: int[] = {0, 1, 2}, n_sequence: int = 3) {\n  output out: int\n  state count: int = 0\n  logical action next\n\n  reaction(startup, next) -> out, next {=\n    lf_set(out, self->sequence[self->count]);\n    self->count++;\n    if (self->count < self->n_sequence) {\n      lf_schedule(next, 0);\n    }\n  =}\n}\n
    \n

    This uses a $logical$ $action$ to repeat the reaction, sending one element of the array in each invocation.

    \n

    In C, arrays do not encode their own length, so a separate parameter n_sequence is used for the array length. Obviously, there is potential here for errors, where the array length doesn’t match the length parameter.

    \n

    Above, the parameter default value is an array with three elements, [0, 1, 2]. The syntax for giving this default value is that of a Lingua Franca list, {0, 1, 2}, which gets converted by the code generator into a C static initializer. The default value can be overridden when instantiating the reactor using a similar syntax:

    \n
      s = new Source(sequence = {1, 2, 3, 4}, n_sequence=4)\n
    \n

    Array Values for States

    \n

    A state variable can also have an array value. For example, the MovingAverage reactor computes the moving average of the last four inputs each time it receives an input:

    \n
    reactor MovingAverageImpl {\n  state delay_line: double[] = {0.0, 0.0, 0.0}\n  state index: int = 0\n  input in: double\n  output out: double\n\n  reaction(in) -> out {=\n    // Calculate the output.\n    double sum = in->value;\n    for (int i = 0; i < 3; i++) {\n      sum += self->delay_line[i];\n    }\n    lf_set(out, sum/4.0);\n\n    // Insert the input in the delay line.\n    self->delay_line[self->index] = in->value;\n\n    // Update the index for the next input.\n    self->index++;\n    if (self->index >= 3) {\n      self->index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an array of doubles with the initial value of the array being a three-element array filled with zeros.

    \n

    States and Parameters with Struct Values

    \n

    States whose type are structs can similarly be initialized. This StructAsState example illustrates this:

    \n
    target C\npreamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nmain reactor StructAsState {\n  state s: hello_t = {"Earth", 42}\n  reaction(startup) {=\n    printf("State s.name=\\"%s\\", value=%d.\\n", self->s.name, self->s.value);\n  =}\n}\n
    \n

    Notice that state s is given type hello_t, which is defined in the $preamble$. The initial value just lists the initial values of each of the fields of the struct in the order they are declared.

    \n

    Parameters are similar:

    \n
    target C\npreamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nmain reactor StructParameter(p: hello_t = {"Earth", 42}) {\n  reaction(startup) {=\n    printf("Parameter p.name=\\"%s\\", value=%d.\\n", self->p.name, self->p.value);\n  =}\n}\n
    \n
    \n
    \n

    Reactor parameters are internally declared as const by the code generator and initialized during reactor instantiation. Thus, the value of a parameter may not be changed. See Parameters and State for examples.

    \n

    Array-Valued Parameters

    \n

    Also parameters can have fixed- or variable-sized array values. The ArrayAsParameter example outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence: std::vector<int> = {0, 1, 2}) {\n  output out: size_t\n  state count: size_t = 0\n  logical action next: void\n\n  reaction(startup, next) -> out, next {=\n    out.set(sequence[count]);\n    count++;\n    if (count < sequence.size()) {\n      next.schedule();\n    }\n  =}\n}\n
    \n

    Here, the type of sequence is explicitly given as std::vector<int>.\nA more compact alternative syntax is as follows:

    \n
    sequence: int[] = {0, 1, 2}
    \n

    The type int[] is converted to std::vector<int> by the code generator.\nAnother alternative syntax is:

    \n
    sequence: int[]({0, 1, 2})
    \n

    Here, the static initializer {0, 1, 2} is passed as a single argument to the constructor of std::vector.

    \n

    The main reactor can be parameterized:

    \n
    main reactor Hello(msg: std::string("World")) {\n  reaction(startup) {=\n    std::cout << "Hello " << msg << "!\\n";\n  =}\n}\n
    \n

    This program will print “Hello World!” by default. However, since msg is a main reactor parameter, the C++ code generator will extend the command-line argument parser and allow to override msg when invoking the program. For instance,

    \n
    bin/Hello --msg Earth\n
    \n

    will result in “Hello Earth!” being printed.

    \n

    State Variables

    \n

    A reactor may declare state variables, which become properties of each instance of the reactor. For example, the following reactor (see Count.lf and CountTest.lf) will produce the output sequence 1, 2, 3, … :

    \n
    reactor Count {\n  state count: int = 0\n  output c: int\n  timer t(0, 1 s)\n  reaction(t) -> c {=\n    count++;\n    c.set(count);\n  =}\n}\n
    \n

    The declaration on the second line gives the variable the name count, declares its type to be int, and initializes its value to 0. The type and initial value can be enclosed in the C++-code delimiters {= ... =} if they are not simple identifiers, but in this case, that is not necessary.

    \n

    In the body of the reaction, the state variable is automatically in scope and can be referenced directly by its name. Since all reactions, state variables, and parameters of a reactor are members of the same class, reactions can also reference state variables (or parameters) using the this pointer: this->name.

    \n

    A state variable may be a time value, declared as follows:

    \n
      state time_value:time = 100 ms;\n
    \n

    The type of the generated time_value variable will be reactor::Duration, which is an alias for std::chrono::nanoseconds.

    \n

    For the C++ target, Lingua Franca provides two alternative styles for initializing state variables. We can write state foo:int(42) or state foo:int{42}. This allows to distinguish between the different initialization styles in C++. foo:int(42) will be translated to int foo(42) and foo:int{42} will be translated to int foo{42} in the generated code. Generally speaking, the {...} style should be preferred in C++, but it is not always applicable. Hence we allow the LF programmer to choose the style. Due to the peculiarities of C++, this is particularly important for more complex data types. For instance, state foo:std::vector<int>(4,2) would be initialized to the list [2,2,2,2] whereas state foo:std::vector<int>{4,2} would be initialized to the list [4,2].

    \n

    State variables can have array values. For example, the [MovingAverage] (https://github.com/lf-lang/lingua-franca/blob/master/test/Cpp/src/MovingAverage.lf) reactor computes the moving average of the last four inputs each time it receives an input:

    \n
    reactor MovingAverageImpl {\n  state delay_line: double[3]{0.0, 0.0, 0.0}\n  state index: int = 0\n  input in: double\n  output out: double\n\n  reaction(in) -> out {=\n    // Calculate the output.\n    double sum = *in.get();\n    for (int i = 0; i < 3; i++) {\n      sum += delay_line[i];\n    }\n    out.set(sum/4.0);\n\n    // Insert the input in the delay line.\n    delay_line[index] = *in.get();\n\n    // Update the index for the next input.\n    index++;\n    if (index >= 3) {\n      index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an fixed-size array of 3 doubles with the initial value of the being filled with zeros (note the curly braces). If the size is given in the type specification, then the code generator will declare the type of the state variable using std::array. In the example above, the type of delay_line is std::array<3, double>. If the size specifier is omitted (e.g. state x:double[]). The code generator will produce a variable-sized array using std::vector.

    \n

    The second line can equivalently be given with an assignment operator:

    \n
      state delay_line: double[3] = {0.0, 0.0, 0.0}
    \n

    State variables with more complex types such as classes or structs can be similarly initialized. See StructAsState.lf.

    \n
    \n
    \n

    Reactor parameters and state variables are referenced in the Python code using\nthe syntax self.<name>, where <name> is the name of the parameter or state\nvariable. The following\nStride\nexample modifies the Count reactor in State\nDeclaration to\ninclude both a parameter and a state variable:

    \n
    reactor Count(stride=1) {\n  state count = 1\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(self.count)\n    self.count += self.stride\n  =}\n}\n
    \n

    This defines a stride parameter with initial value 1 and a count state\nvariable with the same initial value. These are referenced in the reaction with\nthe syntax self.stride and self.count respectively. Note that state\nvariables and parameters do not have types in the Python reactor target. See Parameters\nand State for more examples.

    \n

    The Reactor Class:\nThe code generator synthesizes a class in Python for each reactor class in LF,\nwith a constructor (i.e., def __init__(self, ...):) that creates an instance\nof this class and initializes its parameters and state variables as instance\nvariables.\nThese parameters and state variables can then subsequently be accessed directly\nusing the self reference in the body of reactions.

    \n

    It may be tempting to declare state variables in the $preamble$, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    count = 0\n  =}\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(count)\n    count += 1\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance\nof the reactor, those instances will share the same variable count (because\ncount will be a class variable). Hence,\ndon’t do this! Sharing variables across instances of reactors violates a\nbasic principle, which is that reactors communicate only by sending messages to\none another. Sharing variables will make your program nondeterministic. If you\nhave multiple instances of the above FlawedCount reactor, the outputs produced\nby each instance will not be predictable, and in a multithreaded implementation,\nwill also not be repeatable.

    \n

    Array Expressions for State Variables and Parameters

    \n

    Array parameters and state variables are implemented using Python lists and initialized using a parentheized list. In the following example, the\nparameter sequence and the state variable x have an initial value that is a Python list [1, 2, 3]:

    \n
    reactor Foo(param = {= [1, 2, 3] =}) {\n  state x = {= [1, 2, 3] =}\n  ...\n}\n
    \n

    Their elements may be accessed as arrays in the body of a reaction, for example self.x[i], where i is an array index.

    \n

    The parameter may be overridden with a different list at instantiation:

    \n
    main reactor {\n  f = new Foo(param = {= [3, 4, 5, 6]} )\n}\n
    \n

    As with any ordinary Python list or tuple, len() can been used to deduce the\nlength. In the above, len(self.x) and len(self.param) will return the lengths of the two lists.

    \n

    Assigning Arbitrary Initial Expressions to State Variables and Parameters

    \n

    As used for lists above, the code delimiters {= ... =} can allow for assignment of arbitrary Python\nexpressions as initial values for state variables and parameters. The following example, taken from\nStructAsState.lf\ndemonstrates this usage:

    \n
    main reactor StructAsState {\n  preamble {=\n    class hello:\n      def __init__(self, name, value):\n        self.name = name\n        self.value = value\n  =}\n  state s = {= self.hello("Earth", 42) =}\n\n  reaction(startup) {=\n    # will print "State s.name="Earth", value=42."\n    print("State s.name=\\"{:s}\\", value={:d}.".format(self.s.name, self.s.value))\n  =}\n}\n
    \n

    Notice that a class hello is defined in the preamble. The state variable s is then initialized to an instance of hello constructed within the {= ... =} delimiters.

    \n
    \n
    \n

    In the TypeScript target, all TypeScript types are generally acceptable for parameters and state variables. Custom types (and classes) must be defined in the $preamble$ before they may be used.

    \n

    To benefit from type checking, you should declare types for your reactor elements. If a type isn’t declared for a state variable, it is assigned the type unknown.

    \n

    For example, the following reactor will produce the output sequence 0, 1, 2, 3, … :

    \n
    reactor Count {\n  state count:number = 0;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    count++;\n    y = count;\n  =}\n}\n
    \n

    The declaration on the second line gives the variable the name “count”, declares its type to be number, and initializes its value to 0. The type and initial value can be enclosed in the Typescript-code delimiters {= ... =} if they are not simple identifiers, but in this case, that is not necessary.

    \n

    In the body of the reaction, the reactor’s state variable is referenced by way of a local variable of the same name. The local variable will contain the current value of the state at the beginning of the reaction. The final value of the local variable will be used to update the state at the end of the reaction.

    \n

    It may be tempting to declare state variables in the preamble, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    let count = 0;\n  =}\n  output y:number;\n  timer t(0, 100 msec);\n  reaction(t) -> y {=\n    count++;\n    y = count;\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance of the reactor, those instances will share the same variable count. Hence, don’t do this! Sharing variables across instances of reactors violates a basic principle, which is that reactors communicate only by sending messages to one another. Sharing variables will make your program nondeterministic. If you have multiple instances of the above FlawedCount reactor, the outputs produced by each instance will not be predictable, and in an asynchronous implementation, will also not be repeatable.

    \n

    A state variable may be a time value, declared as follows:

    \n
      state time_value:time = 100 ms\n
    \n

    The time_value variable will be of type TimeValue, which is an object used to represent a time in the TypeScript Target. Refer to the section on timed behavior for more information.

    \n

    A state variable can have an array or object value. For example, the following reactor computes the moving average of the last four inputs each time it receives an input (from MovingAverageImpl):

    \n
    reactor MovingAverage {\n  state delay_line: {= Array<number> =} = {= [0.0, 0.0, 0.0] =}\n  state index: number = 0\n  input x: number\n  output out: number\n\n  reaction(x) -> out {=\n    x = x as number;\n    // Calculate the output.\n    let sum = x;\n    for (let i = 0; i < 3; i++) {\n      sum += delay_line[i];\n    }\n    out = sum/4.0;\n\n    // Insert the input in the delay line.\n    delay_line[index] = x;\n\n    // Update the index for the next input.\n    index++;\n    if (index >= 3) {\n      index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an array of numbers with the initial value of the array being a three-element array filled with zeros.

    \n

    States whose type are objects can similarly be initialized. Declarations can take an object literal as the initial value:

    \n
    state myLiteral:{= {foo: number, bar: string} =} = {= {foo: 42, bar: "baz"} =};\n
    \n

    or use new:

    \n
    state mySet:{=Set<number>=} = {= new Set<number>() =};\n
    \n

    Reactor parameters are also referenced in the TypeScript code as local variables. The example below modifies the Count reactor so that its stride is a parameter:

    \n
    target TypeScript\nreactor Count(stride:number = 1) {\n  state count:number = 0;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    y = count;\n    count += stride;\n  =}\n}\nreactor Display {\n  input x:number;\n  reaction(x) {=\n    console.log("Received: " + x + ".");\n  =}\n}\nmain reactor Stride {\n  c = new Count(stride = 2);\n  d = new Display();\n  c.y -> d.x;\n}\n
    \n

    The second line defines the stride parameter, gives its type, and gives its initial value. As with state variables, the type and initial value can be enclosed in {= ... =} if necessary. The parameter is referenced in the reaction by referring to the local variable stride.

    \n

    When the reactor is instantiated, the default parameter value can be overridden. This is done in the above example near the bottom with the line:

    \n
      c = new Count(stride = 2);\n
    \n

    If there is more than one parameter, use a comma separated list of assignments.

    \n

    Parameters in Lingua Franca are immutable. To encourage correct usage, parameter variables within a reaction are local const variables. If you feel tempted to use a mutable parameter, instead try using the parameter to initialize state and modify the state variable instead. This is illustrated below by a further modification to the Stride example where it takes an initial “start” value for count as a second parameter:

    \n
    target TypeScript\nreactor Count(stride:number = 1, start:number = 5) {\n  state count:number = start;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    y = count;\n    count += stride;\n  =}\n}\nreactor Display {\n  input x:number;\n  reaction(x) {=\n    console.log("Received: " + x + ".");\n  =}\n}\nmain reactor Stride {\n  c = new Count(stride = 2, start = 10);\n  d = new Display();\n  c.y -> d.x;\n}\n
    \n

    Array or Object Parameters

    \n

    Parameters can have array or object values. Here is an example that outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence:{=Array<number>=} = {= [0, 1, 2] =}) {\n  output out:number;\n  state count:number(0);\n  logical action next;\n  reaction(startup, next) -> out, next {=\n    out = sequence[count];\n    count++;\n    if (count < sequence.length) {\n      actions.next.schedule(0, null);\n    }\n  =}\n}\n
    \n

    Above, the parameter default value is an array with three elements, [0, 1, 2]. The syntax for giving this default value is a TypeScript array literal. Since this is TypeScript syntax, not Lingua Franca syntax, the initial value needs to be surrounded with the target code delimiters, {= ... =}. The default value can be overridden when instantiating the reactor using a similar syntax:

    \n
      s = new Source(sequence = {= [1, 2, 3, 4] =});\n
    \n

    Both default and overridden values for parameters can also be created with the new keyword:

    \n
    reactor Source(sequence:{=Array<number>=} = {= new Array<number>() =}) {\n
    \n

    and

    \n
    s = new Source(sequence = {= new Array<number() =});\n
    \n
    \n
    \n

    Parameters and state variables in Rust are accessed on the self structure, as shown in Parameter Declaration.

    \n
    \n

    Inputs and Outputs

    \n
    \n

    In the body of a reaction in the C target, the value of an input is obtained using the syntax name->value, where name is the name of the input port. See, for example, the Destination reactor in Input and Output Declarations.

    \n

    To set the value of outputs, use lf_set. See, for example, the Double reactor in Input and Output Declarations.)

    \n

    An output may even be set in different reactions of the same reactor at the same tag. In this case, one reaction may wish to test whether the previously invoked reaction has set the output. It can check name->is_present to determine whether the output has been set. For example, the Source reactor in the test case TestForPreviousOutput will always produce the output 42:

    \n
    reactor Source {\n  output out: int\n  reaction(startup) -> out {=\n    // Set a seed for random number generation based on the current time.\n    srand(time(0));\n    // Randomly produce an output or not.\n    if (rand() % 2) {\n      lf_set(out, 21);\n    }\n  =}\n  reaction(startup) -> out {=\n    if (out->is_present) {\n      lf_set(out, 2 * out->value);\n    } else {\n      lf_set(out, 42);\n    }\n  =}\n}\n
    \n

    The first reaction may or may not set the output to 21. The second reaction doubles the output if it has been previously produced and otherwise produces 42.

    \n

    Sending and Receiving Data

    \n

    You can define your own data types in C and send and receive those. Consider the StructAsType example:

    \n
    preamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nreactor StructAsType {\n  output out:hello_t;\n  reaction(startup) -> out {=\n    struct hello_t temp = {"Earth", 42};\n    lf_set(out, temp);\n  =}\n}\n
    \n

    The $preamble$ code defines a struct data type. In the reaction to $startup$, the reactor creates an instance of this struct on the stack (as a local variable named temp) and then copies that struct to the output using the lf_set macro.

    \n

    For large structs, it may be inefficient to create a struct on the stack and copy it to the output, as done above. You can use a pointer type instead. See below for details.

    \n

    A reactor receiving the struct message uses the struct as normal in C:

    \n
    reactor Print() {\n  input in:hello_t;\n  reaction(in) {=\n    printf("Received: name = %s, value = %d\\n", in->value.name, in->value.value);\n  =}\n}\n
    \n

    The preamble should not be repeated in this reactor definition if the two reactors are defined together because this will trigger an error when the compiler thinks that hello_t is being redefined.

    \n

    Persistent Inputs

    \n

    In the C target, inputs are persistent. You can read an input even when there is no event present and the value of that input will be the most recently received value or an instance of the input type filled with zeros. For example:

    \n
    target C\nreactor Source {\n  output out: int\n  timer t(100 ms, 200 ms)\n  state count: int = 1\n  reaction(t) -> out {=\n    lf_set(out, self->count++);\n  =}\n}\nreactor Sink {\n  input in: int\n  timer t(0, 100 ms)\n  reaction(t) in {=\n    printf("Value of the input is %d at time %lld\\n", in->value, lf_time_logical_elapsed());\n  =}\n}\nmain reactor {\n  source = new Source()\n  sink = new Sink()\n  source.out -> sink.in\n}\n
    \n

    The Source reactor produces output 1 at 100ms and 2 at 300ms.\nThe Sink reactor reads every 100ms starting at 0.\nNotice that it uses the input in but is not triggered by it.\nThe result of running this program is:

    \n
    Value of the input is 0 at time 0\nValue of the input is 1 at time 100000000\nValue of the input is 1 at time 200000000\nValue of the input is 2 at time 300000000\nValue of the input is 2 at time 400000000\n...
    \n

    The first output is 0 (an int initialized with zero), and subsequently, each output is read twice.

    \n

    Fixed Length Array Inputs and Outputs

    \n

    When inputs and outputs are fixed-length arrays, the memory to contain the array is automatically provided as part of the reactor instance. You can write directly to it, and then just call lf_set_present to alert the system that the output is present. For example:

    \n
    reactor Source {\n  output out: int[3]\n  reaction(startup) -> out {=\n    out->value[0] = 0;\n    out->value[1] = 1;\n    out->value[2] = 2;\n    lf_set_present(out);\n  =}\n}\n
    \n

    In general, this will work for any data type that can be copied by a simple assignment operator (see below for how to handle more complex data types).

    \n

    Reading the array is equally simple:

    \n
    reactor Print(scale: int(1)) {  // The scale parameter is just for testing.\n  input in: int[3]\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < 3; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value[i]);\n    }\n   printf("]\\n");\n  =}\n}\n
    \n

    Variable Length Array Inputs and Outputs

    \n

    Above, the array size is fixed and must be known throughout the program. A more flexible mechanism leaves the array size unspecified in the types of the inputs and outputs and uses lf_set_array instead of lf_set to inform the system of the array length. For example,

    \n
    reactor Source {\n  output out: int[]\n  reaction(startup) -> out {=\n    // Dynamically allocate an output array of length 3.\n    int* array = (int*)malloc(3 * sizeof(int));\n    // Populate the array.\n    array[0] = 0;\n    array[1] = 1;\n    array[2] = 2;\n    // Set the output, specifying the array length.\n    lf_set_array(out, array, 3);\n  =}\n}\n
    \n

    The array length will be available at the receiving end, which may look like this:

    \n
    reactor Print {\n  input in: int[]\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < in->length; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value[i]);\n    }\n    printf("]\\n");\n  =}\n}\n
    \n

    Dynamically Allocated Data

    \n

    A much more flexible way to communicate complex data types is to set dynamically allocated memory on an output port. This can be done in a way that automatically handles freeing the memory when all users of the data are done with it. The reactor that allocates the memory cannot know when downstream reactors are done with the data, so Lingua Franca provides utilities for managing this using reference counting. You can specify a destructor on a port and pass a pointer to a dynamically allocated object as illustrated in the SetDestructor example.

    \n

    Suppose the data structure of interest, its constructor, destructor, and copy_constructor are defined as follows:

    \n
    preamble {=\n  typedef struct int_array_t {\n    int* data;\n    size_t length;\n  } int_array_t;\n\n  int_array_t* int_array_constructor(size_t length) {\n    int_array_t* result = (int_array_t*) malloc(sizeof(int_array_t));\n    result->data = (int*) calloc(length, sizeof(int));\n    result->length = length;\n    return result;\n  }\n\n  void int_array_destructor(void* array) {\n    free(((int_array_t*) array)->data);\n    free(array);\n  }\n\n  void* int_array_copy_constructor(void* array) {\n    int_array_t* source = (int_array_t*) array;\n    int_array_t* copy = (int_array_t*) malloc(sizeof(int_array_t));\n    copy->data = (int*) calloc(source->length, sizeof(int));\n    copy->length = source->length;\n    for (size_t i = 0; i < source->length; i++) {\n      copy->data[i] = source->data[i];\n    }\n    return (void*) copy;\n  }\n=}\n
    \n

    Then, the sender reactor would use lf_set_destructor to specify how the memory set on an output port should be freed:

    \n
    reactor Source {\n  output out:int_array_t*;\n  reaction(startup) -> out {=\n    lf_set_destructor(out, int_array_destructor);\n    lf_set_copy_constructor(out, int_array_copy_constructor);\n  }\n  reaction(startup) -> out {=\n    int_array_t* array =  int_array_constructor(2);\n    for (size_t i = 0; i < array->length; i++) {\n      array->data[i] = i;\n    }\n    lf_set(out, array);\n  =}\n}\n
    \n

    The first reaction specifies the destructor and copy constructor (the latter of which will be used if any downstream reactor has a mutable input or wishes to make a writable copy).

    \n

    IMPORTANT: The array constructed should be sent to only one output port using lf_set. If you need to send it to more than one output port or to use it as the payload of an action, you should use lf_set_token.

    \n

    FIXME: Show how to do this.

    \n

    A reactor receiving this array is straightforward. It just references the array elements as usual in C, as illustrated by this example:

    \n
    reactor Print() {\n  input in:int_array_t*;\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < in->value->length; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value->data[i]);\n    }\n    printf("]\\n");\n  =}\n}\n
    \n

    The deallocation of memory for the data will occur automatically after the last reactor that receives a pointer to the data has finished using it, using the destructor specified by lf_set_destructor or free if none specified.

    \n

    Occasionally, you will want an input or output type to be a pointer, but you don’t want the automatic memory allocation and deallocation. A simple example is a string type, which in C is char*. Consider the following (erroneous) reactor:

    \n
    reactor Erroneous {\n  output out:char*;\n  reaction(startup) -> out {=\n    lf_set(out, "Hello World");\n  =}\n}\n
    \n

    An output data type that ends with * signals to Lingua Franca that the message\nis dynamically allocated and must be freed downstream after all recipients are\ndone with it. But the \"Hello World\" string here is statically allocated, so an\nerror will occur when the last downstream reactor to use this message attempts\nto free the allocated memory. To avoid this for strings, you can use a special\nstring type as follows:

    \n
    reactor Fixed {\n  output out:string;\n  reaction(startup) -> out {=\n    lf_set(out, "Hello World");\n  =}\n}\n
    \n

    The string type is equivalent to char*, but since it doesn’t end with *, it does not signal to Lingua Franca that the type is dynamically allocated. Lingua Franca only handles allocation and deallocation for types that are specified literally with a final * in the type name. The same trick can be used for any type where you don’t want automatic allocation and deallocation. E.g., the SendsPointer example looks like this:

    \n
    reactor SendsPointer  {\n  preamble {=\n    typedef int* int_pointer;\n  =}\n  output out:int_pointer\n  reaction(startup) -> out {=\n    static int my_constant = 42;\n    lf_set(out, &my_constant;)\n  =}\n}\n
    \n

    The above technique can be used to abuse the reactor model of computation by communicating pointers to shared variables. This is generally a bad idea unless those shared variables are immutable. The result will likely be nondeterministic. Also, communicating pointers across machines that do not share memory will not work at all.

    \n

    Mutable Inputs

    \n

    Although it cannot be enforced in C, a receiving reactor should not modify the values provided by an input. Inputs are logically immutable because there may be several recipients. Any recipient that wishes to modify the input should make a copy of it. Fortunately, a utility is provided for this pattern. Consider the ArrayScale example, here modified to use the above int_array_t data type:

    \n
    reactor ArrayScale(scale:int(2)) {\n  mutable input in:int_array_t*;\n  output out:int_array_t*;\n  reaction(in) -> out {=\n    for(int i = 0; i < in->length; i++) {\n      in->value[i] *= self->scale;\n    }\n    lf_set_token(out, in->token);\n  =}\n}\n
    \n

    Here, the input is declared $mutable$, which means that any reaction is free to\nmodify the input. If this reactor is the only recipient of the array or the last\nrecipient of the array, then this will not make a copy of the array but rather use\nthe original array. Otherwise, it will use a copy. By default, memcpy is used to copy the data. However, the sender can also specify\na copy constructor to be used by calling lf_set_copy_constructor on the\noutput port, as explained below.

    \n

    Important: Notice that the above ArrayScale reactor modifies the array and then forwards it to its output port using the lf_set_token() macro. That macro further delegates to downstream reactors the responsibility for freeing dynamically allocated memory once all readers have completed their work. It will not work to just use lf_set, passing it the value.\nThis will result in a memory error, yielding a message like the following:

    \n
        malloc: *** error for object 0x600002674070: pointer being freed was not allocated
    \n

    If the above code were not to forward the array, then the dynamically allocated memory will be automatically freed when this reactor is done with it.

    \n

    Three of the above reactors can be combined into a pipeline as follows:

    \n
    main reactor ArrayScaleTest {\n  s = new Source();\n  c = new ArrayScale();\n  p = new Print();\n  s.out -> c.in;\n  c.out -> p.in;\n}\n
    \n

    In this composite, the array is allocated by ArrayPrint, modified by ArrayScale, and deallocated (freed) after Print has reacted. No copy is necessary because ArrayScale is the only recipient of the original array.

    \n

    Inputs and outputs can also be dynamically allocated structs. In fact, Lingua Franca’s C target will treat any input or output data type that ends with [] or * specially by providing utilities for allocating memory and modifying and forwarding. Deallocation of the allocated memory is automatic. The complete set of utilities is given below.

    \n

    String Types

    \n

    String types in C are char*. But, as explained above, types ending with * are interpreted specially to provide automatic memory management, which we generally don’t want with strings (a string that is a compile-time constant must not be freed). You could enclose the type as {= char* =}, but to avoid this awkwardness, the header files include a typedef that permits using string instead of char*. For example (from DelayString.lf):

    \n
    reactor DelayString(delay:time = 100 ms)) {\n  input in:string;\n  output out:string;\n  logical action a:string;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // The following copies the char*, not the string.\n    lf_schedule_copy(a, self->delay, &(in->value), 1);\n  =}\n}\n
    \n

    Macros For Setting Output Values

    \n

    In all of the following, <out> is the name of the output and <value> is the value to be sent.

    \n
    \n

    lf_set(<out>, <value>);

    \n
    \n

    Set the specified output (or input of a contained reactor) to the specified\nvalue using shallow copy. lf_set can be used with all supported data types\n(including type declarations that end with * or []).

    \n
    \n

    lf_set_token(<out>, <token>);

    \n
    \n

    This version is used to directly set the underlying reference-counted token in\noutputs with a type declaration ending with * (any pointer) or [] (any\narray). The <value> argument should be a struct of type token_t. It should\nbe rarely necessary to have the need to create your own (dynamically allocated)\ninstance of token_t.

    \n

    Consider the\nSetToken.lf\nexample:

    \n
    reactor Source {\n  output out:int*\n  logical action a:int\n  reaction(startup) -> a {=\n    lf_schedule_int(a, MSEC(200), 42);\n  =}\n  reaction(a) -> out {=\n    lf_set_token(out, a->token);\n  =}\n}\n
    \n

    Here, the first reaction schedules an integer-valued action to trigger after 200 milliseconds. As explained below, action payloads are carried by tokens. The second reaction grabs the token rather than the value using the syntax a->token (the name of the action followed by ->token). It then forwards the token to the output. The output data type is int* not int because the token carries a pointer to dynamically allocated memory that contains the value. All inputs and outputs with types ending in * or [] are carried by tokens.

    \n
    \n

    lf_set_destructor(<out>, <destructor>);

    \n
    \n

    Specify the destructor destructor used to deallocate any dynamic data set on the output port out.

    \n
    \n

    lf_set_copy_constructor(<out>, <copy_constructor>);

    \n
    \n

    Specify the copy_constructor used to copy construct any dynamic data set on the output port out if the receiving port is $mutable$.

    \n

    lf_set (and lf_set_token) will overwrite any output value previously set at the same logical time and will cause the final output value to be sent to all reactors connected to the output. They also set a local <out>->is_present variable to true. This can be used to subsequently test whether the output value has been set.

    \n
    \n
    \n

    In the body of a reaction in the C++ target, the value of an input is obtained using the syntax *name.get(), where name is the name of the input port. Similarly, outputs are set using a set() method on an output port. For examples, see Inputs and Outputs.

    \n

    Note that get() always returns a pointer to the actual value. Thus the pointer needs to be dereferenced with * to obtain the value. (See Sending and Receiving Large Data Types for an explanation of the exact mechanisms behind this pointer access).\nTo determine whether an input is present, name.is_present() can be used. Since get() returns a nullptr if no value is present, name.get() != nullptr can be used alternatively for checking presence.

    \n

    Sending and Receiving Large Data Types

    \n

    You can define your own data types in C++ or use types defined in a library and send and receive those. Consider the StructAsType example:

    \n
    public preamble {=\n  struct Hello {\n    std::string name;\n    int value;\n  };\n=}\nreactor StructAsType {\n  output out: Hello;\n  reaction(startup) -> out {=\n    Hello hello{"Earth, 42};\n    out.set(hello);\n  =}\n}\n
    \n

    The $public$ $preamble$ code defines a struct data type. In the reaction to $startup$, the reactor creates an instance of this struct on the stack (as a local variable named hello) and then copies that instance to the output using the set() method. For this reason, the C++ reactor runtime provides more sophisticated ways to allocate objects and send them via ports.

    \n

    The C++ library defines two types of smart pointers that the runtime uses internally to implement the exchange of data between ports. These are reactor::MutableValuePtr<T> and reactor::ImmutableValuePtr<T>. reactor::MutableValuePtr<T> is a wrapper around std::unique_ptr and provides read and write access to the value hold, while ensuring that the value has a unique owner. In contrast, reactor::ImmutableValuePtr<T> is a wrapper around std::shared_pointer and provides read only (const) access to the value it holds. This allows data to be shared between reactions of various reactors, while guarantee data consistency. Similar to std::make_unique and std::make_shared, the reactor library provides convenient function for creating mutable and immutable values pointers: reactor::make_mutable_value<T>(...) and reactor::make_immutable_value<T>(...).

    \n

    In fact this code from the example above:

    \n
        Hello hello{"Earth, 42"};\n    out.set(hello);\n
    \n

    implicitly invokes reactor::make_immutable_value<Hello>(hello) and could be rewritten as

    \n
        Hello hello{"Earth, 42"};\n    out.set(reactor::make_immutable_value<Hello>(hello));\n
    \n

    This will invoke the copy constructor of Hello, copying its content from the hello instance to the newly created reactor::ImmutableValuePtr<Hello>.

    \n

    Since copying large objects is inefficient, the move semantics of C++ can be used to move the ownership of object instead of copying it. This can be done in the following two ways. First, by directly creating a mutable or immutable value pointer, where a mutable pointer allows modification of the object after it has been created:

    \n
        auto hello = reactor::make_mutable_value<Hello>("Earth", 42);\n    hello->name = "Mars";\n    out.set(std::move(hello));\n
    \n

    An example of this can be found in StructPrint.lf. Not that after the call to std::move, hello is nullptr and the reaction cannot modify the object anymore. Alternatively, if no modification is requires, the object can be instantiated directly in the call to set() as follows:

    \n
        out.set({"Earth", 42});\n
    \n

    An example of this can be found in StructAsTypeDirect.

    \n

    Getting a value from an input port of type T via get() always returns an reactor::ImmutableValuePtr<T>. This ensures that the value cannot be modified by multiple reactors receiving the same value, as this could lead to an inconsistent state and nondeterminism in a multi-threaded execution. An immutable value pointer can be converted to a mutable pointer by calling get_mutable_copy. For instance, the ArrayScale reactor modifies elements of the array it receives before sending it to the next reactor:

    \n
    reactor Scale(scale:int = 2) {\n  input in:int[3];\n  output out:int[3];\n\n  reaction(in) -> out {=\n    auto array = in.get().get_mutable_copy();\n    for(int i = 0; i < array->size(); i++) {\n      (*array)[i] = (*array)[i] * scale;\n    }\n    out.set(std::move(array));\n  =}\n}\n
    \n

    Currently get_mutable_copy() always copies the contained value to safely create a mutable pointer. However, a future implementation could optimize this by checking if any other reaction is accessing the same value. If not, the value can simply be moved from the immutable pointer to a mutable one.

    \n
    \n
    \n

    In the body of a reaction in the Python target, the value of an input is\nobtained using the syntax name.value, where name is the name of the input\nport. To determine whether an input is present, use name.is_present. To\nproduce an output, use the syntax name.set(value). The value can be any\nvalid Python object. For simple examples, see Inputs and\nOutputs.

    \n

    Sending and Receiving Objects

    \n

    You can define your own data types in Python and send and receive those. Consider the StructAsType example:

    \n
    target Python {\n  files: include/hello.py\n}\npreamble {=\n  import hello\n=}\nreactor Source {\n  output out;\n  reaction(startup) -> out {=\n    temp = hello.hello("Earth", 42)\n    out.set(temp)\n  =}\n}\n
    \n

    The top-level preamble has imported the hello module, which contains the following class:

    \n
    class hello:\n    def __init__(self, name = "", value = 0):\n        self.name = name\n        self.value = value\n
    \n

    In the reaction to startup, the reactor has created an instance object of this class (as local variable named temp) and passed it downstream using the set method on output port out.

    \n

    The set method is defined as follows:

    \n
    \n

    <port>.set(<value>): Set the specified output port (or input of a contained\nreactor) to the specified value. This value can be any Python object\n(including None and objects of type Any). The value is\ncopied and therefore the variable carrying the value can be subsequently\nmodified without changing the output.

    \n
    \n

    A reactor receiving the class object message can subsequently access the object\nusing <port>.value:

    \n
    reactor Print(expected(42)) {\n  input _in;\n  reaction(_in) {=\n    print("Received: name = {:s}, value = {:d}\\n".format(_in.value.name,\n                                                         _in.value.value))\n  =}\n}\n
    \n

    Note: The hello module has been imported using a top-level preamble, therefore, the contents of the module are available to all reactors defined in the current Lingua Franca file (a similar situation arises if the hello class itself was in the top-level preamble).

    \n
    \n
    \n

    In the TypeScript target, all TypeScript types are generally acceptable for inputs and outputs with one notable exception:

    \n
      \n
    • undefined is not a valid type for an input, output, or action. This is because undefined is used to designate the absence of an input, output, or action during a reaction.
    • \n
    \n

    As with parameters and state variables, custom types (and classes) must be defined in the preamble before they may be used.

    \n

    To benefit from type checking, you should declare types for your reactor elements. If a type isn’t declared for an input, output, or action, it is assigned the reactor-ts type Present which is defined as

    \n
    export type Present = (number | string | boolean | symbol | object | null);\n
    \n

    In the body of a reaction in the TypeScript target, inputs are simply referred to by name. An input of type t is available within the body of a reaction as a local variable of type t | undefined. To determine whether an input is present, test the value of the input against undefined. An undefined input is not present.

    \n

    WARNING Be sure to use the === or !== operator and not == or != to test against undefined. In JavaScript/TypeScript the comparison undefined == null yields the value true. It may also be tempting to rely upon the falsy evaluation of undefined within an if statement, but this may introduce bugs. For example a reaction that tests the presence of input x with if (x) { ... } will not correctly identify potentially valid present values such as 0, false, or \"\".

    \n

    For example, the Determinism.lf test case includes the following reactor:

    \n
    reactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    let sum = 0;\n    if (x !== undefined) {\n      sum += x;\n    }\n    if (y !== undefined) {\n      sum += y;\n    }\n    console.log("Received " + sum);\n    if (sum != 2) {\n      util.requestErrorStop("FAILURE: Expected 2.")\n    }\n  =}\n}\n
    \n

    The reaction refers to the inputs x and y by name and tests for their presence by testing x and y against undefined. If a reaction is triggered by just one input, then normally it is not necessary to test for its presence. It will always be present. However TypeScript’s type system is not smart enough to know such an input will never have type undefined if there’s no test against undefined within the reaction. An explicit type annotation (for example x = x as t; where t is the type of the input) may be necessary to avoid type errors from the compiler. In the above example, there are two triggers, so the reaction has no assurance that both will be present.

    \n

    Inputs declared in the uses part of the reaction do not trigger the reaction. Consider this modification of the above reaction:

    \n
    reaction(x) y {=\n  let sum = x as number;\n  if (y !== undefined) {\n    sum += y;\n  }\n  console.log("Received " + sum + ".");\n=}\n
    \n

    It is no longer necessary to test for the presence of x because that is the only trigger. The input y, however, may or may not be present at the logical time that this reaction is triggered. Hence, the code must test for its presence.

    \n

    The effects portion of the reaction specification can include outputs and actions. Actions will be described below. Like inputs, an output of type t is available within the body of a reaction as a local variable of type t | undefined. The local variable for each output is initialized to the output’s current value. Outputs are set by assigning a (non-undefined) value to its local variable (no changes will be made to an output if it has the value undefined at the end of a reaction). Whatever value an output’s local variable has at the end of the reaction will be set to that output. If an output’s local variable has the value undefined at the end of the reaction, that output will not be set and connected downstream inputs will be absent. For example, we can further modify the above example as follows:

    \n
    output z:number;\nreaction(x) y -> z {=\n  let sum = x as number;\n  if (y !== undefined) {\n    sum += y;\n  }\n  z = sum;\n=}\n
    \n

    If an output gets set more than once at any logical time, downstream reactors will see only the final value that is set. Since the order in which reactions of a reactor are invoked at a logical time is deterministic, and whether inputs are present depends only on their timestamps, the final value set for an output will also be deterministic.

    \n

    An output may even be set in different reactions of the same reactor at the same logical time. In this case, one reaction may wish to test whether the previously invoked reaction has set the output. It can do that using a !== undefined test for that output. For example, the following reactor will always produce the output 42:

    \n
    reactor TestForPreviousOutput {\n  output out:number;\n  reaction(startup) -> out {=\n    if (Math.random() > 0.5) {\n      out = 21;\n    }\n  =}\n  reaction(startup) -> out {=\n    let previous_output = out;\n    if (previous_output) {\n      out = 2 * previous_output;\n    } else {\n      out = 42;\n    }\n  =}\n}\n
    \n

    The first reaction may or may not set the output to 21. The second reaction doubles the output if it has been previously produced and otherwise produces 42.

    \n

    Sending and Receiving Custom Types

    \n

    You can define your own data types in TypeScript and send and receive those. Consider the following example:

    \n
    reactor CustomType {\n  preamble {=\n    type custom = string | null;\n  =}\n  output out:custom;\n  reaction(startup) -> out {=\n    out = null;\n  =}\n}\n
    \n

    The $preamble$ code defines a custom union type of string and null.

    \n
    \n
    \n

    Inputs and outputs in the Rust target are accessed using the set and get methods of the ctx objects, as shown in Inputs and Outputs.

    \n
    \n

    Time

    \n
    \n

    In the C target, the value of a time instant or interval is an integer specifying a number of nanoseconds. An instant is the number of nanoseconds that have elapsed since January 1, 1970. An interval is the difference between two instants. When an LF program starts executing, logical time is (normally) set to the instant provided by the operating system. (On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    Time in the C target is a int64_t, which is a 64-bit signed number. Since a 64-bit number has a limited range, this measure of time instants will overflow in approximately the year 2262. For better code clarity, two types are defined in tag.h, instant_t and interval_t, which you can use for time instants and intervals respectively. These are both equivalent to int64_t, but using those types will insulate your code against changes and platform-specific customizations.

    \n

    Lingua Franca uses a superdense model of time. A reaction is invoked at a logical tag, a struct consisting of a time value (an instant_t, which is a int64_t) and a microstep value (a microstep_t, which is an uint32_t). The tag is guaranteed to not increase during the execution of a reaction. Outputs produced by a reaction have the same tag as the inputs, actions, or timers that trigger the reaction, and hence are logically simultaneous.

    \n

    The time structs and functions for working with time are defined in tag.h. The most useful functions are:

    \n
      \n
    • tag_t lf_tag(): Get the current tag at which this reaction has been invoked.
    • \n
    • int lf_tag_compare(tag_t, tag_t): Compare two tags, returning -1, 0, or 1 for less than, equal, and greater than.
    • \n
    • instant_t lf_time_logical(): Get the current logical time (the first part of the current tag).
    • \n
    • interval_t lf_time_logical_elapsed(): Get the logical time elapsed since program start.
    • \n
    \n

    There are also some useful functions for accessing physical time:

    \n
      \n
    • instant_t lf_time_physical(): Get the current physical time.
    • \n
    • instant_t lf_time_physical_elapsed(): Get the physical time elapsed since program start.
    • \n
    • instant_t lf_time_start(): Get the starting physical and logical time.
    • \n
    \n

    The last of these is both a physical and logical time because, at the start of execution, the starting logical time is set equal to the current physical time as measured by a local clock.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime example:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t logical = lf_time_logical();\n    printf("Logical time is %ld.\\n", logical);\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    Start execution at time Sun Oct 13 10:18:36 2019\nplus 353609000 nanoseconds.\nLogical time is 1570987116353609000.\nLogical time is 1570987117353609000.\nLogical time is 1570987118353609000.\n...
    \n

    The first two lines give the current time-of-day provided by the execution platform at the start of execution. This is used to initialize logical time. Subsequent values of logical time are printed out in their raw form, rather than the friendlier form in the first two lines. If you look closely, you will see that each number is one second larger than the previous number, where one second is 1000000000 nanoseconds.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    interval_t elapsed = lf_time_logical_elapsed();\n    printf("Elapsed logical time is %ld.\\n", elapsed);\n  =}\n}\n
    \n

    This will produce:

    \n
    Start execution at time Sun Oct 13 10:25:22 2019\nplus 833273000 nanoseconds.\nElapsed logical time is 0.\nElapsed logical time is 1000000000.\nElapsed logical time is 2000000000.\n...
    \n

    You can also get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t physical = lf_time_physical();\n    printf("Physical time is %ld.\\n", physical);\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Start execution at time Sun Oct 13 10:35:59 2019\nplus 984992000 nanoseconds.\nPhysical time is 1570988159986108000.\nPhysical time is 1570988160990219000.\nPhysical time is 1570988161990067000.\n...
    \n

    Finally, you can get elapsed physical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t elapsed_physical = lf_time_physical_elapsed();\n    printf("Elapsed physical time is %ld.\\n", elapsed_physical);\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Elapsed physical time is 657000.\nElapsed physical time is 1001856000.\nElapsed physical time is 2004761000.\n...
    \n

    Notice that these numbers are increasing by roughly one second each time. If you set the fast target parameter to true, then logical time will elapse much faster than physical time.

    \n

    Working with nanoseconds in C code can be tedious if you are interested in longer durations. For convenience, a set of macros are available to the C programmer to convert time units into the required nanoseconds. For example, you can specify 200 msec in C code as MSEC(200) or two weeks as WEEKS(2). The provided macros are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these. Examples are given in the next section.

    \n
    \n
    \n

    Timers are specified exactly as in the Time and Timers. When working with time in the C++ code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    The reactor-cpp library uses std::chrono for representing time. Specifically, the library defines two types for representing durations and timepoints: reactor::Duration and reactor::TimePoint. reactor::Duration is an alias for std::chrono::nanosecods. reactor::TimePoint is alias for std::chrono::time_point<std::chrono::system_clock, std::chrono::nanoseconds>. As you can see from these definitions, the smallest time step that can be represented is one nanosecond. Note that reactor::TimePoint describes a specific point in time and is associated with a specific clock, whereas reactor::Duration defines a time interval between two time points.

    \n

    Lingua Franca uses a superdense model of logical time. A reaction is invoked at a logical tag. In the C++ library, a tag is represented by the class reactor::Tag. In essence, this class is a tuple of a reactor::TimePoint representing a specific point in logical time and a microstep value (of type reactor::mstep_t, which is an alias for unsigned long). reactor::Tag provides two methods for getting the time point or the microstep:

    \n
    const TimePoint& time_point() const;\nconst mstep_t& micro_step() const;\n
    \n

    The C++ code in reaction bodies has access to library functions that allow to retrieve the current logical or physical time:

    \n
      \n
    • TimePoint get_physical_time(): Get the current physical time.
    • \n
    • TimePoint get_logcial_time(): Get the current logical time.
    • \n
    • Duration get_elapsed_physical_time(): Get the physical time elapsed since program start.
    • \n
    • Duration get_elapsed_logical_time(): Get the logical time elapsed since program start.
    • \n
    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime example:

    \n
    main reactor {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    auto logical = get_logical_time();\n    std::cout << "Logical time is " << logical << std::endl;\n  =}\n}\n
    \n

    Note that the << operator is overloaded for both reactor::TimePoint and reactor::Duration and will print the time information accordingly.

    \n

    When executing the above program, you will see something like this:

    \n
    [INFO]  Starting the execution\nLogical time is 2021-05-19 14:06:09.496828396\nLogical time is 2021-05-19 14:06:10.496828396\nLogical time is 2021-05-19 14:06:11.496828396\nLogical time is 2021-05-19 14:06:11.496828396\n...
    \n

    If you look closely, you will see that each printed logical time is one second larger than the previous one.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    auto elapsed = get_elapsed_logical_time();\n    std::cout << "Elapsed logical time is " << elapsed << std::endl;\n    std::cout << "In seconds: " <<  std::chrono::duration_cast<std::chrono::seconds>(elapsed) << std::endl;\n  =}\n}\n
    \n

    Using std::chrono it is also possible to convert between time units and directly print the number of elapsed seconds as seen above. The resulting output of this program will be:

    \n
    [INFO]  Starting the execution\nElapsed logical time is 0 nsecs\nIn seconds: 0 secs\nElapsed logical time is 1000000000 nsecs\nIn seconds: 1 secs\nElapsed logical time is 2000000000 nsecs\nIn seconds: 2 secs\n...
    \n

    You can also get physical and elapsed physical time:

    \n
    main reactor {\n  timer t(0, 1 sec);\n\treaction(t) {=\n    auto logical = get_logical_time();\n    auto physical = get_physical_time();\n    auto elapsed = get_elapsed_physical_time();\n    std::cout << "Physical time is " << physical << std::endl;\n    std::cout << "Elapsed physical time is " << elapsed << std::endl;\n    std::cout << "Time lag is " << physical - logical << std::endl;\n  =}\n}\n
    \n

    Notice that the physical times are increasing by roughly one second in each reaction. The output also shows the lag between physical and logical time. If you set the fast target parameter to true, then physical time will elapse much faster than logical time. The above program will produce something like this:

    \n
    [INFO]  Starting the execution\nPhysical time is 2021-05-19 14:25:18.070523014\nElapsed physical time is 2601601 nsecs\nTime lag is 2598229 nsecs\nPhysical time is 2021-05-19 14:25:19.068038275\nElapsed physical time is 1000113888 nsecs\nTime lag is 113490 nsecs\n[INFO]  Physical time is Terminating the execution\n2021-05-19 14:25:20.068153026\nElapsed physical time is 2000228689 nsecs\nTime lag is 228241 nsecs
    \n

    For specifying time durations in code chrono provides convenient literal operators in std::chrono_literals. This namespace is automatically included for all reaction bodies. Thus, we can simply write:

    \n
    std::cout << 42us << std::endl;\nstd::cout << 1ms << std::endl;\nstd::cout << 3s << std::endl;\n
    \n

    which prints:

    \n
    42 usecs\n1 msecs\n3 secs
    \n
    \n
    \n

    Timers are specified exactly as in the Time and Timers. When working with time in the Python code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    In the Python target, similar to the C target, the value of a time instant or\ninterval is an integer specifying a number of nanoseconds. An instant is the\nnumber of nanoseconds that have elapsed since January 1, 1970. An interval is\nthe difference between two instants.

    \n

    The functions for working with time and tags are:

    \n
      \n
    • lf.tag() -> Tag: Returns a Tag instance of the current tag at which this reaction has been invoked.
    • \n
    • lf.tag_compare(Tag, Tag) -> int: Compare two Tag instances, returning -1, 0, or 1 for less than, equal, and greater than. Tags can also be compared using rich comparators (ex. <, >, ==), which returns True or False.
    • \n
    • lf.time.logical() -> int: Get the current logical time (the first part of the current tag).
    • \n
    • lf.time.logical_elapsed() -> int: Get the logical time elapsed since program start.
    • \n
    \n

    Tags can be initialized using Tag(time=some_number, microstep=some_other_number).

    \n

    There are also some useful functions for accessing physical time:

    \n
      \n
    • lf.time.physical() -> int: Get the current physical time.
    • \n
    • lf.time.physical_elapsed() -> int: Get the physical time elapsed since program start.
    • \n
    • lf.time.start() -> int: Get the starting physical and logical time.
    • \n
    \n

    The last of these is both a physical and a logical time because, at the start of execution, the starting logical time is set equal to the current physical time as measured by a local clock.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime.lf example:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    logical = lf.time.logical()\n    print("Logical time is ", logical)\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nLogical time is  1604587862864237900\nLogical time is  1604587863864237900\nLogical time is  1604587864864237900\n...
    \n

    The first two lines give the current time-of-day provided by the execution platform at the start of execution. This is used to initialize logical time. Subsequent values of logical time are printed out in their raw form, rather than the friendlier form in the first two lines. If you look closely, you will see that each number is one second larger than the previous number, where one second is 1000000000 nanoseconds.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    elapsed = lf.time.logical_elapsed()\n    print("Elapsed logical time is ", elapsed)\n  =}\n}\n
    \n

    This will produce:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nElapsed logical time is  0\nElapsed logical time is  1000000000\nElapsed logical time is  2000000000\n...
    \n

    You can also get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    physical = lf.time.physical()\n    print("Physical time is ", physical)\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nPhysical time is  1604587862864343500\nPhysical time is  1604587863864401900\nPhysical time is  1604587864864395200\n...
    \n

    Finally, you can get elapsed physical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    elapsed_physical = lf.time.physical_elapsed()\n    print("Elapsed physical time is ", elapsed_physical)\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nElapsed physical time is  110200\nElapsed physical time is  1000185400\nElapsed physical time is  2000178600\n...
    \n

    Notice that these numbers are increasing by roughly one second each time. If you set the fast target parameter to true, then logical time will elapse much faster than physical time.

    \n

    Working with nanoseconds in the Python code can be tedious if you are interested in longer durations. For convenience, a set of functions are available to the Python programmer to convert time units into the required nanoseconds. For example, you can specify 200 msec in Python code as MSEC(200) or two weeks as WEEKS(2). The provided functions are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these. Examples are given in the next section.

    \n
    \n
    \n

    See Summary of Time Functions and Utility Function Reference for a quick API reference.

    \n

    Timers are specified exactly as in the Time and Timers section. When working with time in the TypeScript code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    A TimeValue is an class defined in the TypeScript target library file time.ts to represent a time instant or interval. For your convenience TimeValue and other classes from the time.ts library mentioned in these instructions are automatically imported into scope of your reactions. An instant is the number of nanoseconds that have elapsed since January 1, 1970. An interval is the difference between two instants. When an LF program starts executing, logical time is (normally) set to the instant provided by the operating system. (On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    Internally a TimeValue uses two numbers to represent the time. To prevent overflow (which would occur for time intervals spanning more than 0.29 years if a single JavaScript number, which has 2^53 bits of precision, were to be used), we use two numbers to store a time value. The first number denotes the number of whole seconds in the interval or instant; the second number denotes the remaining number of nanoseconds in the interval or instant. The first number represents the number of seconds, the second number represents the number of nanoseconds. These fields are not accessible to the programmer, instead TimeValues may be manipulated by an API with functions for addition, subtraction, and comparison.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider:

    \n
    target TypeScript;\nmain reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let logical:TimeValue = util.getCurrentLogicalTime()\n    console.log("Logical time is " + logical + ".");\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    Logical time is (1584666585 secs; 805146880 nsecs).\nLogical time is (1584666586 secs; 805146880 nsecs).\nLogical time is (1584666587 secs; 805146880 nsecs).\n...
    \n

    Subsequent values of logical time are printed out in their raw form, of seconds and nanoseconds. If you look closely, you will see that each number is one second larger than the previous number.

    \n

    You can also obtain the elapsed logical time since the start of execution, rather than exact logical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let logical:TimeValue = util.getElapsedLogicalTime()\n    console.log("Logical time is " + logical + ".");\n  =}\n}\n
    \n

    This will produce:

    \n
    Logical time is (0 secs; 0 nsecs).\nLogical time is (1 secs; 0 nsecs).\nLogical time is (2 secs; 0 nsecs).\n...
    \n

    You can get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let physical:TimeValue = util.getCurrentPhysicalTime()\n    console.log("Physical time is " + physical + ".");\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Physical time is (1584666801 secs; 644171008 nsecs).\nPhysical time is (1584666802 secs; 642269952 nsecs).\nPhysical time is (1584666803 secs; 642278912 nsecs).\n...
    \n

    Notice that these numbers are increasing by roughly one second each time.

    \n

    You can also get elapsed physical time from the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let physical:TimeValue = util.getElapsedPhysicalTime()\n    console.log("Physical time is " + physical + ".");\n  =}\n}\n
    \n

    This will produce something like:

    \n
    Physical time is (0 secs; 2260992 nsecs).\nPhysical time is (1 secs; 166912 nsecs).\nPhysical time is (2 secs; 136960 nsecs).\n...
    \n

    You can create a TimeValue yourself with the UnitBasedTimeValue class. UnitBasedTimeValue is a subclass of TimeValue and can be used wherever you could also use a TimeValue directly obtained from one of the util functions. A UnitBasedTimeValue is constructed with a whole number and a TimeUnit. A TimeUnit is an enum from the time.ts library with convenient labels for common time units. These are nsec, usec, msec, sec (or secs), minute (or minutes), hour (or hours), day (or days), and week (or weeks).

    \n

    This reactor has an example of a UnitBasedTimeValue.

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let myTimeValue:TimeValue = new UnitBasedTimeValue(200, TimeUnit.msec);\n    let logical:TimeValue = util.getCurrentLogicalTime()\n    console.log("My custom time value is " + myTimeValue + ".");\n  =}\n
    \n

    This will produce:

    \n
    My custom time value is 200 msec.\nMy custom time value is 200 msec.\nMy custom time value is 200 msec.\n...
    \n

    Tags

    \n

    The TypeScript target provides a utility to get the current Tag of a reaction. Recall that time in Lingua Franca is superdense and each TimeValue is paired with an integer “microstep” index to track the number of iterations at a particular TimeValue. A Tag is this combination of a TimeValue and a “microstep”. The time.ts library provides functions for adding, subtracting, and comparing Tags.

    \n

    You can get the current Tag in your reactions. This example illustrates tags with a Zero-Delay Action:

    \n
    target TypeScript;\nmain reactor GetTime {\n  timer t(0, 1 sec);\n  logical action a;\n  reaction(t) -> a {=\n    let superdense:Tag = util.getCurrentTag();\n    console.log("First iteration - the tag is: " + superdense + ".");\n    actions.a.schedule(0, null);\n  =}\n  reaction(a) {=\n    let superdense:Tag = util.getCurrentTag();\n    let timePart:TimeValue = superdense.time;\n    let microstepPart:number = superdense.microstep;\n    console.log("Second iteration - the time part of the tag is:  " + timePart + ".");\n    console.log("Second iteration - the microstep part of the tag is:  " + microstepPart + ".");\n  =}\n}\n
    \n

    This will produce:

    \n
    First iteration - the tag is: ((1584669987 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669987 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\nFirst iteration - the tag is: ((1584669988 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669988 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\nFirst iteration - the tag is: ((1584669989 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669989 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\n...
    \n

    The first reaction prints the “First iteration” part of the output at microstep 0. The second reaction occurs one microstep later (explained in Scheduling Delayed Reactions) and illustrates how to split a Tag into its constituent TimeValue and microstep.

    \n

    Summary of Time Functions

    \n

    See Time. These time functions are defined in the time.ts library of reactor-ts.

    \n

    UnitBasedTimeValue(value: number, unit:TimeUnit) Constructor for UnitBasedTimeValue, a programmer-friendly subclass of TimeValue. Use a number and a TimeUnit enum.

    \n
    enum TimeUnit {\n  nsec,\n  usec,\n  msec,\n  sec,\n  secs,\n  minute,\n  minutes,\n  hour,\n  hours,\n  day,\n  days,\n  week,\n  weeks,\n}\n
    \n

    TimeValue.add(other: TimeValue): TimeValue Adds this to other.

    \n

    TimeValue.subtract(other: TimeValue): TimeValue Subtracts other from this. A negative result is an error.

    \n

    TimeValue.difference(other: TimeValue): TimeValue Obtain absolute value of other minus this.

    \n

    TimeValue.isEqualTo(other: TimeValue): boolean Returns true if this and other represents the same TimeValue. Otherwise false.

    \n

    TimeValue.isZero(): boolean Returns true if this represents a 0 TimeValue.

    \n

    TimeValue.isEarlierThan(other: TimeValue): boolean Returns true if this < other. Otherwise false.

    \n

    Tag.isSmallerThan(other: Tag): boolean Returns true if this < other. Otherwise false.

    \n

    Tag.isSimultaneousWith(other: Tag): boolean Returns true if this and other represents the same Tag. Otherwise false.

    \n

    Tag.getLaterTag(delay: TimeValue): Tag Returns a tag with the time part of this TimeValue incremented by delay.

    \n

    Tag.getMicroStepLater(): Tag Returns a tag with the microstep part of this TimeValue incremented by 1.

    \n

    getTimeDifference(other: Tag): TimeValue Returns a TimeValue that represents the absolute (i.e., positive) time difference between this and other.

    \n
    \n
    \n

    FIXME: details needed here on time in Rust.

    \n
    \n

    Actions

    \n
    \n

    Actions are described in Actions. If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. This is particularly useful for physical actions that are externally triggered because it enables the action to convey information to the reactor. This could be, for example, the body of an incoming network message or a numerical reading from a sensor.

    \n

    Recall from Composing Reactors that the $after$ keyword on a connection between ports introduces a logical delay. This is actually implemented using a logical action. We illustrate how this is done using the DelayInt example:

    \n
    reactor Delay(delay: time = 100 ms) {\n  input in: int\n  output out: int\n  logical action a: int\n  reaction(a) -> out {=\n    if (a->has_value && a->is_present) lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // Use specialized form of schedule for integer payloads.\n    lf_schedule_int(a, self->delay, in->value);\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
      delay = new Delay();\n  source.out -> delay.in;\n  delay.in -> sink.out\n
    \n

    is equivalent to

    \n
        source.out -> sink.in after 100 ms\n
    \n

    (except that our Delay reactor will only work with data type int).

    \n

    Note: The reaction to a is given before the reaction to in above. This is important because if both inputs are present at the same tag, the first reaction must be executed before the second. Because of this reaction ordering, it is possible to create a program that has a feedback loop where the output of the Delay reactor propagates back to an input at the same tag. If the reactions were given in the opposite order, then such a program would result in a causality loop.

    \n

    In the Delay reactor, the action a is specified with a type int. The reaction to the input in declares as its effect the action a. This declaration makes it possible for the reaction to schedule a future triggering of a. The reaction uses one of several variants of the lf_schedule function, namely lf_schedule_int, a convenience function provided because integer payloads on actions are very common. We will see below, however, that payloads can have any data type.

    \n

    The first reaction declares that it is triggered by a and has effect out. To\nread the value, it uses the a->value variable. Because this reaction is first,\nthe out at any logical time can be produced before the input in is even\nknown to be present. Hence, this reactor can be used in a feedback loop, where\nout triggers a downstream reactor to send a message back to in of this same\nreactor. If the reactions were given in the opposite order, there would be a\ncausality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it as follows:

    \n
      reaction(a) -> out {=\n    if (a->has_value) {\n      lf_set(out, a->value);\n    }\n  =}\n
    \n

    It is possible to both be triggered by and schedule an action in the same\nreaction. For example, the\nfollowing CountSelf\nreactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay: time = 100 msec) {\n  output out: int\n  logical action a: int\n  reaction(startup) -> a, out {=\n    lf_set(out, 0);\n    lf_schedule_int(a, self->delay, 1);\n  =}\n  reaction(a) -> a, out {=\n    lf_set(out, a->value);\n    lf_schedule_int(a, self->delay, a->value + 1);\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    The C++ provides a simple interface for scheduling actions via a schedule() method. Actions are described in the Actions document. Consider the Schedule reactor:

    \n
    reactor Schedule {\n  input x: int\n  logical action a: void\n  reaction(x) -> a {=\n    a.schedule(200ms);\n  =}\n\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time.count()\n          << " after start" << std::endl;\n  =}\n}\n
    \n

    When this reactor receives an input x, it calls schedule() on the action a, specifying a logical time offset of 200 milliseconds. The action a will be triggered at a logical time 200 milliseconds after the arrival of input x. At that logical time, the second reaction will trigger and will use the get_elapsed_logical_time() function to determine how much logical time has elapsed since the start of execution.

    \n

    Notice that after the logical time offset of 200 msec, there may be another input x simultaneous with the action a. Because the reaction to a is given first, it will execute first. This becomes important when such a reactor is put into a feedback loop (see below).

    \n

    Physical actions work exactly as described in the Physical Actions section.

    \n

    Zero-Delay Actions

    \n

    If the specified delay in a schedule() is omitted or is zero, then the action a will be triggered one microstep later in superdense time (see Superdense Time). Hence, if the input x arrives at metric logical time t, and you call schedule() in one of the following ways:

    \n
    a.schedule();\na.schedule(0s);\na.schedule(reactor::Duration::zero());\n
    \n

    then when the reaction to a is triggered, the input x will be absent (it was present at the previous microstep). The reaction to x and the reaction to a occur at the same metric time t, but separated by one microstep, so these two reactions are not logically simultaneous.

    \n

    As discussed above the he metric time is visible to the programmer and can be obtained in a reaction using either get_elapsed_logical_time() or get_logical_time().

    \n

    As described in the Action document, action declarations can have a min_delay parameter. This modifies the timestamp further. Also, the action declaration may be physical rather than logical, in which case, the assigned timestamp will depend on the physical clock of the executing platform.

    \n

    Actions With Values

    \n

    If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. This is particularly useful for physical actions that are externally triggered because it enables the action to convey information to the reactor. This could be, for example, the body of an incoming network message or a numerical reading from a sensor.

    \n

    Recall from the Composing Reactors section that the after keyword on a connection between ports introduces a logical delay. This is actually implemented using a logical action. We illustrate how this is done using the DelayInt example:

    \n
    reactor Delay(delay: time = 100 ms) {\n  input in: int\n  output out: int\n  logical action d: int\n  reaction(d) -> out {=\n    if (d.is_present()) {\n      out.set(d.get());\n    }\n  =}\n  reaction(in) -> d {=\n    d.schedule(in.get(), delay);\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
    d = new Delay();\nsource.out -> d.in;\nd.in -> sink.out\n
    \n

    is equivalent to

    \n
    source.out -> sink.in after 100 ms\n
    \n

    (except that our Delay reactor will only work with data type int).

    \n

    Note: The reaction to d is given before the reaction to in above. This is important because if both inputs are present at the same tag, the first reaction must be executed before the second. Because of this reaction ordering, it is possible to create a program that has a feedback loop where the output of the Delay reactor propagates back to an input at the same tag. If the reactions were given in the opposite order, then such a program would result in a causality loop.

    \n

    The action d is specified with a type int. The reaction to the input in declares as its effect the action d. This declaration makes it possible for the reaction to schedule a future triggering of d. In the C++ target, actions use the same mechanism for passing data via value pointers as do ports. In the example above, the reactor::ImmutablValuePtr<int> derived by the call to in.get() is passed directly to schedule(). Similarly, the value can later be retrieved from the action with d.get() and passed to the output port.

    \n

    The first reaction declares that it is triggered by d and has effect out. Because this reaction is first, the out at any logical time can be produced before the input in is even known to be present. Hence, this reactor can be used in a feedback loop, where out triggers a downstream reactor to send a message back to in of this same reactor. If the reactions were given in the opposite order, there would be causality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it using is_present():

    \n
    reaction(d) -> out {=\n  if (d.is_present()) {\n    out.set(d.get());\n  }\n=}\n
    \n

    It is possible to both be triggered by and schedule an action the same reaction. For example, this reactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay:time(100 msec)) {\n  output out:int;\n  logical action a:int;\n  reaction(startup) -> a, out {=\n    out.set(0);\n    a.schedule_int(1, delay);\n  =}\n  reaction(a) -> a, out {=\n    out.set(a.get());\n    a.schedule_int(*a.get() + 1, delay);\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    Actions are described here. Actions can carry a\nvalue, a Python object that becomes available to any reaction triggered by\nthe action. This is particularly useful for physical actions that are externally\ntriggered because it enables the action to convey information to the reactor.\nThis could be, for example, the body of an incoming network message or a\nnumerical reading from a sensor. Note that actions do not have a data\ntype in the Python target, even when they carry a value.

    \n

    Recall from Composing Reactors that the\n$after$ keyword on a connection between ports introduces a logical delay. This\nis actually implemented using a logical action. We illustrate how this is done\nusing the\nDelay reactor in the DelayInt\nexample:

    \n
    reactor Delay(delay = 100 ms) {\n  input _in\n  output out\n  logical action a\n  reaction(a) -> out {=\n    if (a.value is not None) and a.is_present:\n      out.set(a.value)\n  =}\n  reaction(_in) -> a {=\n    a.schedule(self.delay, _in.value)\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
        delay = new Delay()\n    <source_port_reference> -> delay._in\n    delay._in -> <destination_port_reference>\n
    \n

    is equivalent to

    \n
        <source_port_reference> -> <destination_port_reference> after 100 ms\n
    \n

    In the Delay reactor, the reaction to the input _in declares as its effect\nthe action a. This declaration makes it possible for the reaction to schedule\na future triggering of a using the\na.schedule()\nmethod.

    \n

    The first reaction declares that it is triggered by a and has effect out. To\nread the value, it uses the a.value syntax. Because this reaction is first,\nthe out at any logical time can be produced before the input _in is even\nknown to be present. Hence, this reactor can be used in a feedback loop, where\nout triggers a downstream reactor to send a message back to _in of this same\nreactor. If the reactions were given in the opposite order, there would be a\ncausality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it as follows:

    \n
      reaction(a) -> out {=\n    if (a.value is not None):\n      out.set(a.value)\n  =}\n
    \n

    It is possible to both be triggered by and schedule an action in the same\nreaction. For example, the\nfollowing CountSelf\nreactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay = 100 ms) {\n  output out\n  logical action a\n  reaction(startup) -> a, out {=\n    out.set(0)\n    a.schedule(self.delay, 1)\n  =}\n  reaction(a) -> a, out {=\n    out.set(a.value)\n    a.schedule(self.delay, a.value + 1)\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    Each action listed as an effect for a reaction is available as a schedulable object in the reaction body via the actions object. The TypeScript target provides a special actions object with a property for each schedulable action. Schedulable actions (of type t) have the object method:

    \n
    schedule: (extraDelay: TimeValue | 0, value?: T) => void;\n
    \n

    The first argument can either be the literal 0 (shorthand for 0 seconds) or a TimeValue/UnitBasedTimeValue. The second argument is the value for the action. Consider the following reactor:

    \n
    target TypeScript;\nreactor Schedule {\n  input x:number;\n  logical action a;\n  reaction(x) -> a {=\n    actions.a.schedule(new UnitBasedTimeValue(200, TimeUnit.msec), null);\n  =}\n  reaction(a) {=\n    let elapsedTime = util.getElapsedLogicalTime();\n    console.log("Action triggered at logical time " + elapsedTime + " after start.");\n  =}\n}\n
    \n

    When this reactor receives an input x, it calls schedule() on the action a, so it will be triggered at the logical time offset (200 msec) with a null value. The action a will be triggered at a logical time 200 milliseconds after the arrival of input x. This will trigger the second reaction, which will use the util.getElapsedLogicalTime() function to determine how much logical time has elapsed since the start of execution. The second argument to the schedule() function is a value, data that can be carried by the action, which is explained below. In the above example, there is no value.

    \n

    Zero-Delay Actions

    \n

    If the specified delay in a schedule() call is zero, then the action a will be triggered one microstep later in superdense time (see Superdense Time). Hence, if the input x arrives at metric logical time t, and you call schedule() as follows:

    \n
    actions.a.schedule(0);\n
    \n

    then when a reaction to a is triggered, the input x will be absent (it was present at the previous microstep). The reaction to x and the reaction to a occur at the same metric time t, but separated by one microstep, so these two reactions are not logically simultaneous. These reactions execute with different Tags.

    \n

    Actions With Values

    \n

    If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. The most common use of this is to implement a logical delay, where a value provided at an input is produced on an output with a larger logical timestamp. To accomplish that, it is much easier to use the after keyword on a connection between reactors. Nevertheless, in this section, we explain how to directly use actions with value. In fact, the after keyword is implemented as described below.

    \n

    If you are familiar with other targets (like C) you may notice it is much easier to schedule actions with values in TypeScript because of TypeScript/JavaScript’s garbage collected memory management. The following example implements a logical delay using an action with a value.

    \n
    reactor Delay(delay:time(100 ms)) {\n  input x:number;\n  output out:number;\n  logical action a:number;\n  reaction(x) -> a {=\n    actions.a.schedule(delay, x as number);\n  =}\n  reaction(a) -> out {=\n    if (a !== null){\n      out = a as number\n    }\n  =}\n}\n
    \n

    The action a is specified with a type number. The first reaction declares a as its effect. This declaration makes it possible for the reaction to schedule a future triggering of a. It’s necessary to explicitly annotate the type of x as a number in the schedule function because TypeScript doesn’t know the only trigger of a reaction must be present in that reaction.

    \n

    The second reaction declares that it is triggered by a and has effect out. When a reaction triggers or uses an action the value of that action is made available within the reaction as a local variable with the name of the action. This variable will take on the value of the action and it will have the value undefined if that action is absent because it was not scheduled for this reaction.

    \n

    The local variable cannot be used directly to schedule an action. As described above, an action a can only be scheduled in a reaction when it is 1) declared as an effect and 2) referenced through a property of the actions object. The reason for this implementation is that actions.a refers to the action, not its value, and it is possible to use both the action and the value in the same reaction. For example, the following reaction will produce a counting sequence after it is triggered the first time:

    \n
    reaction(a) -> out, a {=\n  if (a !== null) {\n    a = a as number;\n    out = a;\n    let newValue = a++;\n    actions.a.schedule(delay, newValue);\n  }\n=}\n
    \n
    \n
    \n

    Actions may carry values if they mention a data type, for instance:

    \n
    logical action act: u32;\n
    \n

    Within a reaction, you can schedule that action with a value like so

    \n
    ctx.schedule_with_v(act, Asap, 30);\n
    \n

    you can get the value from another reaction like so:

    \n
    if let Some(value) = ctx.get_action(act) {\n  // a value is present at this tag\n} else {\n  // value was not specified\n}\n
    \n

    If an action does not mention a data type, the type defaults to ().

    \n
    \n

    Schedule Functions

    \n
    \n

    Actions with values can be rather tricky to use because the value must usually be carried in dynamically allocated memory. It will not work for value to refer to a state variable of the reactor because that state variable will likely have changed value by the time the reactions to the action are invoked. Several variants of the lf_schedule function are provided to make it easier to pass values across time in varying circumstances.

    \n
    \n

    lf_schedule(<action>, <offset>);

    \n
    \n

    This is the simplest version as it carries no value. The action need not have a data type.

    \n
    \n

    lf_schedule_int(<action>, <offset>, <value>);

    \n
    \n

    This version carries an int value. The data type of the action is required to be int.

    \n
    \n

    lf_schedule_token(<action>, <offset>, <value>);

    \n
    \n

    This version carries a token, which has type token_t and points to the value, which can have any type. There is a create_token() function that can be used to create a token, but programmers will rarely need to use this. Instead, you can use lf_schedule_value() (see below), which will automatically create a token. Alternatively, for inputs with types ending in * or [], the value is wrapped in a token, and the token can be obtained using the syntax inputname->token in a reaction and then forwarded using lf_schedule_token() (see Dynamically Allocated Structs above). If the input is mutable, the reaction can then even modify the value pointed to by the token and/or use lf_schedule_token() to send the token to a future logical time. For example, the DelayPointer reactor realizes a logical delay for any data type carried by a token:

    \n
    reactor DelayPointer(delay:time(100 ms)) {\n  input in:void*;\n  output out:void*;\n  logical action a:void*;\n  reaction(a) -> out {=\n    // Using lf_set_token delegates responsibility for\n    // freeing the allocated memory downstream.\n    lf_set_token(out, a->token);\n  =}\n  reaction(in) -> a {=\n    // Schedule the actual token from the input rather than\n    // a new token with a copy of the input value.\n    lf_schedule_token(a, self->delay, in->token);\n  =}\n}\n
    \n
    \n

    lf_schedule_value(<action>, <offset>, <value>, <length>);

    \n
    \n

    This version is used to send into the future a value that has been dynamically allocated using malloc. It will be automatically freed when it is no longer needed. The value argument is a pointer to the memory containing the value. The length argument should be 1 if it is a not an array and the array length otherwise. This length will be needed downstream to interpret the data correctly. See ScheduleValue.lf.

    \n
    \n

    lf_schedule_copy(<action>, <offset>, <value>, <length>);

    \n
    \n

    This version is for sending a copy of some data pointed to by the <value> argument. The data is assumed to be a scalar or array of type matching the <action> type. The <length> argument should be 1 if it is a not an array and the array length otherwise. This length will be needed downstream to interpret the data correctly.

    \n

    Occasionally, an action payload may not be dynamically allocated nor freed. For example, it could be a pointer to a statically allocated string. If you know this to be the case, the DelayString reactor will realize a logical time delay on such a string:

    \n
    reactor DelayString(delay:time(100 msec)) {\n  input in:string;\n  output out:string;\n  logical action a:string;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // The following copies the char*, not the string.\n    lf_schedule_copy(a, self->delay, &(in->value), 1);\n  =}\n}\n
    \n

    The data type string is an alias for char*, but Lingua Franca does not know this, so it creates a token that contains a copy of the pointer to the string rather than a copy of the string itself.

    \n
    \n
    \n

    FIXME: Give a list of schedule() functions with descriptions.

    \n
    \n
    \n

    The Python reactor target provides a .schedule() method to trigger an action at a\nfuture logical time. The .schedule() method also optionally allows for a value\nto be sent into the future. For example, take the\nScheduleValue.lf:

    \n
    main reactor ScheduleValue {\n  logical action a;\n  reaction(startup) -> a {=\n    value = "Hello"\n    a.schedule(0, value)\n  =}\n  reaction(a) {=\n    print("Received: ", a.value)\n    if a.value != "Hello":\n      sys.stderr.write("FAILURE: Should have received 'Hello'\\n")\n      exit(1)\n  =}\n}\n
    \n

    In this example, the logical action a is scheduled one\nmicrostep in the future with a string value\ncontaining \"Hello\".

    \n
    \n
    \n

    FIXME: List them here

    \n
    \n
    \n

    Within a reaction, actions may be scheduled using the schedule function:

    \n
    // schedule without additional delay\nctx.schedule(act, Asap);\n// schedule with an additional delay\nctx.schedule(act, after!(20 ms));\n// that's shorthand for\nctx.schedule(act, After(Duration.of_millis(20)));\n
    \n
    \n

    Stopping Execution

    \n
    \n

    A reaction may request that the execution stop after all events with the current timestamp have been processed by calling the built-in method request_stop(), which takes no arguments. In a non-federated execution, the actual last tag of the program will be one microstep later than the tag at which request_stop() was called. For example, if the current tag is (2 seconds, 0), the last (stop) tag will be (2 seconds, 1). In a federated execution, however, the stop time will likely be larger than the current logical time. All federates are assured of stopping at the same logical time.

    \n
    \n

    The timeout target property will take precedence over this function. For example, if a program has a timeout of 2 seconds and request_stop() is called at the (2 seconds, 0) tag, the last tag will still be (2 seconds, 0).

    \n
    \n
    \n
    \n

    A reaction may request that the execution stop after all events with the current timestamp have been processed by calling the built-in method lf.request_stop(), which takes no arguments. In a non-federated execution, the actual last tag of the program will be one microstep later than the tag at which lf.request_stop() was called. For example, if the current tag is (2 seconds, 0), the last (stop) tag will be (2 seconds, 1). In a federated execution, however, the stop time will likely be larger than the current logical time. All federates are assured of stopping at the same logical time.

    \n
    \n

    The timeout target property will take precedence over this function. For example, if a program has a timeout of 2 seconds and request_stop() is called at the (2 seconds, 0) tag, the last tag will still be (2 seconds, 0).

    \n
    \n
    \n
    \n

    A reaction may request that the execution stops after all events with the current timestamp have been processed by calling environment()->sync_shutdown(). There is also a method environment()->async_shutdown()\nwhich may be invoked from outside an reaction, like an external thread.

    \n
    \n
    \n

    A reaction may request that the execution stop by calling the function util.requestShutdown() which takes no arguments. Execution will not stop immediately when this function is called; all events with the current tag will finish processing and execution will continue for one more microstep to give shutdown triggers a chance to execute. After this additional step, execution will terminate.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n

    Log and Debug Information

    \n
    \n

    A suite of useful functions is provided in util.h for producing messages to be made visible when the generated program is run. Of course, you can always use printf, but this is not a good choice for logging or debug information, and it is not a good choice when output needs to be redirected to a window or some other user interface (see for example the sensor simulator). Also, in federated execution, these functions identify which federate is producing the message. The functions are listed below. The arguments for all of these are identical to printf with the exception that a trailing newline is automatically added and therefore need not be included in the format string.

    \n
      \n
    • \n

      LF_PRINT_DEBUG(format, ...): Use this for verbose messages that are only needed during debugging. Nothing is printed unless the target parameter logging is set to debug. THe overhead is minimized when nothing is to be printed.

      \n
    • \n
    • \n

      LF_PRINT_LOG(format, ...): Use this for messages that are useful logs of the execution. Nothing is printed unless the target parameter logging is set to log or debug. This is a macro so that overhead is minimized when nothing is to be printed.

      \n
    • \n
    • \n

      lf_print(format, ...): Use this for messages that should normally be printed but may need to be redirected to a user interface such as a window or terminal (see register_print_function below). These messages can be suppressed by setting the logging target property to warn or error.

      \n
    • \n
    • \n

      lf_print_warning(format, ...): Use this for warning messages. These messages can be suppressed by setting the logging target property to error.

      \n
    • \n
    • \n

      lf_print_error(format, ...): Use this for error messages. These messages are not suppressed by any logging target property.

      \n
    • \n
    • \n

      lf_print_error_and_exit(format, ...): Use this for catastrophic errors.

      \n
    • \n
    \n

    In addition, a utility function is provided to register a function to redirect printed outputs:

    \n
      \n
    • lf_register_print_function(function): Register a function that will be used instead of printf to print messages generated by any of the above functions. The function should accept the same arguments as printf.
    • \n
    \n
    \n
    \n

    The reactor-cpp library provides logging utilities in logging.hh for producing messages to be made visible when the generated program is run. Of course std::cout or printf can be used for the same purpose, but the logging mechanism provided by reactor-cpp is thread-safe ensuring that messages produced in parallel reactions are not interleaved with each other and provides common way for turning messages of a certain severity on and off.

    \n

    In particular, reactor-cpp provides the following logging interfaces:

    \n
      \n
    • reactor::log::Debug(): for verbose debug messages
    • \n
    • reactor::log::Info(): for info messages of general interest, info is the default severity level
    • \n
    • reactor::log::Warning(): for warning messages
    • \n
    • reactor::log::Error(): for errors
    • \n
    \n

    These utilities can be used analogues to std::cout. For instance:

    \n
    reactor::log::Info() << "Hello World! It is " << get_physical_time();\n
    \n

    Note that unlike std::cout the new line delimiter is automatically added to the end of the message.

    \n

    Which type of messages are actually produced by the compiled program can be controlled with the log-level target property.

    \n
    \n
    \n

    The Python supports the logging target specification. This will cause the runtime to produce more or less information about the execution. However, user-facing functions for different logging levels are not yet implemented (see issue #619).

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    The executable reacts to the environment variable RUST_LOG, which sets the logging level of the application. Possible values are\noff, error, warn, info, debug, trace

    \n

    Error and warning logs are on by default. Enabling a level enables all greater levels (i.e., RUST_LOG=info also enables warn and error, but not trace or debug).

    \n

    Logging can also be turned on with the --log-level CLI option, if the application features a CLI.

    \n

    Note that the logging target property is ignored, as its levels do not match the Rust standard levels we use (those of the log crate).

    \n

    Note that when building with a release profile (i.e., target property build-type is not Debug), all log statements with level debug and trace are removed from the executable, and cannot be turned on at runtime. A warning is produced by the executable if you try to use these levels explicitly.

    \n
    \n

    Libraries Available to Programmers

    \n
    \n

    Libraries Available in All Programs

    \n

    Reactions in C can use a number of pre-defined functions, macros, and constants without having to explicitly include any header files:

    \n
      \n
    • \n

      Time and tags (tag.h):

      \n
        \n
      • Specifying time value, such as MSEC and FOREVER
      • \n
      • Time data types, such as tag_t and instant_t
      • \n
      • Obtaining tag and time information, e.g. lf_time_logical and lf_time_physical
      • \n
      \n
    • \n
    • \n

      Ports

      \n
        \n
      • Writing to output ports, such as lf_set and lf_set_token (set.h)
      • \n
      • Iterating over sparse multiports, such as lf_multiport_iterator and lf_multiport_next (port.h)
      • \n
      \n
    • \n
    • \n

      Scheduling actions

      \n
        \n
      • Schedule future events, such as lf_schedule and lf_schedule_value (api.h)
      • \n
      \n
    • \n
    • \n

      File Access

      \n
        \n
      • LF_SOURCE_DIRECTORY: A C string giving the full path to the directory containing the .lf file of the program.
      • \n
      • LF_PACKAGE_DIRECTORY: A C string giving the full path to the directory that is the root of the project or package (normally, the directory above the src directory).
      • \n
      • LF_FILE_SEPARATOR: A C string giving the file separator for the platform containing the .lf file (”/” for Unix-like systems, ”\\” for Windows).
      • \n
      \n
    • \n
    \n

    These are useful when your application needs to open and read additional files. For example, the following C code can be used to open a file in a subdirectory called dir of the directory that contains the .lf file:

    \n
        const char* path = LF_SOURCE_DIRECTORY LF_FILE_SEPARATOR "dir" LF_FILE_SEPARATOR "filename"\n    FILE* fp = fopen(path, "rb");
    \n
      \n
    • \n

      Miscellaneous

      \n
        \n
      • Changing modes in modal models, lf_set_mode (set.h)
      • \n
      • Checking deadlines, lf_check_deadline (api.h)
      • \n
      • Defining and recording tracepoints, such as register_user_trace_event and tracepoint (trace.h)
      • \n
      • Printing utilities, such as lf_print and lf_print_error (util.h)
      • \n
      • Logging utilities, such as LF_PRINT_LOG and LF_PRINT_DEBUG (util.h)
      • \n
      \n
    • \n
    \n

    Standard C Libraries

    \n

    The generated C code automatically includes the following standard C libraries (see also the C standard library header files):

    \n
      \n
    • limits.h (Defines INT_MIN, INT_MAX, etc.)
    • \n
    • stdbool.h (Defines bool datatype and true and false constants)
    • \n
    • stddef.h (Defines size_t, NULL, etc.)
    • \n
    • stdint.h (Defines int64_t, int32_t, etc.)
    • \n
    • stdlib.h (Defines exit, getenv, atoi, etc.)
    • \n
    \n

    Hence, programmers are free to use functions from these libraries without explicitly providing a #include statement. Nevertheless, providing one is harmless and may be good form. In particular, future releases may not include these header files

    \n

    Available Libraries Requiring #include

    \n

    More sophisticated library functions require a #include statement in a $preamble$.\nSpecifically, platform.h includes the following:

    \n
      \n
    • Sleep functions such as lf_sleep
    • \n
    • Mutual exclusion such as lf_critial_section_enter and lf_critical_section_exit
    • \n
    • Threading functions such as lf_thread_create
    • \n
    \n

    The threading functions are only available for platforms that support multithreading.

    \n

    Available Libraries Requiring #include, a files entry, and a cmake-include

    \n

    A few utility libraries are provided, but require considerably more setup.\nThese also help to illustrate how to incorporate your own libraries.

    \n\n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    Scheduler Target Property

    \n

    The scheduler target property is used to select the scheduler used by the C runtime. This scheduler determines the exact order in which reactions are processed, as long as the order complies with the deterministic semantics of Lingua Franca. It also assigns reactions to user-level threads and can thereby influence the assignment of reactions to processors.

    \n

    Because the C runtime scheduler operates at a higher level of abstraction than the OS, none of the scheduling policies that we currently support allow preemption; furthermore, they do not control migration of threads between processors.

    \n

    Another limitation of these schedulers is that they are constrained to process the reaction graph breadth-first. We define the level of a reaction r to be the length of the longest chain of causally dependent reactions that are all (causally) upstream of r. Current LF schedulers process one level of reactions at a time, but this constraint is more restrictive than necessary to implement Lingua Franca’s semantics and is notable only for its effect on execution times.

    \n

    The following schedulers are available:

    \n
      \n
    • NP (non-preemptive): This scheduler is the default scheduler. It ignores deadlines.
    • \n
    • GEDF_NP (global earliest-deadline-first, non-preemptive): When the semantics of Lingua Franca allows for concurrent execution of two or more ready reactions with the same level at a particular tag, this scheduler will prioritize the reaction with the earliest deadline to run first. Reactions with no explicit deadline implicitly have an infinitely late deadline.
    • \n
    • adaptive: This experimental scheduler behaves similarly to the NP scheduler, with the additional limitation that it is designed for applications that have potentially wide variability in physical execution times. It performs experiments to measure execution times at runtime to determine the degree of exploitable parallelism in various parts of the program. This lets it automate judgments which are made more naively by the other schedulers and which are typically made by the programmer in general-purpose languages.
    • \n
    \n
    \n

    Target Implementation Details

    \n
    \n

    Included Libraries

    \n

    Definitions for the following do not need to be explicitly included because the code generator exposes them in the user namespace automatically:

    \n
      \n
    • Functions and macros used to set ports and iterate over multiports
    • \n
    • Functions and macros used to schedule actions
    • \n
    • Functions and macros used to set a reactor’s mode
    • \n
    • Functions and macros used to create trace points
    • \n
    • Logging utility functions
    • \n
    • Typedefs relating to time and logical time, including tag_t, instant_t, interval_t, and microstep_t
    • \n
    • API functions for obtaining timing information about the current program execution, including the current physical and logical time
    • \n
    \n

    Some standard C libraries are exposed to the user through reactor.h, including stddef.h,\nstdio.h, and stdlib.h. In addition, math.h gets automatically included. However, users who wish to avoid breaking changes between releases should\nconsider including these libraries explicitly instead of relying on their being exposed by the\nruntime.

    \n

    Users who wish to include functionality that has a platform-specific implementation may choose to\nexplicitly include platform.h, which provides a uniform interface for various concurrency\nprimitives and sleep functions.

    \n

    Multithreaded Implementation

    \n

    By default, the C runtime system uses multiple worker threads in order to take advantage of multicore execution. The number of worker threads will match the number of cores on the machine unless the workers argument is given in the target statement or the --workers command-line argument is given.

    \n

    Upon initialization, the main thread will create the specified number of worker threads.\nExecution proceeds in a manner similar to the single threaded implementation\nexcept that the worker threads concurrently draw reactions from the reaction queue.\nThe execution algorithm ensures that no reaction executes until all reactions that it depends on have executed or it has been determined that they will not execute at the current tag.

    \n

    Single Threaded Implementation

    \n

    By giving the single-threaded (target option)[/docs/handbook/target-declaration#single-threaded] or the --single-threaded (command-line argument)[/docs/handbook/target-declaration#command-line-arguments], the generated program will execute the program using only a single thread. This option is most useful for creating programs to run on bare-metal microprocessors that have no threading support. On such platforms, mutual exclusion is typically realized by disabling interrupts.

    \n

    The execution strategy is to have two queues of pending accessor invocations, one that is sorted by\ntag (the event queue) and one that is sorted by priority (the reaction queue).\nExecution proceeds as follows:

    \n
      \n
    1. \n

      At initialization, an event for each timer is put on the event queue and logical time is initialized to the current time, represented as the number of nanoseconds elapsed since January 1, 1970.

      \n
    2. \n
    3. \n

      At each logical time, pull all events from event queue that have the same earliest tag, find the reactions that these events trigger, and put them on the reaction queue. If there are no events on the event queue, then exit the program (unless the --keepalive true (command-line argument)[/docs/handbook/target-declaration#command-line-arguments] is given).

      \n
    4. \n
    5. \n

      Wait until physical time matches or exceeds that earliest timestamp (unless the --fast true (command-line argument)[/docs/handbook/target-declaration#command-line-arguments] is given). Then advance logical time to match that earliest timestamp.

      \n
    6. \n
    7. \n

      Execute reactions in order of priority from the reaction queue. These reactions may produce outputs, which results in more events getting put on the reaction queue. Those reactions are assured of having lower priority than the reaction that is executing. If a reaction calls lf_schedule(), an event will be put on the event queue, not the reaction queue.

      \n
    8. \n
    9. \n

      When the reaction queue is empty, go to 2.

      \n
    10. \n
    \n
    \n
    \n

    Unlike the C target, the Cpp target implements more of the analysis and setup of a Lingua Franca in the runtime system. The runtime system is define in the reactor-cpp repository on GitHub. See that repo for details.

    \n
    \n
    \n

    The Python target is built on top of the C runtime to enable maximum efficiency where possible. It uses the single-threaded C runtime by default but will switch to the multi-threaded C runtime if a physical action is detected. The threading target property can be used to override this behavior.

    \n

    Running lfc on a XXX.lf program that uses the Python target specification on a\nLinux machine will create the following files (other operating systems will have\na slightly different structure and/or files):

    \n
    ├── src\n│   └── XXX.lf\n└── src-gen\n    └── XXX\n        ###### Files related to the Python C extension module for XXX ######\n        ├── build               # Temporary files for setuptools\n        ├── core                # Core C runtime files\n        ├── ctarget.c           # C target API implementations\n        ├── ctarget.h           # C target API definitions\n        ├── LinguaFrancaXXX*.so # The Python C extension module for XXX\n        ├── pythontarget.c      # Python target API implementations\n        ├── pythontarget.h      # Python target API definitions\n        ├── setup.py            # Setup file used to build the Python C extension\n        ├── XXX.c               # Source code of the Python C extension\n        ###### Files containing the Python code ######\n        └── XXX.py              # Python file containing reactors and reaction code\n
    \n

    There are two major components in the src-gen/XXX directory that together enable the execution of a Python target application:

    \n
      \n
    • An XXX.py file containing the user code (e.g., reactor definitions and reactions).
    • \n
    • The source code for a Python C extension module called LinguaFrancaXXX containing the C runtime, as well as hooks to execute the user-defined reactions.
    • \n
    \n

    The interactions between the src-gen/XXX/XXX.py file and the LinguaFrancaXXX module are explained below.

    \n

    The XXX.py file containing user code

    \n

    The XXX.py file contains all the reactor definitions in the form of Python classes. The contents of a reactor are converted as follows:

    \n
      \n
    • Each Reaction in a reactor definition will be converted to a class method.
    • \n
    • Each Parameter will be converted to a class property to make it read-only.
    • \n
    • Each State variable will be converted to an instance variable.
    • \n
    • Each trigger and effect will be converted to an object passed as a method function argument to reaction methods, allowing the body of the reaction to access them.
    • \n
    • Each reactor Preamble will be put in the class definition verbatim.
    • \n
    \n

    Finally, each reactor class instantiation will be converted to a Python object class instantiation.

    \n

    For example, imagine the following program:

    \n
    # src/XXX.lf\ntarget Python;\nreactor Foo(bar(0)) {\n  preamble {=\n    import random\n  =}\n  state baz\n  input _in\n  logical action act\n  reaction(_in, act) {=\n    # Body of the reaction\n    self.random.seed() # Note the use of self\n  =}\n}\nmain reactor {\n  foo = new Foo()\n}\n
    \n

    Th reactor Foo and its instance, foo, will be converted to

    \n
    # src-gen/XXX/XXX.py\n...\n# Python class for reactor Foo\nclass _Foo:\n\n    # From the preamble, verbatim:\n    import random\n    def __init__(self, **kwargs):\n        #Define parameters and their default values\n        self._bar = 0\n        # Handle parameters that are set in instantiation\n        self.__dict__.update(kwargs)\n\n        # Define state variables\n        self.baz = None\n\n    @property\n    def bar(self):\n        return self._bar\n\n    def reaction_function_0(self , _in, act):\n        # Body of the reaction\n        self.random.seed() # Note the use of self\n        return 0\n\n\n# Instantiate classes\nxxx_lf = [None] * 1\nxxx_foo_lf = [None] * 1\n# Start initializing XXX of class XXX\nfor xxx_i in range(1):\n    bank_index = xxx_i\n    xxx_lf[0] = _XXX(\n        _bank_index = 0,\n    )\n    # Start initializing XXX.foo of class Foo\n    for xxx_foo_i in range(1):\n        bank_index = xxx_foo_i\n        xxx_foo_lf[0] = _Foo(\n            _bank_index = 0,\n            _bar=0,\n        )\n...\n
    \n

    The generated LinguaFrancaXXX Python module (a C extension module)

    \n

    The rest of the files in src-gen/XXX form a Python C extension\nmodule\ncalled LinguaFrancaXXX that can be built by executing python3 setup.py build_ext --inplace in the src-gen/XXX/ folder. In this case, Python will\nread the instructions in the src-gen/XXX/setup.py file and build a\nLinguaFrancaXXX module in src-gen/XXX/. The --inplace flag puts the\ncompiled extension (the LinguaFrancaXXX*.so in the example above) in the\nsrc-gen directory alongside the XXX.py file.

    \n

    As mentioned before, the LinguaFrancaXXX module is separate from\nsrc-gen/XXX/XXX.py but interacts with it. Next, we explain this interaction.

    \n

    Interactions between XXX.py and LinguaFrancaXXX

    \n

    The LinguaFrancaXXX module is imported in src-gen/XXX/XXX.py:

    \n
    from LinguaFrancaXXX import *\n
    \n

    This is done to enable the main function in src-gen/XXX/XXX.py to make a call to the start() function, which is part of the generated (and installed) LinguaFrancaXXX module. This function will start the main event handling loop of the C runtime.

    \n

    From then on, LinguaFrancaXXX will call reactions that are defined in src-gen/XXX/XXX.py when needed.

    \n

    The LinguaFrancaBase package

    \n

    LinguaFrancaBase is a package that contains several helper methods and definitions that are necessary for the Python target to work. This module is installable via python3 -m pip install LinguaFrancaBase but is automatically installed if needed during the installation of LinguaFrancaXXX. The source code of this package can be found on GitHub.

    \n

    This package’s modules are imported in the XXX.py program:

    \n
    from LinguaFrancaBase.constants import * #Useful constants\nfrom LinguaFrancaBase.functions import * #Useful helper functions\nfrom LinguaFrancaBase.classes import * #Useful classes\n
    \n

    Already imported Python modules

    \n

    The following packages are already imported and thus do not need to be re-imported by the user:

    \n
    import os\nimport sys\nimport copy\n
    \n
    \n
    \n

    When a TypeScript reactor is compiled, the generated code is placed inside a project directory. This is because there are two steps of compilation. First, the Lingua Franca compiler generates a TypeScript project from the TypeScript reactor code. Second, the Lingua Franca compiler runs a TypeScript compiler on the generated TypeScript project to produce executable JavaScript. This is illustrated below:

    \n
    Lingua Franca (.lf) ==> TypeScript (.ts) ==> JavaScript (.js)\n
    \n

    Assuming the directory containing our Lingua Franca file Foo.lf is named TS, the compiler will generate the following:

    \n
      \n
    1. TS/package.json
    2. \n
    3. TS/node_modules
    4. \n
    5. TS/Foo/tsconfig.json
    6. \n
    7. TS/Foo/babel.config.js
    8. \n
    9. TS/Foo/src/
    10. \n
    11. TS/Foo/dist/
    12. \n
    \n

    Items 1, 3, and 4 are configuration files for the generated project. Item 2 is a node_modules directory with contents specified by item 1. Item 5 is the directory for generated TypeScript code. Item 6 is the directory for compiled JavaScript code. In addition to the generated code for your Lingua Franca program, items 5 and 6 include libraries from the reactor-ts submodule.

    \n

    The Lingua Franca compiler automatically invokes other programs as it compiles a Lingua Franca (.lf) file to a Node.js executable JavaScript (.js) file. The files package.json, babel.config.js, and tsconfig.json are used to configure the behavior of those other programs. Whenever you compile a .lf file for the first time, the Lingua Franca compiler will copy default versions of these configuration files into the new project so the other programs can run. The Lingua Franca compiler will only copy a default configuration file into a project if that file is not already present in the generated project. This means you, the reactor programmer, may safely modify these configuration files to control the finer points of compilation. Beware, other generated files in the project’s src and dist directories may be overwritten by the compiler.

    \n

    package.json

    \n

    Node.js uses a package.json file to describe metadata relevant to a Node project. This includes a list of project dependencies (i.e. modules) used by the project. When the Lingua Franca compiler copies a default package.json file into a Lingua Franca project that doesn’t already have a package.json, the compiler runs the command npm install to create a node_modules directory. The default package.json only lists dependencies for the reactor-ts submodule. Follow these instructions to modify package.json if you want to use other Node modules in your reactors.

    \n

    tsconfig.json

    \n

    After generating a TypeScript program from a .lf file, the Lingua Franca compiler uses the TypeScript compiler tsc to run a type check. The behavior of tsc is configured by the tsconfig.json file. You probably won’t need to modify tsconfig.json, but you can if you know what you’re doing.

    \n

    babel.config.js

    \n

    If the tsc type check was successful, the Lingua Franca compiler uses babel to compile the generated TypeScript code into JavaScript. (This blog post articulates the advantages of using babel over tsc to generate JavaScript.) There are many different flavors of JavaScript and the babel.config.js file specifies exactly what babel should generate. This is the file to edit if you want the Lingua Franca compiler to produce a different version of JavaScript as its final output.

    \n

    Debugging Type Errors

    \n

    Let’s take the minimal reactor example, and intentionally break it by adding a type error into the reaction.

    \n
    target TypeScript;\nmain reactor ReactionTypeError {\n    timer t;\n    reaction(t) {=\n        let foo:number = "THIS IS NOT A NUMBER";\n        console.log("Hello World.");\n    =}\n}\n
    \n

    This reactor will not compile, and should you attempt to compile it you will get an output from the compiler which looks something like this:

    \n
    --- Standard output from command:\nsrc/ReactionTypeError.ts(23,25): error TS2322: Type '"THIS IS NOT A NUMBER"' is not assignable to type 'number'.\n\n--- End of standard output.
    \n

    In particular the output

    \n
    src/ReactionTypeError.ts(23,25): error TS2322: Type '"THIS IS NOT A NUMBER"' is not assignable to type 'number'.
    \n

    identifies the problem: surprisingly, the string \"THIS IS NOT A NUMBER\" is not a number. However the line information (23,25) is a little confusing because it points to the location of the type error in the generated .ts file ReactionTypeError/src/ReactionTypeError.ts not in the original .lf file ReactionTypeError.lf. The .ts files produced by the TypeScript code generator are quite readable if you are familiar with the reactor-ts submodule, but even if you aren’t familiar it is not too difficult to track down the problem. Just open ReactionTypeError/src/ReactionTypeError.ts in your favorite text editor (we recommend Visual Studio for its excellent TypeScript integration) and look at line 23.

    \n
    14        this.addReaction(\n15            new Triggers(this.t),\n16            new Args(this.t),\n17            function (this, __t: Readable<Tag>) {\n18                // =============== START react prologue\n19                const util = this.util;\n20                let t = __t.get();\n21                // =============== END react prologue\n22                try {\n23                    let foo:number = "THIS IS NOT A NUMBER";\n24                    console.log("Hello World.");\n25                } finally {\n26                    // =============== START react epilogue\n27\n28                    // =============== END react epilogue\n29                }\n30            }\n31        );\n
    \n

    There (inside the try block) we can find the problematic reaction code. Reaction code is copied verbatim into generated .ts files.

    \n

    It can be a bit harder to interpret type errors outside of reaction code, but most type error messages are still relatively clear. For example if you attempt to connect a reactor output to an incompatibly typed input like:

    \n
    target TypeScript;\nmain reactor ConnectionError {\n    s = new Sender();\n    r = new Receiver();\n    s.foo -> r.bar;\n}\nreactor Sender {\n    output foo:number;\n}\nreactor Receiver {\n    input bar:string;\n}\n
    \n

    you should get an error like

    \n
    --- Standard output from command:\nsrc/InputTypeError.ts(36,23): error TS2345: Argument of type 'OutPort<number>' is not assignable to parameter of type 'Port<string>'.\n  Types of property 'value' are incompatible.\n    Type 'number | undefined' is not assignable to type 'string | undefined'.\n      Type 'number' is not assignable to type 'string | undefined'.\n\n--- End of standard output.
    \n

    The key message being Argument of type 'OutPort<number>' is not assignable to parameter of type 'Port<string>'.

    \n

    One last tip: if you attempt to reference a port, action, timer etc. named foo that isn’t declared in the triggers, uses, or effects declaration of the reaction, you will get the error Cannot find name 'foo' in the reaction body.

    \n

    Utility Function Reference

    \n

    These utility functions may be called within a TypeScript reaction:

    \n

    util.requestShutdown(): void Ends execution after one microstep. See Stopping Execution.

    \n

    util.getCurrentTag(): Tag Gets the current (logical) tag. See Tags.

    \n

    util.getCurrentLogicalTime(): TimeValue Gets the current logical TimeValue. See Time.

    \n

    util.getCurrentPhysicalTime(): TimeValue Gets the current physical TimeValue. See Time.

    \n

    util.getElapsedLogicalTime(): TimeValue Gets the elapsed logical TimeValue from execution start. See Time.

    \n

    util.getElapsedPhysicalTime(): TimeValue Gets the elapsed physical TimeValue from execution start. See Time.

    \n

    util.success(): void Invokes the reactor-ts App’s default success callback. FIXME: Currently doesn’t do anything in Lingua Franca.

    \n

    util.failure(): void Invokes the reactor-ts App’s default failure callback. Throws an error.

    \n

    Building Reactor-ts Documentation

    \n

    To build and view proper documentation for time.ts (and other reactor-ts libraries), install typedoc and run

    \n
    typedoc --out docs src\n
    \n

    from the root of the reactor-ts. You probably already have the reactor-ts submodule at

    \n
    lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/
    \n

    You should see an output like.

    \n
    Using TypeScript 3.8.3 from /usr/local/lib/node_modules/typescript/lib\nRendering [========================================] 100%\n\nDocumentation generated at /Users/<username>/git/lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/docs
    \n

    Open that path in a browser with /index.html appended to the end like

    \n
    /Users/<username>/git/lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/docs/index.html
    \n

    to navigate the docs.

    \n
    \n
    \n

    Target Properties

    \n

    Target properties may be mentioned like so:

    \n
    target Rust {\n    // enables single-file project layout\n    single-file-project: false,\n    // timeout for the execution. The program will shutdown at most after the specified duration.\n    timeout: 3 sec,\n\n    cargo-features: ["cli"]\n}\n
    \n

    See Target Declaration for the full list of supported target properties.

    \n

    The Executable

    \n

    The executable name is the name of the main reactor transformed to snake_case: main reactor RustProgram will generate rust_program. See Command-Line Arguments for details.

    \n

    File layout

    \n

    The Rust code generator generates a Cargo project with a classical layout:

    \n
    ├── Cargo.lock\n├── Cargo.toml\n├── src\n│   ├── main.rs\n│   └── reactors\n│       ├── mod.rs\n|       ├── ...\n|\n└── target\n    ├── ...
    \n

    The module structure is as follows:

    \n
      \n
    • the crate has a module reactors
    • \n
    • each LF reactor has its own submodule of reactors. For instance, Minimal.lf will generate minimal.rs. The name is transformed to snake_case.
    • \n
    \n

    This means that to refer to the contents of another reactor module, e.g. that of Other.lf, you have to write super::other::Foo. This is relevant to access preamble items.

    \n

    Single-file layout

    \n

    The Rust target supports an alternative file layout, where all reactors are generated into the main.rs file, making the project fit in a single file (excluding Cargo.toml). The module structure is unchanged: the file still contains a mod reactors { ... } within which each reactor has its mod foo { ... }. You can thus change the layout without having to update any LF code.

    \n

    Set the target property single-file-project: true to use this layout.

    \n

    Note: this alternative layout is provided for the purposes of making self-contained benchmark files. Generating actual runnable benchmarks from an LF file may be explored in the future.

    \n

    Specifying dependencies

    \n

    The Rust code generator leverages Cargo to allow LF programs to profit from Rust’s large package ecosystem. The code generator may also link support files written in pure Rust into the generated crate. Target properties are used to achieve all this.

    \n

    Adding cargo dependencies

    \n

    The cargo-dependencies target property may be used to specify dependencies on crates coming from crates.io. Here’s an example:

    \n
    target Rust {\n   cargo-dependencies: {\n      termcolor: "0.8"\n   }\n};\n
    \n

    The value of the cargo-dependencies property is a map of crate identifiers to a dependency-spec. An informal example follows:

    \n
    cargo-dependencies: {\n   // Name-of-the-crate: "version"\n   rand: "0.8",\n   // Equivalent to using an explicit map:\n   rand: {\n     version: "0.8"\n   },\n   // The map allows specifying more details\n   rand: {\n     // A path to a local unpublished crate.\n     // Note 'path' is mutually exclusive with 'git'.\n     path: "/home/me/Git/local-rand-clone"\n   },\n   rand: {\n     // A URL to a git repo\n     git: "https://github.com/me/rand",\n     // Specify an explicit Git revision number\n     rev: "abcdef1234"\n   },\n   rand: {\n     version: "0.8",\n     // you can specify cargo features\n     features: ["some-cargo-feature",]\n   }\n}\n
    \n

    When a dependency-spec is specified as an object, its key-value pairs correspond directly to those of a Cargo dependency specification. For instance for the following dependency spec:

    \n
       rand: {\n     version: "0.8",\n     // you can specify cargo features\n     features: ["some-cargo-feature",]\n   }\n
    \n

    we add the following to the generated Cargo.toml:

    \n
    [dependencies.rand]\nversion = "0.8"\nfeatures = ["some-cargo-feature"]\n
    \n

    Not all keys are necessarily supported though, e.g. the registry key is not supported (yet).

    \n

    Configuring the runtime

    \n

    The runtime crate can be configured just like other crates, using the cargo-dependencies target property, e.g.:

    \n
    cargo-dependencies: {\n   reactor_rt: {\n     features: ["parallel-runtime"]\n   }\n}\n
    \n

    The dependency is always included, with defaults picked by LFC. The location information (path/git/version key) is optional.\nSee reactor_rt for the supported features.

    \n

    Linking support files

    \n

    You can link-in additional rust modules using the rust-include target property:

    \n
    target Rust {\n  rust-include: ["foo.rs"]\n};\n
    \n

    The property is a list of paths (relative to the directory containing the .lf file). Each path should either point to a Rust file (.rs), or a directory that contains a mod.rs file. Each of those will be copied to the src directory of the generated Cargo project, and linked in to the main.rs file.

    \n

    To refer to the included module, you can use e.g. crate::foo if your module is named foo.

    \n

    Generation scheme

    \n

    Each reactor generates its own struct which contains state variables. For instance,

    \n\n\n\n\n\n\n\n\n\n\n\n\n\n
    LFGenerated Rust
    \n
    reactor SomeReactor {\n  state field: u32(0)\n}\n
    \n
    \n
    struct SomeReactor {\n  field: u32\n}\n
    \n
    \n

    In the following we refer to that struct as the state struct.

    \n

    Reactions

    \n

    Reactions are each generated in a separate method of the reactor struct. Reaction names are unspecified and may be mangled to prevent explicit calling. The parameters of that method are

    \n
      \n
    • &mut self: the state struct described above,
    • \n
    • ctx: &mut ReactionCtx: the context object for the reaction execution,
    • \n
    • For each dependency, a parameter is generated.\n
        \n
      • If the dependency is a component of this reactor, the name of the parameter is just the name of the component
      • \n
      • If the dependency is a port of a child reactor, the name of the parameter is <name of the child instance>__<name of the port>, e.g. child__out for child.out.
      • \n
      • The type of the parameter depends on the kind of dependency and of component:
      • \n
      \n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
      ComponentUse/trigger dependencyEffect dependency
      \n\n\n

      Port of type T

      \n
      \n

      &ReadablePort<T>

      \n
      \n

      WritablePort<T>

      \n
      \n

      Logical action of type T

      \n
      \n

      &LogicalAction<T>

      \n
      \n

      &mut LogicalAction<T>

      \n
      \n

      Physical action of type T

      \n
      \n

      &PhysicalActionRef<T>

      \n
      \n

      &mut PhysicalActionRef<T>

      \n
      Timer\n

      &Timer

      \n
      \n

      n/a

      \n
      \n

      Port bank of type T

      \n
      \n

      &ReadablePortBank<T>

      \n
      \n

      WritablePortBank<T>

      \n
      \n

      Undeclared dependencies, and dependencies on timers and startup or shutdown, do not generate a parameter.

      \n

      The ReactionCtx object is a mediator to manipulate all those dependency objects. It has methods to set ports, schedule actions, retrieve the current logical time, etc.

      \n

      For instance:

      \n
      reactor Source {\n    output out: i32;\n    reaction(startup) -> out {=\n        ctx.set(out, 76600)\n    =}\n}\n
      \n

      In this example, the context object ctx is used to set a port to a value. The port is in scope as out.

      \n
      \n

      ⚠ TODO when the runtime crate is public link to the docs, they should be the most exhaustive documentation.

      \n
      \n
    \n````","headings":[{"value":"Overview","depth":2},{"value":"Requirements","depth":2},{"value":"Limitations","depth":2},{"value":"The Target Specification","depth":2},{"value":"Parameters and State Variables","depth":2},{"value":"Array Values for Parameters","depth":3},{"value":"Array Values for States","depth":3},{"value":"States and Parameters with Struct Values","depth":3},{"value":"Array-Valued Parameters","depth":3},{"value":"State Variables","depth":3},{"value":"Array Expressions for State Variables and Parameters","depth":3},{"value":"Assigning Arbitrary Initial Expressions to State Variables and Parameters","depth":3},{"value":"Array or Object Parameters","depth":3},{"value":"Inputs and Outputs","depth":2},{"value":"Sending and Receiving Data","depth":3},{"value":"Persistent Inputs","depth":3},{"value":"Fixed Length Array Inputs and Outputs","depth":3},{"value":"Variable Length Array Inputs and Outputs","depth":3},{"value":"Dynamically Allocated Data","depth":3},{"value":"Mutable Inputs","depth":3},{"value":"String Types","depth":3},{"value":"Macros For Setting Output Values","depth":3},{"value":"Sending and Receiving Large Data Types","depth":3},{"value":"Sending and Receiving Objects","depth":3},{"value":"Sending and Receiving Custom Types","depth":3},{"value":"Time","depth":2},{"value":"Tags","depth":3},{"value":"Summary of Time Functions","depth":3},{"value":"Actions","depth":2},{"value":"Zero-Delay Actions","depth":3},{"value":"Actions With Values","depth":3},{"value":"Zero-Delay Actions","depth":3},{"value":"Actions With Values","depth":3},{"value":"Schedule Functions","depth":3},{"value":"Stopping Execution","depth":2},{"value":"Log and Debug Information","depth":2},{"value":"Libraries Available to Programmers","depth":2},{"value":"Libraries Available in All Programs","depth":4},{"value":"Standard C Libraries","depth":4},{"value":"Available Libraries Requiring #include","depth":4},{"value":"Available Libraries Requiring #include, a files entry, and a cmake-include","depth":4},{"value":"Scheduler Target Property","depth":2},{"value":"Target Implementation Details","depth":2},{"value":"Included Libraries","depth":3},{"value":"Multithreaded Implementation","depth":3},{"value":"Single Threaded Implementation","depth":3},{"value":"The XXX.py file containing user code","depth":3},{"value":"The generated LinguaFrancaXXX Python module (a C extension module)","depth":3},{"value":"Interactions between XXX.py and LinguaFrancaXXX","depth":3},{"value":"The LinguaFrancaBase package","depth":3},{"value":"Already imported Python modules","depth":3},{"value":"package.json","depth":3},{"value":"tsconfig.json","depth":3},{"value":"babel.config.js","depth":3},{"value":"Debugging Type Errors","depth":3},{"value":"Utility Function Reference","depth":3},{"value":"Building Reactor-ts Documentation","depth":3},{"value":"Target Properties","depth":3},{"value":"The Executable","depth":3},{"value":"File layout","depth":3},{"value":"Single-file layout","depth":4},{"value":"Specifying dependencies","depth":3},{"value":"Adding cargo dependencies","depth":4},{"value":"Configuring the runtime","depth":4},{"value":"Linking support files","depth":4},{"value":"Generation scheme","depth":3},{"value":"Reactions","depth":4}],"frontmatter":{"permalink":"/docs/handbook/target-language-details","title":"Target Language Details","oneline":"Detailed reference for each target langauge.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Expressions","oneline":"Expressions in Lingua Franca.","permalink":"/docs/handbook/expressions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Target Declaration","oneline":"The target declaration and its parameters in Lingua Franca.","permalink":"/docs/handbook/target-declaration"}}}},"pageContext":{"id":"3-target-language-details","slug":"/docs/handbook/target-language-details","repoPath":"/packages/documentation/copy/en/reference/Target Language Details.md","previousID":"24e3b5ae-ac22-5a79-956a-4258d40ae77c","nextID":"de456861-0847-5726-aba2-da3ba779bd9d","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/target-language-details","result":{"data":{"markdownRemark":{"id":"3af06b85-61a2-5d81-a7ad-232624b3eaf6","excerpt":"$page-showing-target$ Overview In the C reactor target for Lingua Franca, reactions are written in C and the code generator generates one or more standalone C…","html":"

    $page-showing-target$

    \n

    Overview

    \n
    \n

    In the C reactor target for Lingua Franca, reactions are written in C and the code generator generates one or more standalone C programs that can be compiled and run on several platforms. It has been tested on macOS, Linux, Windows, and at least one bare-iron embedded platform. The single-threaded version (which you get by setting the threading target parameter to false) is the most portable, requiring only a handful of common C libraries (see Included Libraries below). The multithreaded version requires a small subset of the POSIX thread library (pthreads) and transparently executes in parallel on a multicore machine while preserving the deterministic semantics of Lingua Franca.

    \n

    Note that C is not a safe language. There are many ways that a programmer can circumvent the semantics of Lingua Franca and introduce nondeterminism and illegal memory accesses. For example, it is easy for a programmer to mistakenly send a message that is a pointer to data on the stack. The destination reactors will very likely read invalid data. It is also easy to create memory leaks, where memory is allocated and never freed. Here, we provide some guidelines for a style for writing reactors that will be safe.

    \n

    NOTE: If you intend to use C++ code or import C++ libraries in the C target, we provide a special CCpp target that automatically uses a C++ compiler by default. Alternatively, you might want to use the Cpp target.

    \n
    \n
    \n

    In the C++ reactor target for Lingua Franca, reactions are written in C++ and the code generator generates a standalone C++ program that can be compiled and run on all major platforms. Our continuous integration ensures compatibility with Windows, macOS, and Linux.\nThe C++ target solely depends on a working C++ build system including a recent C++ compiler (supporting C++17) and CMake (>= 3.5). It relies on the reactor-cpp runtime, which is automatically fetched and compiled in the background by the Lingua Franca compiler.

    \n

    Note that C++ is not a safe language. There are many ways that a programmer can circumvent the semantics of Lingua Franca and introduce nondeterminism and illegal memory accesses. For example, it is easy for a programmer to mistakenly send a message that is a pointer to data on the stack. The destination reactors will very likely read invalid data. It is also easy to create memory leaks, where memory is allocated and never freed. Note, however, that the C++ reactor library is designed to prevent common errors and to encourage a safe modern C++ style. Here, we introduce the specifics of writing Reactor programs in C++ and present some guidelines for a style that will be safe.

    \n
    \n
    \n

    In the Python reactor target for Lingua Franca, reactions are written in Python. The user-written reactors are then generated into a Python 3 script that can be executed on several platforms. The Python target has been tested on Linux, macOS, and Windows. To facilitate efficient and fast execution of Python code, the generated program relies on a C extension to facilitate Lingua Franca API such as set and schedule. To learn more about the structure of the generated Python program see Implementation Details.

    \n

    Python reactors can bring the vast library of scientific modules that exist for Python into a Lingua Franca program. Moreover, since the Python reactor target is based on a fast and efficient C runtime library, Lingua Franca programs can execute much faster than native equivalent Python programs in many cases. Finally, interoperability with C reactors is planned for the future.

    \n

    In comparison to the C target, the Python target can be up to an order of magnitude slower. However, depending on the type of application and the implementation optimizations in Python, you can achieve an on-par performance to the C target in many applications.

    \n

    NOTE: A Python C\nextension is\ngenerated for each Lingua Franca program (see Implementation\nDetails). This extension module will\nhave the name LinguaFranca[your_LF_program_name].

    \n
    \n
    \n

    In the TypeScript reactor target for Lingua Franca, reactions are written in TypeScript and the code generator generates a standalone TypeScript program that can be compiled to JavaScript and run on Node.js.

    \n

    TypeScript reactors bring the strengths of TypeScript and Node.js to Lingua Franca programming. The TypeScript language and its associated tools enable static type checking for both reaction code and Lingua Franca elements like ports and actions. The Node.js JavaScript runtime provides an execution environment for asynchronous network applications. With Node.js comes Node Package Manager (npm) and its large library of supporting modules.

    \n

    In terms of raw performance on CPU intensive operations, TypeScript reactors are about two orders of magnitude slower than C reactors. But excelling at CPU intensive operations isn’t really the point of Node.js (or by extension TypeScript reactors). Node.js is about achieving high throughput on network applications by efficiently handling asynchronous I/O operations. Keep this in mind when choosing the right Lingua Franca target for your application.

    \n
    \n
    \n

    Important: The Rust target is still quite preliminary. This is early WIP documentation to let you try it out if you’re curious

    \n

    In the Rust reactor target for Lingua Franca, reactions are written in Rust and the code generator generates a standalone Rust program that can be compiled and run on platforms supported by rustc. The program depends on a runtime library distributed as the crate reactor_rt, and depends on the Rust standard library.

    \n

    Documentation for the runtime API is available here: https://lf-lang.github.io/reactor-rust/

    \n

    LF-Rust generates a Cargo project per compiled main reactor. This specification assumes in some places that the user is somewhat familiar with how Cargo works.\nIf you’re not, here’s a primer:

    \n
      \n
    • a Rust project (and its library artifact) are called a crate.
    • \n
    • Cargo is the Rust package manager and build tool. LF/Rust uses Cargo to build the generated project.
    • \n
    • Rust has extensive support for conditional compilation. Cargo features are commonly used to enable or disable the compilation of parts of a crate. A feature may also pull in additional dependencies. Cargo features only influence the compilation process; if you don’t mention the correct feature flags at compilation time, those features cannot be made available at runtime. The Rust reactor runtime crate uses Cargo features to conditionally enable some features, e.g., command-line argument parsing.
    • \n
    \n
    \n

    Requirements

    \n
    \n

    The following tools are required in order to compile the generated C source code:

    \n
      \n
    • A C compiler such as gcc
    • \n
    • A recent version of cmake (at least 3.5)
    • \n
    \n
    \n
    \n

    The following tools are required in order to compile the generated C++ source code:

    \n
      \n
    • A recent C++ compiler supporting C++17
    • \n
    • A recent version of cmake (at least 3.5)
    • \n
    \n
    \n
    \n

    To use this target, install Python 3 on your machine. See downloading Python.

    \n

    NOTE: The Python target requires a C implementation of Python (nicknamed CPython). This is what you will get if you use the above link, or with most of the alternative Python installations such as Anaconda. See Python download alternatives for more details.

    \n

    The Python reactor target relies on setuptools to be able to compile a Python\nC extension for each LF\nprogram.

    \n\n

    To install setuptools using pip3, do this:

    \n
    pip3 install setuptools\n
    \n
    \n
    \n

    First, make sure Node.js is installed on your machine. You can download Node.js here. The npm package manager comes along with Node.

    \n

    After installing Node, you may optionally install the TypeScript compiler.

    \n
    npm install -g typescript\n
    \n

    TypeScript reactor projects are created with a local copy of the TypeScript compiler, but having the TypeScript compiler globally installed can be useful for debugging type errors and type checking on the command line.

    \n
    \n
    \n

    In order to compile the generated Rust source code, you need a recent version of Cargo, the Rust package manager. See How to Install Rust and Cargo if you don’t have them on your system.

    \n

    You can use a development version of the runtime library by setting the LFC option --external-runtime-path to the root directory of the runtime library crate sources. If this variable is mentioned, LFC will ask Cargo to fetch the runtime library from there.

    \n
    \n

    Limitations

    \n
    \n
      \n
    • The C target does make any distinction between $private$ and $public$ $preamble$.
    • \n
    \n
    \n
    \n

    The C++ target does not yet implement:

    \n\n
    \n
    \n
      \n
    • The Lingua Franca lexer does not support single-quoted strings in Python. This limitation also applies to target property values. You must use double quotes.
    • \n
    \n
    \n
    \n
      \n
    • \n

      The $federated$ implementation in the TypeScript target is still quite preliminary.

      \n
    • \n
    • \n

      The TypeScript target does not yet implement methods.

      \n
    • \n
    • \n

      The TypeScript target does not yet implement modal reactors

      \n
    • \n
    \n
    \n
    \n

    The Rust target does not yet implement:

    \n\n
    \n

    The Target Specification

    \n
    \n

    To have Lingua Franca generate C code, start your .lf file with one of the following target specifications:

    \n
      target C <options>\n  target CCpp <options>\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in C, you may want to include the final semicolon for uniformity.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n

    The second form, CCpp, is used when you wish to use a C++ compiler to compile\nthe generated code, thereby allowing your C reactors to call C++ code.

    \n\n

    Here is a minimal example of a program written in the CCpp target, taken from HelloWorldCCPP.lf:

    \n
    target CCpp\nreactor HelloWorld {\n  preamble {=\n    #include <iostream> // Note that no C++ header will be included by default.\n  =}\n  reaction(startup) {=\n    std::cout << "Hello World." << std::endl;\n  =}\n}\nmain reactor {\n  a = new HelloWorld()\n}\n
    \n

    Note: Unless some feature in the C target is needed, we recommend using the Cpp target that uses a runtime that is written natively in C++.

    \n

    Note: A .lf file that uses the CCpp target cannot and should not be imported to a .lf file that uses the C target. Although these two targets use essentially the same runtime, such a scenario can cause unintended compile errors.

    \n
    \n
    \n

    To have Lingua Franca generate C++ code, start your .lf file with the following target specification:

    \n
        target Cpp\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in C++, you may want to include the final semicolon for uniformity.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n
    \n
    \n

    To have Lingua Franca generate Python code, start your .lf file with the following target specification:

    \n
        target Python\n
    \n

    Note that for all LF statements, a final semicolon is optional.

    \n

    For options to the target specification, see detailed documentation of the target options.

    \n
    \n
    \n

    To have Lingua Franca generate TypeScript code, start your .lf file with the following target specification:

    \n
        target TypeScript\n
    \n

    Note that for all LF statements, a final semicolon is optional.

    \n

    The supported target parameters and command-line options are documented in the Target Declaration documentation.

    \n
    \n
    \n

    To have Lingua Franca generate Rust code, start your .lf file with the following target specification:

    \n
        target Rust\n
    \n

    Note that for all LF statements, a final semicolon is optional. If you are writing your code in Rust, you may want to include the final semicolon for uniformity.

    \n
    \n

    Parameters and State Variables

    \n
    \n

    Reactor parameters and state variables are referenced in the C code using the\nself struct. The following\nStride\nexample modifies the Count reactor in State\nDeclaration to\ninclude both a parameter and a state variable:

    \n
    reactor Count(stride: int = 1) {\n  state count: int = 1\n  output y: int\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    lf_set(y, self->count);\n    self->count += self->stride;\n  =}\n}\n
    \n

    This defines a stride parameter with type int and initial value 1 and\na count state variable with the same type and initial value.\nThese are referenced in the reaction with the syntax self->stride and self->count respectively.

    \n

    The self Struct:\nThe code generator synthesizes a struct type in C for each reactor class and a constructor that creates an instance of this struct. By convention, these instances are called self and are visible within each reactor body. The self struct contains the parameters, state variables, and values associated with actions and ports of the reactor. Parameters and state variables are accessed directly on the self struct, whereas ports and actions are directly in scope by name, as we will see below. Let’s begin with parameters.

    \n

    It may be tempting to declare state variables in the $preamble$, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    int count = 0;\n  =}\n  output y: int\n  timer t(0, 100 msec)\n\n  reaction(t) -> y {=\n    lf_set(y, count++);\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance of the reactor, those instances will share the same variable count. Hence, don’t do this! Sharing variables across instances of reactors violates a basic principle, which is that reactors communicate only by sending messages to one another. Sharing variables will make your program nondeterministic. If you have multiple instances of the above FlawedCount reactor, the outputs produced by each instance will not be predictable, and in a multithreaded implementation, will also not be repeatable.

    \n

    Array Values for Parameters

    \n

    Parameters and state variables can have array values, though some care is needed. The ArrayAsParameter example outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence: int[] = {0, 1, 2}, n_sequence: int = 3) {\n  output out: int\n  state count: int = 0\n  logical action next\n\n  reaction(startup, next) -> out, next {=\n    lf_set(out, self->sequence[self->count]);\n    self->count++;\n    if (self->count < self->n_sequence) {\n      lf_schedule(next, 0);\n    }\n  =}\n}\n
    \n

    This uses a $logical$ $action$ to repeat the reaction, sending one element of the array in each invocation.

    \n

    In C, arrays do not encode their own length, so a separate parameter n_sequence is used for the array length. Obviously, there is potential here for errors, where the array length doesn’t match the length parameter.

    \n

    Above, the parameter default value is an array with three elements, [0, 1, 2]. The syntax for giving this default value is that of a Lingua Franca list, {0, 1, 2}, which gets converted by the code generator into a C static initializer. The default value can be overridden when instantiating the reactor using a similar syntax:

    \n
      s = new Source(sequence = {1, 2, 3, 4}, n_sequence=4)\n
    \n

    Array Values for States

    \n

    A state variable can also have an array value. For example, the MovingAverage reactor computes the moving average of the last four inputs each time it receives an input:

    \n
    reactor MovingAverageImpl {\n  state delay_line: double[] = {0.0, 0.0, 0.0}\n  state index: int = 0\n  input in: double\n  output out: double\n\n  reaction(in) -> out {=\n    // Calculate the output.\n    double sum = in->value;\n    for (int i = 0; i < 3; i++) {\n      sum += self->delay_line[i];\n    }\n    lf_set(out, sum/4.0);\n\n    // Insert the input in the delay line.\n    self->delay_line[self->index] = in->value;\n\n    // Update the index for the next input.\n    self->index++;\n    if (self->index >= 3) {\n      self->index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an array of doubles with the initial value of the array being a three-element array filled with zeros.

    \n

    States and Parameters with Struct Values

    \n

    States whose type are structs can similarly be initialized. This StructAsState example illustrates this:

    \n
    target C\npreamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nmain reactor StructAsState {\n  state s: hello_t = {"Earth", 42}\n  reaction(startup) {=\n    printf("State s.name=\\"%s\\", value=%d.\\n", self->s.name, self->s.value);\n  =}\n}\n
    \n

    Notice that state s is given type hello_t, which is defined in the $preamble$. The initial value just lists the initial values of each of the fields of the struct in the order they are declared.

    \n

    Parameters are similar:

    \n
    target C\npreamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nmain reactor StructParameter(p: hello_t = {"Earth", 42}) {\n  reaction(startup) {=\n    printf("Parameter p.name=\\"%s\\", value=%d.\\n", self->p.name, self->p.value);\n  =}\n}\n
    \n
    \n
    \n

    Reactor parameters are internally declared as const by the code generator and initialized during reactor instantiation. Thus, the value of a parameter may not be changed. See Parameters and State for examples.

    \n

    Array-Valued Parameters

    \n

    Also parameters can have fixed- or variable-sized array values. The ArrayAsParameter example outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence: std::vector<int> = {0, 1, 2}) {\n  output out: size_t\n  state count: size_t = 0\n  logical action next: void\n\n  reaction(startup, next) -> out, next {=\n    out.set(sequence[count]);\n    count++;\n    if (count < sequence.size()) {\n      next.schedule();\n    }\n  =}\n}\n
    \n

    Here, the type of sequence is explicitly given as std::vector<int>.\nA more compact alternative syntax is as follows:

    \n
    sequence: int[] = {0, 1, 2}
    \n

    The type int[] is converted to std::vector<int> by the code generator.\nAnother alternative syntax is:

    \n
    sequence: int[]({0, 1, 2})
    \n

    Here, the static initializer {0, 1, 2} is passed as a single argument to the constructor of std::vector.

    \n

    The main reactor can be parameterized:

    \n
    main reactor Hello(msg: std::string("World")) {\n  reaction(startup) {=\n    std::cout << "Hello " << msg << "!\\n";\n  =}\n}\n
    \n

    This program will print “Hello World!” by default. However, since msg is a main reactor parameter, the C++ code generator will extend the command-line argument parser and allow to override msg when invoking the program. For instance,

    \n
    bin/Hello --msg Earth\n
    \n

    will result in “Hello Earth!” being printed.

    \n

    State Variables

    \n

    A reactor may declare state variables, which become properties of each instance of the reactor. For example, the following reactor (see Count.lf and CountTest.lf) will produce the output sequence 1, 2, 3, … :

    \n
    reactor Count {\n  state count: int = 0\n  output c: int\n  timer t(0, 1 s)\n  reaction(t) -> c {=\n    count++;\n    c.set(count);\n  =}\n}\n
    \n

    The declaration on the second line gives the variable the name count, declares its type to be int, and initializes its value to 0. The type and initial value can be enclosed in the C++-code delimiters {= ... =} if they are not simple identifiers, but in this case, that is not necessary.

    \n

    In the body of the reaction, the state variable is automatically in scope and can be referenced directly by its name. Since all reactions, state variables, and parameters of a reactor are members of the same class, reactions can also reference state variables (or parameters) using the this pointer: this->name.

    \n

    A state variable may be a time value, declared as follows:

    \n
      state time_value:time = 100 ms;\n
    \n

    The type of the generated time_value variable will be reactor::Duration, which is an alias for std::chrono::nanoseconds.

    \n

    For the C++ target, Lingua Franca provides two alternative styles for initializing state variables. We can write state foo:int(42) or state foo:int{42}. This allows to distinguish between the different initialization styles in C++. foo:int(42) will be translated to int foo(42) and foo:int{42} will be translated to int foo{42} in the generated code. Generally speaking, the {...} style should be preferred in C++, but it is not always applicable. Hence we allow the LF programmer to choose the style. Due to the peculiarities of C++, this is particularly important for more complex data types. For instance, state foo:std::vector<int>(4,2) would be initialized to the list [2,2,2,2] whereas state foo:std::vector<int>{4,2} would be initialized to the list [4,2].

    \n

    State variables can have array values. For example, the [MovingAverage] (https://github.com/lf-lang/lingua-franca/blob/master/test/Cpp/src/MovingAverage.lf) reactor computes the moving average of the last four inputs each time it receives an input:

    \n
    reactor MovingAverageImpl {\n  state delay_line: double[3]{0.0, 0.0, 0.0}\n  state index: int = 0\n  input in: double\n  output out: double\n\n  reaction(in) -> out {=\n    // Calculate the output.\n    double sum = *in.get();\n    for (int i = 0; i < 3; i++) {\n      sum += delay_line[i];\n    }\n    out.set(sum/4.0);\n\n    // Insert the input in the delay line.\n    delay_line[index] = *in.get();\n\n    // Update the index for the next input.\n    index++;\n    if (index >= 3) {\n      index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an fixed-size array of 3 doubles with the initial value of the being filled with zeros (note the curly braces). If the size is given in the type specification, then the code generator will declare the type of the state variable using std::array. In the example above, the type of delay_line is std::array<3, double>. If the size specifier is omitted (e.g. state x:double[]). The code generator will produce a variable-sized array using std::vector.

    \n

    The second line can equivalently be given with an assignment operator:

    \n
      state delay_line: double[3] = {0.0, 0.0, 0.0}
    \n

    State variables with more complex types such as classes or structs can be similarly initialized. See StructAsState.lf.

    \n
    \n
    \n

    Reactor parameters and state variables are referenced in the Python code using\nthe syntax self.<name>, where <name> is the name of the parameter or state\nvariable. The following\nStride\nexample modifies the Count reactor in State\nDeclaration to\ninclude both a parameter and a state variable:

    \n
    reactor Count(stride=1) {\n  state count = 1\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(self.count)\n    self.count += self.stride\n  =}\n}\n
    \n

    This defines a stride parameter with initial value 1 and a count state\nvariable with the same initial value. These are referenced in the reaction with\nthe syntax self.stride and self.count respectively. Note that state\nvariables and parameters do not have types in the Python reactor target. See Parameters\nand State for more examples.

    \n

    The Reactor Class:\nThe code generator synthesizes a class in Python for each reactor class in LF,\nwith a constructor (i.e., def __init__(self, ...):) that creates an instance\nof this class and initializes its parameters and state variables as instance\nvariables.\nThese parameters and state variables can then subsequently be accessed directly\nusing the self reference in the body of reactions.

    \n

    It may be tempting to declare state variables in the $preamble$, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    count = 0\n  =}\n  output y\n  timer t(0, 100 msec)\n  reaction(t) -> y {=\n    y.set(count)\n    count += 1\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance\nof the reactor, those instances will share the same variable count (because\ncount will be a class variable). Hence,\ndon’t do this! Sharing variables across instances of reactors violates a\nbasic principle, which is that reactors communicate only by sending messages to\none another. Sharing variables will make your program nondeterministic. If you\nhave multiple instances of the above FlawedCount reactor, the outputs produced\nby each instance will not be predictable, and in a multithreaded implementation,\nwill also not be repeatable.

    \n

    Array Expressions for State Variables and Parameters

    \n

    Array parameters and state variables are implemented using Python lists and initialized using a parentheized list. In the following example, the\nparameter sequence and the state variable x have an initial value that is a Python list [1, 2, 3]:

    \n
    reactor Foo(param = {= [1, 2, 3] =}) {\n  state x = {= [1, 2, 3] =}\n  ...\n}\n
    \n

    Their elements may be accessed as arrays in the body of a reaction, for example self.x[i], where i is an array index.

    \n

    The parameter may be overridden with a different list at instantiation:

    \n
    main reactor {\n  f = new Foo(param = {= [3, 4, 5, 6]} )\n}\n
    \n

    As with any ordinary Python list or tuple, len() can been used to deduce the\nlength. In the above, len(self.x) and len(self.param) will return the lengths of the two lists.

    \n

    Assigning Arbitrary Initial Expressions to State Variables and Parameters

    \n

    As used for lists above, the code delimiters {= ... =} can allow for assignment of arbitrary Python\nexpressions as initial values for state variables and parameters. The following example, taken from\nStructAsState.lf\ndemonstrates this usage:

    \n
    main reactor StructAsState {\n  preamble {=\n    class hello:\n      def __init__(self, name, value):\n        self.name = name\n        self.value = value\n  =}\n  state s = {= self.hello("Earth", 42) =}\n\n  reaction(startup) {=\n    # will print "State s.name="Earth", value=42."\n    print("State s.name=\\"{:s}\\", value={:d}.".format(self.s.name, self.s.value))\n  =}\n}\n
    \n

    Notice that a class hello is defined in the preamble. The state variable s is then initialized to an instance of hello constructed within the {= ... =} delimiters.

    \n
    \n
    \n

    In the TypeScript target, all TypeScript types are generally acceptable for parameters and state variables. Custom types (and classes) must be defined in the $preamble$ before they may be used.

    \n

    To benefit from type checking, you should declare types for your reactor elements. If a type isn’t declared for a state variable, it is assigned the type unknown.

    \n

    For example, the following reactor will produce the output sequence 0, 1, 2, 3, … :

    \n
    reactor Count {\n  state count:number = 0;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    count++;\n    y = count;\n  =}\n}\n
    \n

    The declaration on the second line gives the variable the name “count”, declares its type to be number, and initializes its value to 0. The type and initial value can be enclosed in the Typescript-code delimiters {= ... =} if they are not simple identifiers, but in this case, that is not necessary.

    \n

    In the body of the reaction, the reactor’s state variable is referenced by way of a local variable of the same name. The local variable will contain the current value of the state at the beginning of the reaction. The final value of the local variable will be used to update the state at the end of the reaction.

    \n

    It may be tempting to declare state variables in the preamble, as follows:

    \n
    reactor FlawedCount {\n  preamble {=\n    let count = 0;\n  =}\n  output y:number;\n  timer t(0, 100 msec);\n  reaction(t) -> y {=\n    count++;\n    y = count;\n  =}\n}\n
    \n

    This will produce a sequence of integers, but if there is more than one instance of the reactor, those instances will share the same variable count. Hence, don’t do this! Sharing variables across instances of reactors violates a basic principle, which is that reactors communicate only by sending messages to one another. Sharing variables will make your program nondeterministic. If you have multiple instances of the above FlawedCount reactor, the outputs produced by each instance will not be predictable, and in an asynchronous implementation, will also not be repeatable.

    \n

    A state variable may be a time value, declared as follows:

    \n
      state time_value:time = 100 ms\n
    \n

    The time_value variable will be of type TimeValue, which is an object used to represent a time in the TypeScript Target. Refer to the section on timed behavior for more information.

    \n

    A state variable can have an array or object value. For example, the following reactor computes the moving average of the last four inputs each time it receives an input (from MovingAverageImpl):

    \n
    reactor MovingAverage {\n  state delay_line: {= Array<number> =} = {= [0.0, 0.0, 0.0] =}\n  state index: number = 0\n  input x: number\n  output out: number\n\n  reaction(x) -> out {=\n    x = x as number;\n    // Calculate the output.\n    let sum = x;\n    for (let i = 0; i < 3; i++) {\n      sum += delay_line[i];\n    }\n    out = sum/4.0;\n\n    // Insert the input in the delay line.\n    delay_line[index] = x;\n\n    // Update the index for the next input.\n    index++;\n    if (index >= 3) {\n      index = 0;\n    }\n  =}\n}\n
    \n

    The second line declares that the type of the state variable is an array of numbers with the initial value of the array being a three-element array filled with zeros.

    \n

    States whose type are objects can similarly be initialized. Declarations can take an object literal as the initial value:

    \n
    state myLiteral:{= {foo: number, bar: string} =} = {= {foo: 42, bar: "baz"} =};\n
    \n

    or use new:

    \n
    state mySet:{=Set<number>=} = {= new Set<number>() =};\n
    \n

    Reactor parameters are also referenced in the TypeScript code as local variables. The example below modifies the Count reactor so that its stride is a parameter:

    \n
    target TypeScript\nreactor Count(stride:number = 1) {\n  state count:number = 0;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    y = count;\n    count += stride;\n  =}\n}\nreactor Display {\n  input x:number;\n  reaction(x) {=\n    console.log("Received: " + x + ".");\n  =}\n}\nmain reactor Stride {\n  c = new Count(stride = 2);\n  d = new Display();\n  c.y -> d.x;\n}\n
    \n

    The second line defines the stride parameter, gives its type, and gives its initial value. As with state variables, the type and initial value can be enclosed in {= ... =} if necessary. The parameter is referenced in the reaction by referring to the local variable stride.

    \n

    When the reactor is instantiated, the default parameter value can be overridden. This is done in the above example near the bottom with the line:

    \n
      c = new Count(stride = 2);\n
    \n

    If there is more than one parameter, use a comma separated list of assignments.

    \n

    Parameters in Lingua Franca are immutable. To encourage correct usage, parameter variables within a reaction are local const variables. If you feel tempted to use a mutable parameter, instead try using the parameter to initialize state and modify the state variable instead. This is illustrated below by a further modification to the Stride example where it takes an initial “start” value for count as a second parameter:

    \n
    target TypeScript\nreactor Count(stride:number = 1, start:number = 5) {\n  state count:number = start;\n  output y:number;\n  timer t(0, 100 ms);\n  reaction(t) -> y {=\n    y = count;\n    count += stride;\n  =}\n}\nreactor Display {\n  input x:number;\n  reaction(x) {=\n    console.log("Received: " + x + ".");\n  =}\n}\nmain reactor Stride {\n  c = new Count(stride = 2, start = 10);\n  d = new Display();\n  c.y -> d.x;\n}\n
    \n

    Array or Object Parameters

    \n

    Parameters can have array or object values. Here is an example that outputs the elements of an array as a sequence of individual messages:

    \n
    reactor Source(sequence:{=Array<number>=} = {= [0, 1, 2] =}) {\n  output out:number;\n  state count:number(0);\n  logical action next;\n  reaction(startup, next) -> out, next {=\n    out = sequence[count];\n    count++;\n    if (count < sequence.length) {\n      actions.next.schedule(0, null);\n    }\n  =}\n}\n
    \n

    Above, the parameter default value is an array with three elements, [0, 1, 2]. The syntax for giving this default value is a TypeScript array literal. Since this is TypeScript syntax, not Lingua Franca syntax, the initial value needs to be surrounded with the target code delimiters, {= ... =}. The default value can be overridden when instantiating the reactor using a similar syntax:

    \n
      s = new Source(sequence = {= [1, 2, 3, 4] =});\n
    \n

    Both default and overridden values for parameters can also be created with the new keyword:

    \n
    reactor Source(sequence:{=Array<number>=} = {= new Array<number>() =}) {\n
    \n

    and

    \n
    s = new Source(sequence = {= new Array<number() =});\n
    \n
    \n
    \n

    Parameters and state variables in Rust are accessed on the self structure, as shown in Parameter Declaration.

    \n
    \n

    Inputs and Outputs

    \n
    \n

    In the body of a reaction in the C target, the value of an input is obtained using the syntax name->value, where name is the name of the input port. See, for example, the Destination reactor in Input and Output Declarations.

    \n

    To set the value of outputs, use lf_set. See, for example, the Double reactor in Input and Output Declarations.)

    \n

    An output may even be set in different reactions of the same reactor at the same tag. In this case, one reaction may wish to test whether the previously invoked reaction has set the output. It can check name->is_present to determine whether the output has been set. For example, the Source reactor in the test case TestForPreviousOutput will always produce the output 42:

    \n
    reactor Source {\n  output out: int\n  reaction(startup) -> out {=\n    // Set a seed for random number generation based on the current time.\n    srand(time(0));\n    // Randomly produce an output or not.\n    if (rand() % 2) {\n      lf_set(out, 21);\n    }\n  =}\n  reaction(startup) -> out {=\n    if (out->is_present) {\n      lf_set(out, 2 * out->value);\n    } else {\n      lf_set(out, 42);\n    }\n  =}\n}\n
    \n

    The first reaction may or may not set the output to 21. The second reaction doubles the output if it has been previously produced and otherwise produces 42.

    \n

    Sending and Receiving Data

    \n

    You can define your own data types in C and send and receive those. Consider the StructAsType example:

    \n
    preamble {=\n  typedef struct hello_t {\n    char* name;\n    int value;\n  } hello_t;\n=}\nreactor StructAsType {\n  output out:hello_t;\n  reaction(startup) -> out {=\n    struct hello_t temp = {"Earth", 42};\n    lf_set(out, temp);\n  =}\n}\n
    \n

    The $preamble$ code defines a struct data type. In the reaction to $startup$, the reactor creates an instance of this struct on the stack (as a local variable named temp) and then copies that struct to the output using the lf_set macro.

    \n

    For large structs, it may be inefficient to create a struct on the stack and copy it to the output, as done above. You can use a pointer type instead. See below for details.

    \n

    A reactor receiving the struct message uses the struct as normal in C:

    \n
    reactor Print() {\n  input in:hello_t;\n  reaction(in) {=\n    printf("Received: name = %s, value = %d\\n", in->value.name, in->value.value);\n  =}\n}\n
    \n

    The preamble should not be repeated in this reactor definition if the two reactors are defined together because this will trigger an error when the compiler thinks that hello_t is being redefined.

    \n

    Persistent Inputs

    \n

    In the C target, inputs are persistent. You can read an input even when there is no event present and the value of that input will be the most recently received value or an instance of the input type filled with zeros. For example:

    \n
    target C\nreactor Source {\n  output out: int\n  timer t(100 ms, 200 ms)\n  state count: int = 1\n  reaction(t) -> out {=\n    lf_set(out, self->count++);\n  =}\n}\nreactor Sink {\n  input in: int\n  timer t(0, 100 ms)\n  reaction(t) in {=\n    printf("Value of the input is %d at time %lld\\n", in->value, lf_time_logical_elapsed());\n  =}\n}\nmain reactor {\n  source = new Source()\n  sink = new Sink()\n  source.out -> sink.in\n}\n
    \n

    The Source reactor produces output 1 at 100ms and 2 at 300ms.\nThe Sink reactor reads every 100ms starting at 0.\nNotice that it uses the input in but is not triggered by it.\nThe result of running this program is:

    \n
    Value of the input is 0 at time 0\nValue of the input is 1 at time 100000000\nValue of the input is 1 at time 200000000\nValue of the input is 2 at time 300000000\nValue of the input is 2 at time 400000000\n...
    \n

    The first output is 0 (an int initialized with zero), and subsequently, each output is read twice.

    \n

    Fixed Length Array Inputs and Outputs

    \n

    When inputs and outputs are fixed-length arrays, the memory to contain the array is automatically provided as part of the reactor instance. You can write directly to it, and then just call lf_set_present to alert the system that the output is present. For example:

    \n
    reactor Source {\n  output out: int[3]\n  reaction(startup) -> out {=\n    out->value[0] = 0;\n    out->value[1] = 1;\n    out->value[2] = 2;\n    lf_set_present(out);\n  =}\n}\n
    \n

    In general, this will work for any data type that can be copied by a simple assignment operator (see below for how to handle more complex data types).

    \n

    Reading the array is equally simple:

    \n
    reactor Print(scale: int(1)) {  // The scale parameter is just for testing.\n  input in: int[3]\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < 3; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value[i]);\n    }\n   printf("]\\n");\n  =}\n}\n
    \n

    Variable Length Array Inputs and Outputs

    \n

    Above, the array size is fixed and must be known throughout the program. A more flexible mechanism leaves the array size unspecified in the types of the inputs and outputs and uses lf_set_array instead of lf_set to inform the system of the array length. For example,

    \n
    reactor Source {\n  output out: int[]\n  reaction(startup) -> out {=\n    // Dynamically allocate an output array of length 3.\n    int* array = (int*)malloc(3 * sizeof(int));\n    // Populate the array.\n    array[0] = 0;\n    array[1] = 1;\n    array[2] = 2;\n    // Set the output, specifying the array length.\n    lf_set_array(out, array, 3);\n  =}\n}\n
    \n

    The array length will be available at the receiving end, which may look like this:

    \n
    reactor Print {\n  input in: int[]\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < in->length; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value[i]);\n    }\n    printf("]\\n");\n  =}\n}\n
    \n

    Dynamically Allocated Data

    \n

    A much more flexible way to communicate complex data types is to set dynamically allocated memory on an output port. This can be done in a way that automatically handles freeing the memory when all users of the data are done with it. The reactor that allocates the memory cannot know when downstream reactors are done with the data, so Lingua Franca provides utilities for managing this using reference counting. You can specify a destructor on a port and pass a pointer to a dynamically allocated object as illustrated in the SetDestructor example.

    \n

    Suppose the data structure of interest, its constructor, destructor, and copy_constructor are defined as follows:

    \n
    preamble {=\n  typedef struct int_array_t {\n    int* data;\n    size_t length;\n  } int_array_t;\n\n  int_array_t* int_array_constructor(size_t length) {\n    int_array_t* result = (int_array_t*) malloc(sizeof(int_array_t));\n    result->data = (int*) calloc(length, sizeof(int));\n    result->length = length;\n    return result;\n  }\n\n  void int_array_destructor(void* array) {\n    free(((int_array_t*) array)->data);\n    free(array);\n  }\n\n  void* int_array_copy_constructor(void* array) {\n    int_array_t* source = (int_array_t*) array;\n    int_array_t* copy = (int_array_t*) malloc(sizeof(int_array_t));\n    copy->data = (int*) calloc(source->length, sizeof(int));\n    copy->length = source->length;\n    for (size_t i = 0; i < source->length; i++) {\n      copy->data[i] = source->data[i];\n    }\n    return (void*) copy;\n  }\n=}\n
    \n

    Then, the sender reactor would use lf_set_destructor to specify how the memory set on an output port should be freed:

    \n
    reactor Source {\n  output out:int_array_t*;\n  reaction(startup) -> out {=\n    lf_set_destructor(out, int_array_destructor);\n    lf_set_copy_constructor(out, int_array_copy_constructor);\n  }\n  reaction(startup) -> out {=\n    int_array_t* array =  int_array_constructor(2);\n    for (size_t i = 0; i < array->length; i++) {\n      array->data[i] = i;\n    }\n    lf_set(out, array);\n  =}\n}\n
    \n

    The first reaction specifies the destructor and copy constructor (the latter of which will be used if any downstream reactor has a mutable input or wishes to make a writable copy).

    \n

    IMPORTANT: The array constructed should be sent to only one output port using lf_set. If you need to send it to more than one output port or to use it as the payload of an action, you should use lf_set_token.

    \n

    FIXME: Show how to do this.

    \n

    A reactor receiving this array is straightforward. It just references the array elements as usual in C, as illustrated by this example:

    \n
    reactor Print() {\n  input in:int_array_t*;\n  reaction(in) {=\n    printf("Received: [");\n    for (int i = 0; i < in->value->length; i++) {\n      if (i > 0) printf(", ");\n      printf("%d", in->value->data[i]);\n    }\n    printf("]\\n");\n  =}\n}\n
    \n

    The deallocation of memory for the data will occur automatically after the last reactor that receives a pointer to the data has finished using it, using the destructor specified by lf_set_destructor or free if none specified.

    \n

    Occasionally, you will want an input or output type to be a pointer, but you don’t want the automatic memory allocation and deallocation. A simple example is a string type, which in C is char*. Consider the following (erroneous) reactor:

    \n
    reactor Erroneous {\n  output out:char*;\n  reaction(startup) -> out {=\n    lf_set(out, "Hello World");\n  =}\n}\n
    \n

    An output data type that ends with * signals to Lingua Franca that the message\nis dynamically allocated and must be freed downstream after all recipients are\ndone with it. But the \"Hello World\" string here is statically allocated, so an\nerror will occur when the last downstream reactor to use this message attempts\nto free the allocated memory. To avoid this for strings, you can use a special\nstring type as follows:

    \n
    reactor Fixed {\n  output out:string;\n  reaction(startup) -> out {=\n    lf_set(out, "Hello World");\n  =}\n}\n
    \n

    The string type is equivalent to char*, but since it doesn’t end with *, it does not signal to Lingua Franca that the type is dynamically allocated. Lingua Franca only handles allocation and deallocation for types that are specified literally with a final * in the type name. The same trick can be used for any type where you don’t want automatic allocation and deallocation. E.g., the SendsPointer example looks like this:

    \n
    reactor SendsPointer  {\n  preamble {=\n    typedef int* int_pointer;\n  =}\n  output out:int_pointer\n  reaction(startup) -> out {=\n    static int my_constant = 42;\n    lf_set(out, &my_constant;)\n  =}\n}\n
    \n

    The above technique can be used to abuse the reactor model of computation by communicating pointers to shared variables. This is generally a bad idea unless those shared variables are immutable. The result will likely be nondeterministic. Also, communicating pointers across machines that do not share memory will not work at all.

    \n

    Mutable Inputs

    \n

    Although it cannot be enforced in C, a receiving reactor should not modify the values provided by an input. Inputs are logically immutable because there may be several recipients. Any recipient that wishes to modify the input should make a copy of it. Fortunately, a utility is provided for this pattern. Consider the ArrayScale example, here modified to use the above int_array_t data type:

    \n
    reactor ArrayScale(scale:int(2)) {\n  mutable input in:int_array_t*;\n  output out:int_array_t*;\n  reaction(in) -> out {=\n    for(int i = 0; i < in->length; i++) {\n      in->value[i] *= self->scale;\n    }\n    lf_set_token(out, in->token);\n  =}\n}\n
    \n

    Here, the input is declared $mutable$, which means that any reaction is free to\nmodify the input. If this reactor is the only recipient of the array or the last\nrecipient of the array, then this will not make a copy of the array but rather use\nthe original array. Otherwise, it will use a copy. By default, memcpy is used to copy the data. However, the sender can also specify\na copy constructor to be used by calling lf_set_copy_constructor on the\noutput port, as explained below.

    \n

    Important: Notice that the above ArrayScale reactor modifies the array and then forwards it to its output port using the lf_set_token() macro. That macro further delegates to downstream reactors the responsibility for freeing dynamically allocated memory once all readers have completed their work. It will not work to just use lf_set, passing it the value.\nThis will result in a memory error, yielding a message like the following:

    \n
        malloc: *** error for object 0x600002674070: pointer being freed was not allocated
    \n

    If the above code were not to forward the array, then the dynamically allocated memory will be automatically freed when this reactor is done with it.

    \n

    Three of the above reactors can be combined into a pipeline as follows:

    \n
    main reactor ArrayScaleTest {\n  s = new Source();\n  c = new ArrayScale();\n  p = new Print();\n  s.out -> c.in;\n  c.out -> p.in;\n}\n
    \n

    In this composite, the array is allocated by ArrayPrint, modified by ArrayScale, and deallocated (freed) after Print has reacted. No copy is necessary because ArrayScale is the only recipient of the original array.

    \n

    Inputs and outputs can also be dynamically allocated structs. In fact, Lingua Franca’s C target will treat any input or output data type that ends with [] or * specially by providing utilities for allocating memory and modifying and forwarding. Deallocation of the allocated memory is automatic. The complete set of utilities is given below.

    \n

    String Types

    \n

    String types in C are char*. But, as explained above, types ending with * are interpreted specially to provide automatic memory management, which we generally don’t want with strings (a string that is a compile-time constant must not be freed). You could enclose the type as {= char* =}, but to avoid this awkwardness, the header files include a typedef that permits using string instead of char*. For example (from DelayString.lf):

    \n
    reactor DelayString(delay:time = 100 ms)) {\n  input in:string;\n  output out:string;\n  logical action a:string;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // The following copies the char*, not the string.\n    lf_schedule_copy(a, self->delay, &(in->value), 1);\n  =}\n}\n
    \n

    Macros For Setting Output Values

    \n

    In all of the following, <out> is the name of the output and <value> is the value to be sent.

    \n
    \n

    lf_set(<out>, <value>);

    \n
    \n

    Set the specified output (or input of a contained reactor) to the specified\nvalue using shallow copy. lf_set can be used with all supported data types\n(including type declarations that end with * or []).

    \n
    \n

    lf_set_token(<out>, <token>);

    \n
    \n

    This version is used to directly set the underlying reference-counted token in\noutputs with a type declaration ending with * (any pointer) or [] (any\narray). The <value> argument should be a struct of type token_t. It should\nbe rarely necessary to have the need to create your own (dynamically allocated)\ninstance of token_t.

    \n

    Consider the\nSetToken.lf\nexample:

    \n
    reactor Source {\n  output out:int*\n  logical action a:int\n  reaction(startup) -> a {=\n    lf_schedule_int(a, MSEC(200), 42);\n  =}\n  reaction(a) -> out {=\n    lf_set_token(out, a->token);\n  =}\n}\n
    \n

    Here, the first reaction schedules an integer-valued action to trigger after 200 milliseconds. As explained below, action payloads are carried by tokens. The second reaction grabs the token rather than the value using the syntax a->token (the name of the action followed by ->token). It then forwards the token to the output. The output data type is int* not int because the token carries a pointer to dynamically allocated memory that contains the value. All inputs and outputs with types ending in * or [] are carried by tokens.

    \n
    \n

    lf_set_destructor(<out>, <destructor>);

    \n
    \n

    Specify the destructor destructor used to deallocate any dynamic data set on the output port out.

    \n
    \n

    lf_set_copy_constructor(<out>, <copy_constructor>);

    \n
    \n

    Specify the copy_constructor used to copy construct any dynamic data set on the output port out if the receiving port is $mutable$.

    \n

    lf_set (and lf_set_token) will overwrite any output value previously set at the same logical time and will cause the final output value to be sent to all reactors connected to the output. They also set a local <out>->is_present variable to true. This can be used to subsequently test whether the output value has been set.

    \n
    \n
    \n

    In the body of a reaction in the C++ target, the value of an input is obtained using the syntax *name.get(), where name is the name of the input port. Similarly, outputs are set using a set() method on an output port. For examples, see Inputs and Outputs.

    \n

    Note that get() always returns a pointer to the actual value. Thus the pointer needs to be dereferenced with * to obtain the value. (See Sending and Receiving Large Data Types for an explanation of the exact mechanisms behind this pointer access).\nTo determine whether an input is present, name.is_present() can be used. Since get() returns a nullptr if no value is present, name.get() != nullptr can be used alternatively for checking presence.

    \n

    Sending and Receiving Large Data Types

    \n

    You can define your own data types in C++ or use types defined in a library and send and receive those. Consider the StructAsType example:

    \n
    public preamble {=\n  struct Hello {\n    std::string name;\n    int value;\n  };\n=}\nreactor StructAsType {\n  output out: Hello;\n  reaction(startup) -> out {=\n    Hello hello{"Earth, 42};\n    out.set(hello);\n  =}\n}\n
    \n

    The $public$ $preamble$ code defines a struct data type. In the reaction to $startup$, the reactor creates an instance of this struct on the stack (as a local variable named hello) and then copies that instance to the output using the set() method. For this reason, the C++ reactor runtime provides more sophisticated ways to allocate objects and send them via ports.

    \n

    The C++ library defines two types of smart pointers that the runtime uses internally to implement the exchange of data between ports. These are reactor::MutableValuePtr<T> and reactor::ImmutableValuePtr<T>. reactor::MutableValuePtr<T> is a wrapper around std::unique_ptr and provides read and write access to the value hold, while ensuring that the value has a unique owner. In contrast, reactor::ImmutableValuePtr<T> is a wrapper around std::shared_pointer and provides read only (const) access to the value it holds. This allows data to be shared between reactions of various reactors, while guarantee data consistency. Similar to std::make_unique and std::make_shared, the reactor library provides convenient function for creating mutable and immutable values pointers: reactor::make_mutable_value<T>(...) and reactor::make_immutable_value<T>(...).

    \n

    In fact this code from the example above:

    \n
        Hello hello{"Earth, 42"};\n    out.set(hello);\n
    \n

    implicitly invokes reactor::make_immutable_value<Hello>(hello) and could be rewritten as

    \n
        Hello hello{"Earth, 42"};\n    out.set(reactor::make_immutable_value<Hello>(hello));\n
    \n

    This will invoke the copy constructor of Hello, copying its content from the hello instance to the newly created reactor::ImmutableValuePtr<Hello>.

    \n

    Since copying large objects is inefficient, the move semantics of C++ can be used to move the ownership of object instead of copying it. This can be done in the following two ways. First, by directly creating a mutable or immutable value pointer, where a mutable pointer allows modification of the object after it has been created:

    \n
        auto hello = reactor::make_mutable_value<Hello>("Earth", 42);\n    hello->name = "Mars";\n    out.set(std::move(hello));\n
    \n

    An example of this can be found in StructPrint.lf. Not that after the call to std::move, hello is nullptr and the reaction cannot modify the object anymore. Alternatively, if no modification is requires, the object can be instantiated directly in the call to set() as follows:

    \n
        out.set({"Earth", 42});\n
    \n

    An example of this can be found in StructAsTypeDirect.

    \n

    Getting a value from an input port of type T via get() always returns an reactor::ImmutableValuePtr<T>. This ensures that the value cannot be modified by multiple reactors receiving the same value, as this could lead to an inconsistent state and nondeterminism in a multi-threaded execution. An immutable value pointer can be converted to a mutable pointer by calling get_mutable_copy. For instance, the ArrayScale reactor modifies elements of the array it receives before sending it to the next reactor:

    \n
    reactor Scale(scale:int = 2) {\n  input in:int[3];\n  output out:int[3];\n\n  reaction(in) -> out {=\n    auto array = in.get().get_mutable_copy();\n    for(int i = 0; i < array->size(); i++) {\n      (*array)[i] = (*array)[i] * scale;\n    }\n    out.set(std::move(array));\n  =}\n}\n
    \n

    Currently get_mutable_copy() always copies the contained value to safely create a mutable pointer. However, a future implementation could optimize this by checking if any other reaction is accessing the same value. If not, the value can simply be moved from the immutable pointer to a mutable one.

    \n
    \n
    \n

    In the body of a reaction in the Python target, the value of an input is\nobtained using the syntax name.value, where name is the name of the input\nport. To determine whether an input is present, use name.is_present. To\nproduce an output, use the syntax name.set(value). The value can be any\nvalid Python object. For simple examples, see Inputs and\nOutputs.

    \n

    Sending and Receiving Objects

    \n

    You can define your own data types in Python and send and receive those. Consider the StructAsType example:

    \n
    target Python {\n  files: include/hello.py\n}\npreamble {=\n  import hello\n=}\nreactor Source {\n  output out;\n  reaction(startup) -> out {=\n    temp = hello.hello("Earth", 42)\n    out.set(temp)\n  =}\n}\n
    \n

    The top-level preamble has imported the hello module, which contains the following class:

    \n
    class hello:\n    def __init__(self, name = "", value = 0):\n        self.name = name\n        self.value = value\n
    \n

    In the reaction to startup, the reactor has created an instance object of this class (as local variable named temp) and passed it downstream using the set method on output port out.

    \n

    The set method is defined as follows:

    \n
    \n

    <port>.set(<value>): Set the specified output port (or input of a contained\nreactor) to the specified value. This value can be any Python object\n(including None and objects of type Any). The value is\ncopied and therefore the variable carrying the value can be subsequently\nmodified without changing the output.

    \n
    \n

    A reactor receiving the class object message can subsequently access the object\nusing <port>.value:

    \n
    reactor Print(expected(42)) {\n  input _in;\n  reaction(_in) {=\n    print("Received: name = {:s}, value = {:d}\\n".format(_in.value.name,\n                                                         _in.value.value))\n  =}\n}\n
    \n

    Note: The hello module has been imported using a top-level preamble, therefore, the contents of the module are available to all reactors defined in the current Lingua Franca file (a similar situation arises if the hello class itself was in the top-level preamble).

    \n
    \n
    \n

    In the TypeScript target, all TypeScript types are generally acceptable for inputs and outputs with one notable exception:

    \n
      \n
    • undefined is not a valid type for an input, output, or action. This is because undefined is used to designate the absence of an input, output, or action during a reaction.
    • \n
    \n

    As with parameters and state variables, custom types (and classes) must be defined in the preamble before they may be used.

    \n

    To benefit from type checking, you should declare types for your reactor elements. If a type isn’t declared for an input, output, or action, it is assigned the reactor-ts type Present which is defined as

    \n
    export type Present = (number | string | boolean | symbol | object | null);\n
    \n

    In the body of a reaction in the TypeScript target, inputs are simply referred to by name. An input of type t is available within the body of a reaction as a local variable of type t | undefined. To determine whether an input is present, test the value of the input against undefined. An undefined input is not present.

    \n

    WARNING Be sure to use the === or !== operator and not == or != to test against undefined. In JavaScript/TypeScript the comparison undefined == null yields the value true. It may also be tempting to rely upon the falsy evaluation of undefined within an if statement, but this may introduce bugs. For example a reaction that tests the presence of input x with if (x) { ... } will not correctly identify potentially valid present values such as 0, false, or \"\".

    \n

    For example, the Determinism.lf test case includes the following reactor:

    \n
    reactor Destination {\n  input x: number\n  input y: number\n  reaction(x, y) {=\n    let sum = 0;\n    if (x !== undefined) {\n      sum += x;\n    }\n    if (y !== undefined) {\n      sum += y;\n    }\n    console.log("Received " + sum);\n    if (sum != 2) {\n      util.requestErrorStop("FAILURE: Expected 2.")\n    }\n  =}\n}\n
    \n

    The reaction refers to the inputs x and y by name and tests for their presence by testing x and y against undefined. If a reaction is triggered by just one input, then normally it is not necessary to test for its presence. It will always be present. However TypeScript’s type system is not smart enough to know such an input will never have type undefined if there’s no test against undefined within the reaction. An explicit type annotation (for example x = x as t; where t is the type of the input) may be necessary to avoid type errors from the compiler. In the above example, there are two triggers, so the reaction has no assurance that both will be present.

    \n

    Inputs declared in the uses part of the reaction do not trigger the reaction. Consider this modification of the above reaction:

    \n
    reaction(x) y {=\n  let sum = x as number;\n  if (y !== undefined) {\n    sum += y;\n  }\n  console.log("Received " + sum + ".");\n=}\n
    \n

    It is no longer necessary to test for the presence of x because that is the only trigger. The input y, however, may or may not be present at the logical time that this reaction is triggered. Hence, the code must test for its presence.

    \n

    The effects portion of the reaction specification can include outputs and actions. Actions will be described below. Like inputs, an output of type t is available within the body of a reaction as a local variable of type t | undefined. The local variable for each output is initialized to the output’s current value. Outputs are set by assigning a (non-undefined) value to its local variable (no changes will be made to an output if it has the value undefined at the end of a reaction). Whatever value an output’s local variable has at the end of the reaction will be set to that output. If an output’s local variable has the value undefined at the end of the reaction, that output will not be set and connected downstream inputs will be absent. For example, we can further modify the above example as follows:

    \n
    output z:number;\nreaction(x) y -> z {=\n  let sum = x as number;\n  if (y !== undefined) {\n    sum += y;\n  }\n  z = sum;\n=}\n
    \n

    If an output gets set more than once at any logical time, downstream reactors will see only the final value that is set. Since the order in which reactions of a reactor are invoked at a logical time is deterministic, and whether inputs are present depends only on their timestamps, the final value set for an output will also be deterministic.

    \n

    An output may even be set in different reactions of the same reactor at the same logical time. In this case, one reaction may wish to test whether the previously invoked reaction has set the output. It can do that using a !== undefined test for that output. For example, the following reactor will always produce the output 42:

    \n
    reactor TestForPreviousOutput {\n  output out:number;\n  reaction(startup) -> out {=\n    if (Math.random() > 0.5) {\n      out = 21;\n    }\n  =}\n  reaction(startup) -> out {=\n    let previous_output = out;\n    if (previous_output) {\n      out = 2 * previous_output;\n    } else {\n      out = 42;\n    }\n  =}\n}\n
    \n

    The first reaction may or may not set the output to 21. The second reaction doubles the output if it has been previously produced and otherwise produces 42.

    \n

    Sending and Receiving Custom Types

    \n

    You can define your own data types in TypeScript and send and receive those. Consider the following example:

    \n
    reactor CustomType {\n  preamble {=\n    type custom = string | null;\n  =}\n  output out:custom;\n  reaction(startup) -> out {=\n    out = null;\n  =}\n}\n
    \n

    The $preamble$ code defines a custom union type of string and null.

    \n
    \n
    \n

    Inputs and outputs in the Rust target are accessed using the set and get methods of the ctx objects, as shown in Inputs and Outputs.

    \n
    \n

    Time

    \n
    \n

    In the C target, the value of a time instant or interval is an integer specifying a number of nanoseconds. An instant is the number of nanoseconds that have elapsed since January 1, 1970. An interval is the difference between two instants. When an LF program starts executing, logical time is (normally) set to the instant provided by the operating system. (On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    Time in the C target is a int64_t, which is a 64-bit signed number. Since a 64-bit number has a limited range, this measure of time instants will overflow in approximately the year 2262. For better code clarity, two types are defined in tag.h, instant_t and interval_t, which you can use for time instants and intervals respectively. These are both equivalent to int64_t, but using those types will insulate your code against changes and platform-specific customizations.

    \n

    Lingua Franca uses a superdense model of time. A reaction is invoked at a logical tag, a struct consisting of a time value (an instant_t, which is a int64_t) and a microstep value (a microstep_t, which is an uint32_t). The tag is guaranteed to not increase during the execution of a reaction. Outputs produced by a reaction have the same tag as the inputs, actions, or timers that trigger the reaction, and hence are logically simultaneous.

    \n

    The time structs and functions for working with time are defined in tag.h. The most useful functions are:

    \n
      \n
    • tag_t lf_tag(): Get the current tag at which this reaction has been invoked.
    • \n
    • int lf_tag_compare(tag_t, tag_t): Compare two tags, returning -1, 0, or 1 for less than, equal, and greater than.
    • \n
    • instant_t lf_time_logical(): Get the current logical time (the first part of the current tag).
    • \n
    • interval_t lf_time_logical_elapsed(): Get the logical time elapsed since program start.
    • \n
    \n

    There are also some useful functions for accessing physical time:

    \n
      \n
    • instant_t lf_time_physical(): Get the current physical time.
    • \n
    • instant_t lf_time_physical_elapsed(): Get the physical time elapsed since program start.
    • \n
    • instant_t lf_time_start(): Get the starting physical and logical time.
    • \n
    \n

    The last of these is both a physical and logical time because, at the start of execution, the starting logical time is set equal to the current physical time as measured by a local clock.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime example:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t logical = lf_time_logical();\n    printf("Logical time is %ld.\\n", logical);\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    Start execution at time Sun Oct 13 10:18:36 2019\nplus 353609000 nanoseconds.\nLogical time is 1570987116353609000.\nLogical time is 1570987117353609000.\nLogical time is 1570987118353609000.\n...
    \n

    The first two lines give the current time-of-day provided by the execution platform at the start of execution. This is used to initialize logical time. Subsequent values of logical time are printed out in their raw form, rather than the friendlier form in the first two lines. If you look closely, you will see that each number is one second larger than the previous number, where one second is 1000000000 nanoseconds.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    interval_t elapsed = lf_time_logical_elapsed();\n    printf("Elapsed logical time is %ld.\\n", elapsed);\n  =}\n}\n
    \n

    This will produce:

    \n
    Start execution at time Sun Oct 13 10:25:22 2019\nplus 833273000 nanoseconds.\nElapsed logical time is 0.\nElapsed logical time is 1000000000.\nElapsed logical time is 2000000000.\n...
    \n

    You can also get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t physical = lf_time_physical();\n    printf("Physical time is %ld.\\n", physical);\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Start execution at time Sun Oct 13 10:35:59 2019\nplus 984992000 nanoseconds.\nPhysical time is 1570988159986108000.\nPhysical time is 1570988160990219000.\nPhysical time is 1570988161990067000.\n...
    \n

    Finally, you can get elapsed physical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    instant_t elapsed_physical = lf_time_physical_elapsed();\n    printf("Elapsed physical time is %ld.\\n", elapsed_physical);\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Elapsed physical time is 657000.\nElapsed physical time is 1001856000.\nElapsed physical time is 2004761000.\n...
    \n

    Notice that these numbers are increasing by roughly one second each time. If you set the fast target parameter to true, then logical time will elapse much faster than physical time.

    \n

    Working with nanoseconds in C code can be tedious if you are interested in longer durations. For convenience, a set of macros are available to the C programmer to convert time units into the required nanoseconds. For example, you can specify 200 msec in C code as MSEC(200) or two weeks as WEEKS(2). The provided macros are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these. Examples are given in the next section.

    \n
    \n
    \n

    Timers are specified exactly as in the Time and Timers. When working with time in the C++ code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    The reactor-cpp library uses std::chrono for representing time. Specifically, the library defines two types for representing durations and timepoints: reactor::Duration and reactor::TimePoint. reactor::Duration is an alias for std::chrono::nanosecods. reactor::TimePoint is alias for std::chrono::time_point<std::chrono::system_clock, std::chrono::nanoseconds>. As you can see from these definitions, the smallest time step that can be represented is one nanosecond. Note that reactor::TimePoint describes a specific point in time and is associated with a specific clock, whereas reactor::Duration defines a time interval between two time points.

    \n

    Lingua Franca uses a superdense model of logical time. A reaction is invoked at a logical tag. In the C++ library, a tag is represented by the class reactor::Tag. In essence, this class is a tuple of a reactor::TimePoint representing a specific point in logical time and a microstep value (of type reactor::mstep_t, which is an alias for unsigned long). reactor::Tag provides two methods for getting the time point or the microstep:

    \n
    const TimePoint& time_point() const;\nconst mstep_t& micro_step() const;\n
    \n

    The C++ code in reaction bodies has access to library functions that allow to retrieve the current logical or physical time:

    \n
      \n
    • TimePoint get_physical_time(): Get the current physical time.
    • \n
    • TimePoint get_logcial_time(): Get the current logical time.
    • \n
    • Duration get_elapsed_physical_time(): Get the physical time elapsed since program start.
    • \n
    • Duration get_elapsed_logical_time(): Get the logical time elapsed since program start.
    • \n
    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime example:

    \n
    main reactor {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    auto logical = get_logical_time();\n    std::cout << "Logical time is " << logical << std::endl;\n  =}\n}\n
    \n

    Note that the << operator is overloaded for both reactor::TimePoint and reactor::Duration and will print the time information accordingly.

    \n

    When executing the above program, you will see something like this:

    \n
    [INFO]  Starting the execution\nLogical time is 2021-05-19 14:06:09.496828396\nLogical time is 2021-05-19 14:06:10.496828396\nLogical time is 2021-05-19 14:06:11.496828396\nLogical time is 2021-05-19 14:06:11.496828396\n...
    \n

    If you look closely, you will see that each printed logical time is one second larger than the previous one.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    auto elapsed = get_elapsed_logical_time();\n    std::cout << "Elapsed logical time is " << elapsed << std::endl;\n    std::cout << "In seconds: " <<  std::chrono::duration_cast<std::chrono::seconds>(elapsed) << std::endl;\n  =}\n}\n
    \n

    Using std::chrono it is also possible to convert between time units and directly print the number of elapsed seconds as seen above. The resulting output of this program will be:

    \n
    [INFO]  Starting the execution\nElapsed logical time is 0 nsecs\nIn seconds: 0 secs\nElapsed logical time is 1000000000 nsecs\nIn seconds: 1 secs\nElapsed logical time is 2000000000 nsecs\nIn seconds: 2 secs\n...
    \n

    You can also get physical and elapsed physical time:

    \n
    main reactor {\n  timer t(0, 1 sec);\n\treaction(t) {=\n    auto logical = get_logical_time();\n    auto physical = get_physical_time();\n    auto elapsed = get_elapsed_physical_time();\n    std::cout << "Physical time is " << physical << std::endl;\n    std::cout << "Elapsed physical time is " << elapsed << std::endl;\n    std::cout << "Time lag is " << physical - logical << std::endl;\n  =}\n}\n
    \n

    Notice that the physical times are increasing by roughly one second in each reaction. The output also shows the lag between physical and logical time. If you set the fast target parameter to true, then physical time will elapse much faster than logical time. The above program will produce something like this:

    \n
    [INFO]  Starting the execution\nPhysical time is 2021-05-19 14:25:18.070523014\nElapsed physical time is 2601601 nsecs\nTime lag is 2598229 nsecs\nPhysical time is 2021-05-19 14:25:19.068038275\nElapsed physical time is 1000113888 nsecs\nTime lag is 113490 nsecs\n[INFO]  Physical time is Terminating the execution\n2021-05-19 14:25:20.068153026\nElapsed physical time is 2000228689 nsecs\nTime lag is 228241 nsecs
    \n

    For specifying time durations in code chrono provides convenient literal operators in std::chrono_literals. This namespace is automatically included for all reaction bodies. Thus, we can simply write:

    \n
    std::cout << 42us << std::endl;\nstd::cout << 1ms << std::endl;\nstd::cout << 3s << std::endl;\n
    \n

    which prints:

    \n
    42 usecs\n1 msecs\n3 secs
    \n
    \n
    \n

    Timers are specified exactly as in the Time and Timers. When working with time in the Python code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    In the Python target, similar to the C target, the value of a time instant or\ninterval is an integer specifying a number of nanoseconds. An instant is the\nnumber of nanoseconds that have elapsed since January 1, 1970. An interval is\nthe difference between two instants.

    \n

    The functions for working with time and tags are:

    \n
      \n
    • lf.tag() -> Tag: Returns a Tag instance of the current tag at which this reaction has been invoked.
    • \n
    • lf.tag_compare(Tag, Tag) -> int: Compare two Tag instances, returning -1, 0, or 1 for less than, equal, and greater than. Tags can also be compared using rich comparators (ex. <, >, ==), which returns True or False.
    • \n
    • lf.time.logical() -> int: Get the current logical time (the first part of the current tag).
    • \n
    • lf.time.logical_elapsed() -> int: Get the logical time elapsed since program start.
    • \n
    \n

    Tags can be initialized using Tag(time=some_number, microstep=some_other_number).

    \n

    There are also some useful functions for accessing physical time:

    \n
      \n
    • lf.time.physical() -> int: Get the current physical time.
    • \n
    • lf.time.physical_elapsed() -> int: Get the physical time elapsed since program start.
    • \n
    • lf.time.start() -> int: Get the starting physical and logical time.
    • \n
    \n

    The last of these is both a physical and a logical time because, at the start of execution, the starting logical time is set equal to the current physical time as measured by a local clock.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider the GetTime.lf example:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    logical = lf.time.logical()\n    print("Logical time is ", logical)\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nLogical time is  1604587862864237900\nLogical time is  1604587863864237900\nLogical time is  1604587864864237900\n...
    \n

    The first two lines give the current time-of-day provided by the execution platform at the start of execution. This is used to initialize logical time. Subsequent values of logical time are printed out in their raw form, rather than the friendlier form in the first two lines. If you look closely, you will see that each number is one second larger than the previous number, where one second is 1000000000 nanoseconds.

    \n

    You can also obtain the elapsed logical time since the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    elapsed = lf.time.logical_elapsed()\n    print("Elapsed logical time is ", elapsed)\n  =}\n}\n
    \n

    This will produce:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nElapsed logical time is  0\nElapsed logical time is  1000000000\nElapsed logical time is  2000000000\n...
    \n

    You can also get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    physical = lf.time.physical()\n    print("Physical time is ", physical)\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nPhysical time is  1604587862864343500\nPhysical time is  1604587863864401900\nPhysical time is  1604587864864395200\n...
    \n

    Finally, you can get elapsed physical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    elapsed_physical = lf.time.physical_elapsed()\n    print("Elapsed physical time is ", elapsed_physical)\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    ---- Start execution at time Thu Nov  5 08:51:02 2020\n---- plus 864237900 nanoseconds.\nElapsed physical time is  110200\nElapsed physical time is  1000185400\nElapsed physical time is  2000178600\n...
    \n

    Notice that these numbers are increasing by roughly one second each time. If you set the fast target parameter to true, then logical time will elapse much faster than physical time.

    \n

    Working with nanoseconds in the Python code can be tedious if you are interested in longer durations. For convenience, a set of functions are available to the Python programmer to convert time units into the required nanoseconds. For example, you can specify 200 msec in Python code as MSEC(200) or two weeks as WEEKS(2). The provided functions are NSEC, USEC (for microseconds), MSEC, SEC, MINUTE, HOUR, DAY, and WEEK. You may also use the plural of any of these. Examples are given in the next section.

    \n
    \n
    \n

    See Summary of Time Functions and Utility Function Reference for a quick API reference.

    \n

    Timers are specified exactly as in the Time and Timers section. When working with time in the TypeScript code body of a reaction, however, you will need to know a bit about its internal representation.

    \n

    A TimeValue is an class defined in the TypeScript target library file time.ts to represent a time instant or interval. For your convenience TimeValue and other classes from the time.ts library mentioned in these instructions are automatically imported into scope of your reactions. An instant is the number of nanoseconds that have elapsed since January 1, 1970. An interval is the difference between two instants. When an LF program starts executing, logical time is (normally) set to the instant provided by the operating system. (On some embedded platforms without real-time clocks, it will be set instead to zero.)

    \n

    Internally a TimeValue uses two numbers to represent the time. To prevent overflow (which would occur for time intervals spanning more than 0.29 years if a single JavaScript number, which has 2^53 bits of precision, were to be used), we use two numbers to store a time value. The first number denotes the number of whole seconds in the interval or instant; the second number denotes the remaining number of nanoseconds in the interval or instant. The first number represents the number of seconds, the second number represents the number of nanoseconds. These fields are not accessible to the programmer, instead TimeValues may be manipulated by an API with functions for addition, subtraction, and comparison.

    \n

    A reaction can examine the current logical time (which is constant during the execution of the reaction). For example, consider:

    \n
    target TypeScript;\nmain reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let logical:TimeValue = util.getCurrentLogicalTime()\n    console.log("Logical time is " + logical + ".");\n  =}\n}\n
    \n

    When executed, you will get something like this:

    \n
    Logical time is (1584666585 secs; 805146880 nsecs).\nLogical time is (1584666586 secs; 805146880 nsecs).\nLogical time is (1584666587 secs; 805146880 nsecs).\n...
    \n

    Subsequent values of logical time are printed out in their raw form, of seconds and nanoseconds. If you look closely, you will see that each number is one second larger than the previous number.

    \n

    You can also obtain the elapsed logical time since the start of execution, rather than exact logical time:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let logical:TimeValue = util.getElapsedLogicalTime()\n    console.log("Logical time is " + logical + ".");\n  =}\n}\n
    \n

    This will produce:

    \n
    Logical time is (0 secs; 0 nsecs).\nLogical time is (1 secs; 0 nsecs).\nLogical time is (2 secs; 0 nsecs).\n...
    \n

    You can get physical time, which comes from your platform’s real-time clock:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let physical:TimeValue = util.getCurrentPhysicalTime()\n    console.log("Physical time is " + physical + ".");\n  =}\n}\n
    \n

    This will produce something like this:

    \n
    Physical time is (1584666801 secs; 644171008 nsecs).\nPhysical time is (1584666802 secs; 642269952 nsecs).\nPhysical time is (1584666803 secs; 642278912 nsecs).\n...
    \n

    Notice that these numbers are increasing by roughly one second each time.

    \n

    You can also get elapsed physical time from the start of execution:

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let physical:TimeValue = util.getElapsedPhysicalTime()\n    console.log("Physical time is " + physical + ".");\n  =}\n}\n
    \n

    This will produce something like:

    \n
    Physical time is (0 secs; 2260992 nsecs).\nPhysical time is (1 secs; 166912 nsecs).\nPhysical time is (2 secs; 136960 nsecs).\n...
    \n

    You can create a TimeValue yourself with the UnitBasedTimeValue class. UnitBasedTimeValue is a subclass of TimeValue and can be used wherever you could also use a TimeValue directly obtained from one of the util functions. A UnitBasedTimeValue is constructed with a whole number and a TimeUnit. A TimeUnit is an enum from the time.ts library with convenient labels for common time units. These are nsec, usec, msec, sec (or secs), minute (or minutes), hour (or hours), day (or days), and week (or weeks).

    \n

    This reactor has an example of a UnitBasedTimeValue.

    \n
    main reactor GetTime {\n  timer t(0, 1 sec);\n  reaction(t) {=\n    let myTimeValue:TimeValue = new UnitBasedTimeValue(200, TimeUnit.msec);\n    let logical:TimeValue = util.getCurrentLogicalTime()\n    console.log("My custom time value is " + myTimeValue + ".");\n  =}\n
    \n

    This will produce:

    \n
    My custom time value is 200 msec.\nMy custom time value is 200 msec.\nMy custom time value is 200 msec.\n...
    \n

    Tags

    \n

    The TypeScript target provides a utility to get the current Tag of a reaction. Recall that time in Lingua Franca is superdense and each TimeValue is paired with an integer “microstep” index to track the number of iterations at a particular TimeValue. A Tag is this combination of a TimeValue and a “microstep”. The time.ts library provides functions for adding, subtracting, and comparing Tags.

    \n

    You can get the current Tag in your reactions. This example illustrates tags with a Zero-Delay Action:

    \n
    target TypeScript;\nmain reactor GetTime {\n  timer t(0, 1 sec);\n  logical action a;\n  reaction(t) -> a {=\n    let superdense:Tag = util.getCurrentTag();\n    console.log("First iteration - the tag is: " + superdense + ".");\n    actions.a.schedule(0, null);\n  =}\n  reaction(a) {=\n    let superdense:Tag = util.getCurrentTag();\n    let timePart:TimeValue = superdense.time;\n    let microstepPart:number = superdense.microstep;\n    console.log("Second iteration - the time part of the tag is:  " + timePart + ".");\n    console.log("Second iteration - the microstep part of the tag is:  " + microstepPart + ".");\n  =}\n}\n
    \n

    This will produce:

    \n
    First iteration - the tag is: ((1584669987 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669987 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\nFirst iteration - the tag is: ((1584669988 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669988 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\nFirst iteration - the tag is: ((1584669989 secs; 740464896 nsecs), 0).\nSecond iteration - the time part of the tag is:  (1584669989 secs; 740464896 nsecs).\nSecond iteration - the microstep part of the tag is:  1.\n...
    \n

    The first reaction prints the “First iteration” part of the output at microstep 0. The second reaction occurs one microstep later (explained in Scheduling Delayed Reactions) and illustrates how to split a Tag into its constituent TimeValue and microstep.

    \n

    Summary of Time Functions

    \n

    See Time. These time functions are defined in the time.ts library of reactor-ts.

    \n

    UnitBasedTimeValue(value: number, unit:TimeUnit) Constructor for UnitBasedTimeValue, a programmer-friendly subclass of TimeValue. Use a number and a TimeUnit enum.

    \n
    enum TimeUnit {\n  nsec,\n  usec,\n  msec,\n  sec,\n  secs,\n  minute,\n  minutes,\n  hour,\n  hours,\n  day,\n  days,\n  week,\n  weeks,\n}\n
    \n

    TimeValue.add(other: TimeValue): TimeValue Adds this to other.

    \n

    TimeValue.subtract(other: TimeValue): TimeValue Subtracts other from this. A negative result is an error.

    \n

    TimeValue.difference(other: TimeValue): TimeValue Obtain absolute value of other minus this.

    \n

    TimeValue.isEqualTo(other: TimeValue): boolean Returns true if this and other represents the same TimeValue. Otherwise false.

    \n

    TimeValue.isZero(): boolean Returns true if this represents a 0 TimeValue.

    \n

    TimeValue.isEarlierThan(other: TimeValue): boolean Returns true if this < other. Otherwise false.

    \n

    Tag.isSmallerThan(other: Tag): boolean Returns true if this < other. Otherwise false.

    \n

    Tag.isSimultaneousWith(other: Tag): boolean Returns true if this and other represents the same Tag. Otherwise false.

    \n

    Tag.getLaterTag(delay: TimeValue): Tag Returns a tag with the time part of this TimeValue incremented by delay.

    \n

    Tag.getMicroStepLater(): Tag Returns a tag with the microstep part of this TimeValue incremented by 1.

    \n

    getTimeDifference(other: Tag): TimeValue Returns a TimeValue that represents the absolute (i.e., positive) time difference between this and other.

    \n
    \n
    \n

    FIXME: details needed here on time in Rust.

    \n
    \n

    Actions

    \n
    \n

    Actions are described in Actions. If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. This is particularly useful for physical actions that are externally triggered because it enables the action to convey information to the reactor. This could be, for example, the body of an incoming network message or a numerical reading from a sensor.

    \n

    Recall from Composing Reactors that the $after$ keyword on a connection between ports introduces a logical delay. This is actually implemented using a logical action. We illustrate how this is done using the DelayInt example:

    \n
    reactor Delay(delay: time = 100 ms) {\n  input in: int\n  output out: int\n  logical action a: int\n  reaction(a) -> out {=\n    if (a->has_value && a->is_present) lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // Use specialized form of schedule for integer payloads.\n    lf_schedule_int(a, self->delay, in->value);\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
      delay = new Delay();\n  source.out -> delay.in;\n  delay.in -> sink.out\n
    \n

    is equivalent to

    \n
        source.out -> sink.in after 100 ms\n
    \n

    (except that our Delay reactor will only work with data type int).

    \n

    Note: The reaction to a is given before the reaction to in above. This is important because if both inputs are present at the same tag, the first reaction must be executed before the second. Because of this reaction ordering, it is possible to create a program that has a feedback loop where the output of the Delay reactor propagates back to an input at the same tag. If the reactions were given in the opposite order, then such a program would result in a causality loop.

    \n

    In the Delay reactor, the action a is specified with a type int. The reaction to the input in declares as its effect the action a. This declaration makes it possible for the reaction to schedule a future triggering of a. The reaction uses one of several variants of the lf_schedule function, namely lf_schedule_int, a convenience function provided because integer payloads on actions are very common. We will see below, however, that payloads can have any data type.

    \n

    The first reaction declares that it is triggered by a and has effect out. To\nread the value, it uses the a->value variable. Because this reaction is first,\nthe out at any logical time can be produced before the input in is even\nknown to be present. Hence, this reactor can be used in a feedback loop, where\nout triggers a downstream reactor to send a message back to in of this same\nreactor. If the reactions were given in the opposite order, there would be a\ncausality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it as follows:

    \n
      reaction(a) -> out {=\n    if (a->has_value) {\n      lf_set(out, a->value);\n    }\n  =}\n
    \n

    It is possible to both be triggered by and schedule an action in the same\nreaction. For example, the\nfollowing CountSelf\nreactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay: time = 100 msec) {\n  output out: int\n  logical action a: int\n  reaction(startup) -> a, out {=\n    lf_set(out, 0);\n    lf_schedule_int(a, self->delay, 1);\n  =}\n  reaction(a) -> a, out {=\n    lf_set(out, a->value);\n    lf_schedule_int(a, self->delay, a->value + 1);\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    The C++ provides a simple interface for scheduling actions via a schedule() method. Actions are described in the Actions document. Consider the Schedule reactor:

    \n
    reactor Schedule {\n  input x: int\n  logical action a: void\n  reaction(x) -> a {=\n    a.schedule(200ms);\n  =}\n\n  reaction(a) {=\n    auto elapsed_time = get_elapsed_logical_time();\n    std::cout << "Action triggered at logical time " << elapsed_time.count()\n          << " after start" << std::endl;\n  =}\n}\n
    \n

    When this reactor receives an input x, it calls schedule() on the action a, specifying a logical time offset of 200 milliseconds. The action a will be triggered at a logical time 200 milliseconds after the arrival of input x. At that logical time, the second reaction will trigger and will use the get_elapsed_logical_time() function to determine how much logical time has elapsed since the start of execution.

    \n

    Notice that after the logical time offset of 200 msec, there may be another input x simultaneous with the action a. Because the reaction to a is given first, it will execute first. This becomes important when such a reactor is put into a feedback loop (see below).

    \n

    Physical actions work exactly as described in the Physical Actions section.

    \n

    Zero-Delay Actions

    \n

    If the specified delay in a schedule() is omitted or is zero, then the action a will be triggered one microstep later in superdense time (see Superdense Time). Hence, if the input x arrives at metric logical time t, and you call schedule() in one of the following ways:

    \n
    a.schedule();\na.schedule(0s);\na.schedule(reactor::Duration::zero());\n
    \n

    then when the reaction to a is triggered, the input x will be absent (it was present at the previous microstep). The reaction to x and the reaction to a occur at the same metric time t, but separated by one microstep, so these two reactions are not logically simultaneous.

    \n

    As discussed above the he metric time is visible to the programmer and can be obtained in a reaction using either get_elapsed_logical_time() or get_logical_time().

    \n

    As described in the Action document, action declarations can have a min_delay parameter. This modifies the timestamp further. Also, the action declaration may be physical rather than logical, in which case, the assigned timestamp will depend on the physical clock of the executing platform.

    \n

    Actions With Values

    \n

    If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. This is particularly useful for physical actions that are externally triggered because it enables the action to convey information to the reactor. This could be, for example, the body of an incoming network message or a numerical reading from a sensor.

    \n

    Recall from the Composing Reactors section that the after keyword on a connection between ports introduces a logical delay. This is actually implemented using a logical action. We illustrate how this is done using the DelayInt example:

    \n
    reactor Delay(delay: time = 100 ms) {\n  input in: int\n  output out: int\n  logical action d: int\n  reaction(d) -> out {=\n    if (d.is_present()) {\n      out.set(d.get());\n    }\n  =}\n  reaction(in) -> d {=\n    d.schedule(in.get(), delay);\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
    d = new Delay();\nsource.out -> d.in;\nd.in -> sink.out\n
    \n

    is equivalent to

    \n
    source.out -> sink.in after 100 ms\n
    \n

    (except that our Delay reactor will only work with data type int).

    \n

    Note: The reaction to d is given before the reaction to in above. This is important because if both inputs are present at the same tag, the first reaction must be executed before the second. Because of this reaction ordering, it is possible to create a program that has a feedback loop where the output of the Delay reactor propagates back to an input at the same tag. If the reactions were given in the opposite order, then such a program would result in a causality loop.

    \n

    The action d is specified with a type int. The reaction to the input in declares as its effect the action d. This declaration makes it possible for the reaction to schedule a future triggering of d. In the C++ target, actions use the same mechanism for passing data via value pointers as do ports. In the example above, the reactor::ImmutablValuePtr<int> derived by the call to in.get() is passed directly to schedule(). Similarly, the value can later be retrieved from the action with d.get() and passed to the output port.

    \n

    The first reaction declares that it is triggered by d and has effect out. Because this reaction is first, the out at any logical time can be produced before the input in is even known to be present. Hence, this reactor can be used in a feedback loop, where out triggers a downstream reactor to send a message back to in of this same reactor. If the reactions were given in the opposite order, there would be causality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it using is_present():

    \n
    reaction(d) -> out {=\n  if (d.is_present()) {\n    out.set(d.get());\n  }\n=}\n
    \n

    It is possible to both be triggered by and schedule an action the same reaction. For example, this reactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay:time(100 msec)) {\n  output out:int;\n  logical action a:int;\n  reaction(startup) -> a, out {=\n    out.set(0);\n    a.schedule_int(1, delay);\n  =}\n  reaction(a) -> a, out {=\n    out.set(a.get());\n    a.schedule_int(*a.get() + 1, delay);\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    Actions are described here. Actions can carry a\nvalue, a Python object that becomes available to any reaction triggered by\nthe action. This is particularly useful for physical actions that are externally\ntriggered because it enables the action to convey information to the reactor.\nThis could be, for example, the body of an incoming network message or a\nnumerical reading from a sensor. Note that actions do not have a data\ntype in the Python target, even when they carry a value.

    \n

    Recall from Composing Reactors that the\n$after$ keyword on a connection between ports introduces a logical delay. This\nis actually implemented using a logical action. We illustrate how this is done\nusing the\nDelay reactor in the DelayInt\nexample:

    \n
    reactor Delay(delay = 100 ms) {\n  input _in\n  output out\n  logical action a\n  reaction(a) -> out {=\n    if (a.value is not None) and a.is_present:\n      out.set(a.value)\n  =}\n  reaction(_in) -> a {=\n    a.schedule(self.delay, _in.value)\n  =}\n}\n
    \n

    Using this reactor as follows

    \n
        delay = new Delay()\n    <source_port_reference> -> delay._in\n    delay._in -> <destination_port_reference>\n
    \n

    is equivalent to

    \n
        <source_port_reference> -> <destination_port_reference> after 100 ms\n
    \n

    In the Delay reactor, the reaction to the input _in declares as its effect\nthe action a. This declaration makes it possible for the reaction to schedule\na future triggering of a using the\na.schedule()\nmethod.

    \n

    The first reaction declares that it is triggered by a and has effect out. To\nread the value, it uses the a.value syntax. Because this reaction is first,\nthe out at any logical time can be produced before the input _in is even\nknown to be present. Hence, this reactor can be used in a feedback loop, where\nout triggers a downstream reactor to send a message back to _in of this same\nreactor. If the reactions were given in the opposite order, there would be a\ncausality loop and compilation would fail.

    \n

    If you are not sure whether an action carries a value, you can test for it as follows:

    \n
      reaction(a) -> out {=\n    if (a.value is not None):\n      out.set(a.value)\n  =}\n
    \n

    It is possible to both be triggered by and schedule an action in the same\nreaction. For example, the\nfollowing CountSelf\nreactor will produce a counting sequence after it is triggered the first time:

    \n
    reactor CountSelf(delay = 100 ms) {\n  output out\n  logical action a\n  reaction(startup) -> a, out {=\n    out.set(0)\n    a.schedule(self.delay, 1)\n  =}\n  reaction(a) -> a, out {=\n    out.set(a.value)\n    a.schedule(self.delay, a.value + 1)\n  =}\n}\n
    \n

    Of course, to produce a counting sequence, it would be more efficient to use a state variable.

    \n
    \n
    \n

    Each action listed as an effect for a reaction is available as a schedulable object in the reaction body via the actions object. The TypeScript target provides a special actions object with a property for each schedulable action. Schedulable actions (of type t) have the object method:

    \n
    schedule: (extraDelay: TimeValue | 0, value?: T) => void;\n
    \n

    The first argument can either be the literal 0 (shorthand for 0 seconds) or a TimeValue/UnitBasedTimeValue. The second argument is the value for the action. Consider the following reactor:

    \n
    target TypeScript;\nreactor Schedule {\n  input x:number;\n  logical action a;\n  reaction(x) -> a {=\n    actions.a.schedule(new UnitBasedTimeValue(200, TimeUnit.msec), null);\n  =}\n  reaction(a) {=\n    let elapsedTime = util.getElapsedLogicalTime();\n    console.log("Action triggered at logical time " + elapsedTime + " after start.");\n  =}\n}\n
    \n

    When this reactor receives an input x, it calls schedule() on the action a, so it will be triggered at the logical time offset (200 msec) with a null value. The action a will be triggered at a logical time 200 milliseconds after the arrival of input x. This will trigger the second reaction, which will use the util.getElapsedLogicalTime() function to determine how much logical time has elapsed since the start of execution. The second argument to the schedule() function is a value, data that can be carried by the action, which is explained below. In the above example, there is no value.

    \n

    Zero-Delay Actions

    \n

    If the specified delay in a schedule() call is zero, then the action a will be triggered one microstep later in superdense time (see Superdense Time). Hence, if the input x arrives at metric logical time t, and you call schedule() as follows:

    \n
    actions.a.schedule(0);\n
    \n

    then when a reaction to a is triggered, the input x will be absent (it was present at the previous microstep). The reaction to x and the reaction to a occur at the same metric time t, but separated by one microstep, so these two reactions are not logically simultaneous. These reactions execute with different Tags.

    \n

    Actions With Values

    \n

    If an action is declared with a data type, then it can carry a value, a data value that becomes available to any reaction triggered by the action. The most common use of this is to implement a logical delay, where a value provided at an input is produced on an output with a larger logical timestamp. To accomplish that, it is much easier to use the after keyword on a connection between reactors. Nevertheless, in this section, we explain how to directly use actions with value. In fact, the after keyword is implemented as described below.

    \n

    If you are familiar with other targets (like C) you may notice it is much easier to schedule actions with values in TypeScript because of TypeScript/JavaScript’s garbage collected memory management. The following example implements a logical delay using an action with a value.

    \n
    reactor Delay(delay:time(100 ms)) {\n  input x:number;\n  output out:number;\n  logical action a:number;\n  reaction(x) -> a {=\n    actions.a.schedule(delay, x as number);\n  =}\n  reaction(a) -> out {=\n    if (a !== null){\n      out = a as number\n    }\n  =}\n}\n
    \n

    The action a is specified with a type number. The first reaction declares a as its effect. This declaration makes it possible for the reaction to schedule a future triggering of a. It’s necessary to explicitly annotate the type of x as a number in the schedule function because TypeScript doesn’t know the only trigger of a reaction must be present in that reaction.

    \n

    The second reaction declares that it is triggered by a and has effect out. When a reaction triggers or uses an action the value of that action is made available within the reaction as a local variable with the name of the action. This variable will take on the value of the action and it will have the value undefined if that action is absent because it was not scheduled for this reaction.

    \n

    The local variable cannot be used directly to schedule an action. As described above, an action a can only be scheduled in a reaction when it is 1) declared as an effect and 2) referenced through a property of the actions object. The reason for this implementation is that actions.a refers to the action, not its value, and it is possible to use both the action and the value in the same reaction. For example, the following reaction will produce a counting sequence after it is triggered the first time:

    \n
    reaction(a) -> out, a {=\n  if (a !== null) {\n    a = a as number;\n    out = a;\n    let newValue = a++;\n    actions.a.schedule(delay, newValue);\n  }\n=}\n
    \n
    \n
    \n

    Actions may carry values if they mention a data type, for instance:

    \n
    logical action act: u32;\n
    \n

    Within a reaction, you can schedule that action with a value like so

    \n
    ctx.schedule_with_v(act, Asap, 30);\n
    \n

    you can get the value from another reaction like so:

    \n
    if let Some(value) = ctx.get_action(act) {\n  // a value is present at this tag\n} else {\n  // value was not specified\n}\n
    \n

    If an action does not mention a data type, the type defaults to ().

    \n
    \n

    Schedule Functions

    \n
    \n

    Actions with values can be rather tricky to use because the value must usually be carried in dynamically allocated memory. It will not work for value to refer to a state variable of the reactor because that state variable will likely have changed value by the time the reactions to the action are invoked. Several variants of the lf_schedule function are provided to make it easier to pass values across time in varying circumstances.

    \n
    \n

    lf_schedule(<action>, <offset>);

    \n
    \n

    This is the simplest version as it carries no value. The action need not have a data type.

    \n
    \n

    lf_schedule_int(<action>, <offset>, <value>);

    \n
    \n

    This version carries an int value. The data type of the action is required to be int.

    \n
    \n

    lf_schedule_token(<action>, <offset>, <value>);

    \n
    \n

    This version carries a token, which has type token_t and points to the value, which can have any type. There is a create_token() function that can be used to create a token, but programmers will rarely need to use this. Instead, you can use lf_schedule_value() (see below), which will automatically create a token. Alternatively, for inputs with types ending in * or [], the value is wrapped in a token, and the token can be obtained using the syntax inputname->token in a reaction and then forwarded using lf_schedule_token() (see Dynamically Allocated Structs above). If the input is mutable, the reaction can then even modify the value pointed to by the token and/or use lf_schedule_token() to send the token to a future logical time. For example, the DelayPointer reactor realizes a logical delay for any data type carried by a token:

    \n
    reactor DelayPointer(delay:time(100 ms)) {\n  input in:void*;\n  output out:void*;\n  logical action a:void*;\n  reaction(a) -> out {=\n    // Using lf_set_token delegates responsibility for\n    // freeing the allocated memory downstream.\n    lf_set_token(out, a->token);\n  =}\n  reaction(in) -> a {=\n    // Schedule the actual token from the input rather than\n    // a new token with a copy of the input value.\n    lf_schedule_token(a, self->delay, in->token);\n  =}\n}\n
    \n
    \n

    lf_schedule_value(<action>, <offset>, <value>, <length>);

    \n
    \n

    This version is used to send into the future a value that has been dynamically allocated using malloc. It will be automatically freed when it is no longer needed. The value argument is a pointer to the memory containing the value. The length argument should be 1 if it is a not an array and the array length otherwise. This length will be needed downstream to interpret the data correctly. See ScheduleValue.lf.

    \n
    \n

    lf_schedule_copy(<action>, <offset>, <value>, <length>);

    \n
    \n

    This version is for sending a copy of some data pointed to by the <value> argument. The data is assumed to be a scalar or array of type matching the <action> type. The <length> argument should be 1 if it is a not an array and the array length otherwise. This length will be needed downstream to interpret the data correctly.

    \n

    Occasionally, an action payload may not be dynamically allocated nor freed. For example, it could be a pointer to a statically allocated string. If you know this to be the case, the DelayString reactor will realize a logical time delay on such a string:

    \n
    reactor DelayString(delay:time(100 msec)) {\n  input in:string;\n  output out:string;\n  logical action a:string;\n  reaction(a) -> out {=\n    lf_set(out, a->value);\n  =}\n  reaction(in) -> a {=\n    // The following copies the char*, not the string.\n    lf_schedule_copy(a, self->delay, &(in->value), 1);\n  =}\n}\n
    \n

    The data type string is an alias for char*, but Lingua Franca does not know this, so it creates a token that contains a copy of the pointer to the string rather than a copy of the string itself.

    \n
    \n
    \n

    FIXME: Give a list of schedule() functions with descriptions.

    \n
    \n
    \n

    The Python reactor target provides a .schedule() method to trigger an action at a\nfuture logical time. The .schedule() method also optionally allows for a value\nto be sent into the future. For example, take the\nScheduleValue.lf:

    \n
    main reactor ScheduleValue {\n  logical action a;\n  reaction(startup) -> a {=\n    value = "Hello"\n    a.schedule(0, value)\n  =}\n  reaction(a) {=\n    print("Received: ", a.value)\n    if a.value != "Hello":\n      sys.stderr.write("FAILURE: Should have received 'Hello'\\n")\n      exit(1)\n  =}\n}\n
    \n

    In this example, the logical action a is scheduled one\nmicrostep in the future with a string value\ncontaining \"Hello\".

    \n
    \n
    \n

    FIXME: List them here

    \n
    \n
    \n

    Within a reaction, actions may be scheduled using the schedule function:

    \n
    // schedule without additional delay\nctx.schedule(act, Asap);\n// schedule with an additional delay\nctx.schedule(act, after!(20 ms));\n// that's shorthand for\nctx.schedule(act, After(Duration.of_millis(20)));\n
    \n
    \n

    Stopping Execution

    \n
    \n

    A reaction may request that the execution stop after all events with the current timestamp have been processed by calling the built-in method request_stop(), which takes no arguments. In a non-federated execution, the actual last tag of the program will be one microstep later than the tag at which request_stop() was called. For example, if the current tag is (2 seconds, 0), the last (stop) tag will be (2 seconds, 1). In a federated execution, however, the stop time will likely be larger than the current logical time. All federates are assured of stopping at the same logical time.

    \n
    \n

    The timeout target property will take precedence over this function. For example, if a program has a timeout of 2 seconds and request_stop() is called at the (2 seconds, 0) tag, the last tag will still be (2 seconds, 0).

    \n
    \n
    \n
    \n

    A reaction may request that the execution stop after all events with the current timestamp have been processed by calling the built-in method lf.request_stop(), which takes no arguments. In a non-federated execution, the actual last tag of the program will be one microstep later than the tag at which lf.request_stop() was called. For example, if the current tag is (2 seconds, 0), the last (stop) tag will be (2 seconds, 1). In a federated execution, however, the stop time will likely be larger than the current logical time. All federates are assured of stopping at the same logical time.

    \n
    \n

    The timeout target property will take precedence over this function. For example, if a program has a timeout of 2 seconds and request_stop() is called at the (2 seconds, 0) tag, the last tag will still be (2 seconds, 0).

    \n
    \n
    \n
    \n

    A reaction may request that the execution stops after all events with the current timestamp have been processed by calling environment()->sync_shutdown(). There is also a method environment()->async_shutdown()\nwhich may be invoked from outside an reaction, like an external thread.

    \n
    \n
    \n

    A reaction may request that the execution stop by calling the function util.requestShutdown() which takes no arguments. Execution will not stop immediately when this function is called; all events with the current tag will finish processing and execution will continue for one more microstep to give shutdown triggers a chance to execute. After this additional step, execution will terminate.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n

    Log and Debug Information

    \n
    \n

    A suite of useful functions is provided in util.h for producing messages to be made visible when the generated program is run. Of course, you can always use printf, but this is not a good choice for logging or debug information, and it is not a good choice when output needs to be redirected to a window or some other user interface (see for example the sensor simulator). Also, in federated execution, these functions identify which federate is producing the message. The functions are listed below. The arguments for all of these are identical to printf with the exception that a trailing newline is automatically added and therefore need not be included in the format string.

    \n
      \n
    • \n

      LF_PRINT_DEBUG(format, ...): Use this for verbose messages that are only needed during debugging. Nothing is printed unless the target parameter logging is set to debug. THe overhead is minimized when nothing is to be printed.

      \n
    • \n
    • \n

      LF_PRINT_LOG(format, ...): Use this for messages that are useful logs of the execution. Nothing is printed unless the target parameter logging is set to log or debug. This is a macro so that overhead is minimized when nothing is to be printed.

      \n
    • \n
    • \n

      lf_print(format, ...): Use this for messages that should normally be printed but may need to be redirected to a user interface such as a window or terminal (see register_print_function below). These messages can be suppressed by setting the logging target property to warn or error.

      \n
    • \n
    • \n

      lf_print_warning(format, ...): Use this for warning messages. These messages can be suppressed by setting the logging target property to error.

      \n
    • \n
    • \n

      lf_print_error(format, ...): Use this for error messages. These messages are not suppressed by any logging target property.

      \n
    • \n
    • \n

      lf_print_error_and_exit(format, ...): Use this for catastrophic errors.

      \n
    • \n
    \n

    In addition, a utility function is provided to register a function to redirect printed outputs:

    \n
      \n
    • lf_register_print_function(function): Register a function that will be used instead of printf to print messages generated by any of the above functions. The function should accept the same arguments as printf.
    • \n
    \n
    \n
    \n

    The reactor-cpp library provides logging utilities in logging.hh for producing messages to be made visible when the generated program is run. Of course std::cout or printf can be used for the same purpose, but the logging mechanism provided by reactor-cpp is thread-safe ensuring that messages produced in parallel reactions are not interleaved with each other and provides common way for turning messages of a certain severity on and off.

    \n

    In particular, reactor-cpp provides the following logging interfaces:

    \n
      \n
    • reactor::log::Debug(): for verbose debug messages
    • \n
    • reactor::log::Info(): for info messages of general interest, info is the default severity level
    • \n
    • reactor::log::Warning(): for warning messages
    • \n
    • reactor::log::Error(): for errors
    • \n
    \n

    These utilities can be used analogues to std::cout. For instance:

    \n
    reactor::log::Info() << "Hello World! It is " << get_physical_time();\n
    \n

    Note that unlike std::cout the new line delimiter is automatically added to the end of the message.

    \n

    Which type of messages are actually produced by the compiled program can be controlled with the log-level target property.

    \n
    \n
    \n

    The Python supports the logging target specification. This will cause the runtime to produce more or less information about the execution. However, user-facing functions for different logging levels are not yet implemented (see issue #619).

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    The executable reacts to the environment variable RUST_LOG, which sets the logging level of the application. Possible values are\noff, error, warn, info, debug, trace

    \n

    Error and warning logs are on by default. Enabling a level enables all greater levels (i.e., RUST_LOG=info also enables warn and error, but not trace or debug).

    \n

    Logging can also be turned on with the --log-level CLI option, if the application features a CLI.

    \n

    Note that the logging target property is ignored, as its levels do not match the Rust standard levels we use (those of the log crate).

    \n

    Note that when building with a release profile (i.e., target property build-type is not Debug), all log statements with level debug and trace are removed from the executable, and cannot be turned on at runtime. A warning is produced by the executable if you try to use these levels explicitly.

    \n
    \n

    Libraries Available to Programmers

    \n
    \n

    Libraries Available in All Programs

    \n

    Reactions in C can use a number of pre-defined functions, macros, and constants without having to explicitly include any header files:

    \n
      \n
    • \n

      Time and tags (tag.h):

      \n
        \n
      • Specifying time value, such as MSEC and FOREVER
      • \n
      • Time data types, such as tag_t and instant_t
      • \n
      • Obtaining tag and time information, e.g. lf_time_logical and lf_time_physical
      • \n
      \n
    • \n
    • \n

      Ports

      \n
        \n
      • Writing to output ports, such as lf_set and lf_set_token (set.h)
      • \n
      • Iterating over sparse multiports, such as lf_multiport_iterator and lf_multiport_next (port.h)
      • \n
      \n
    • \n
    • \n

      Scheduling actions

      \n
        \n
      • Schedule future events, such as lf_schedule and lf_schedule_value (api.h)
      • \n
      \n
    • \n
    • \n

      File Access

      \n
        \n
      • LF_SOURCE_DIRECTORY: A C string giving the full path to the directory containing the .lf file of the program.
      • \n
      • LF_PACKAGE_DIRECTORY: A C string giving the full path to the directory that is the root of the project or package (normally, the directory above the src directory).
      • \n
      • LF_FILE_SEPARATOR: A C string giving the file separator for the platform containing the .lf file (”/” for Unix-like systems, ”\\” for Windows).
      • \n
      \n
    • \n
    \n

    These are useful when your application needs to open and read additional files. For example, the following C code can be used to open a file in a subdirectory called dir of the directory that contains the .lf file:

    \n
        const char* path = LF_SOURCE_DIRECTORY LF_FILE_SEPARATOR "dir" LF_FILE_SEPARATOR "filename"\n    FILE* fp = fopen(path, "rb");
    \n
      \n
    • \n

      Miscellaneous

      \n
        \n
      • Changing modes in modal models, lf_set_mode (set.h)
      • \n
      • Checking deadlines, lf_check_deadline (api.h)
      • \n
      • Defining and recording tracepoints, such as register_user_trace_event and tracepoint (trace.h)
      • \n
      • Printing utilities, such as lf_print and lf_print_error (util.h)
      • \n
      • Logging utilities, such as LF_PRINT_LOG and LF_PRINT_DEBUG (util.h)
      • \n
      \n
    • \n
    \n

    Standard C Libraries

    \n

    The generated C code automatically includes the following standard C libraries (see also the C standard library header files):

    \n
      \n
    • limits.h (Defines INT_MIN, INT_MAX, etc.)
    • \n
    • stdbool.h (Defines bool datatype and true and false constants)
    • \n
    • stddef.h (Defines size_t, NULL, etc.)
    • \n
    • stdint.h (Defines int64_t, int32_t, etc.)
    • \n
    • stdlib.h (Defines exit, getenv, atoi, etc.)
    • \n
    \n

    Hence, programmers are free to use functions from these libraries without explicitly providing a #include statement. Nevertheless, providing one is harmless and may be good form. In particular, future releases may not include these header files

    \n

    Available Libraries Requiring #include

    \n

    More sophisticated library functions require a #include statement in a $preamble$.\nSpecifically, platform.h includes the following:

    \n
      \n
    • Sleep functions such as lf_sleep
    • \n
    • Mutual exclusion such as lf_critial_section_enter and lf_critical_section_exit
    • \n
    • Threading functions such as lf_thread_create
    • \n
    \n

    The threading functions are only available for platforms that support multithreading.

    \n

    Available Libraries Requiring #include, a files entry, and a cmake-include

    \n

    A few utility libraries are provided, but require considerably more setup.\nThese also help to illustrate how to incorporate your own libraries.

    \n\n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    FIXME: Details needed here.

    \n
    \n
    \n

    Scheduler Target Property

    \n

    The scheduler target property is used to select the scheduler used by the C runtime. This scheduler determines the exact order in which reactions are processed, as long as the order complies with the deterministic semantics of Lingua Franca. It also assigns reactions to user-level threads and can thereby influence the assignment of reactions to processors.

    \n

    Because the C runtime scheduler operates at a higher level of abstraction than the OS, none of the scheduling policies that we currently support allow preemption; furthermore, they do not control migration of threads between processors.

    \n

    Another limitation of these schedulers is that they are constrained to process the reaction graph breadth-first. We define the level of a reaction r to be the length of the longest chain of causally dependent reactions that are all (causally) upstream of r. Current LF schedulers process one level of reactions at a time, but this constraint is more restrictive than necessary to implement Lingua Franca’s semantics and is notable only for its effect on execution times.

    \n

    The following schedulers are available:

    \n
      \n
    • NP (non-preemptive): This scheduler is the default scheduler. It ignores deadlines.
    • \n
    • GEDF_NP (global earliest-deadline-first, non-preemptive): When the semantics of Lingua Franca allows for concurrent execution of two or more ready reactions with the same level at a particular tag, this scheduler will prioritize the reaction with the earliest deadline to run first. Reactions with no explicit deadline implicitly have an infinitely late deadline.
    • \n
    • adaptive: This experimental scheduler behaves similarly to the NP scheduler, with the additional limitation that it is designed for applications that have potentially wide variability in physical execution times. It performs experiments to measure execution times at runtime to determine the degree of exploitable parallelism in various parts of the program. This lets it automate judgments which are made more naively by the other schedulers and which are typically made by the programmer in general-purpose languages.
    • \n
    \n
    \n

    Target Implementation Details

    \n
    \n

    Included Libraries

    \n

    Definitions for the following do not need to be explicitly included because the code generator exposes them in the user namespace automatically:

    \n
      \n
    • Functions and macros used to set ports and iterate over multiports
    • \n
    • Functions and macros used to schedule actions
    • \n
    • Functions and macros used to set a reactor’s mode
    • \n
    • Functions and macros used to create trace points
    • \n
    • Logging utility functions
    • \n
    • Typedefs relating to time and logical time, including tag_t, instant_t, interval_t, and microstep_t
    • \n
    • API functions for obtaining timing information about the current program execution, including the current physical and logical time
    • \n
    \n

    Some standard C libraries are exposed to the user through reactor.h, including stddef.h,\nstdio.h, and stdlib.h. In addition, math.h gets automatically included. However, users who wish to avoid breaking changes between releases should\nconsider including these libraries explicitly instead of relying on their being exposed by the\nruntime.

    \n

    Users who wish to include functionality that has a platform-specific implementation may choose to\nexplicitly include platform.h, which provides a uniform interface for various concurrency\nprimitives and sleep functions.

    \n

    Multithreaded Implementation

    \n

    By default, the C runtime system uses multiple worker threads in order to take advantage of multicore execution. The number of worker threads will match the number of cores on the machine unless the workers argument is given in the target statement or the --workers command-line argument is given.

    \n

    Upon initialization, the main thread will create the specified number of worker threads.\nExecution proceeds in a manner similar to the single threaded implementation\nexcept that the worker threads concurrently draw reactions from the reaction queue.\nThe execution algorithm ensures that no reaction executes until all reactions that it depends on have executed or it has been determined that they will not execute at the current tag.

    \n

    Single Threaded Implementation

    \n

    By giving the single-threaded (target option)[/docs/handbook/target-declaration#single-threaded] or the --single-threaded (command-line argument)[/docs/handbook/target-declaration#command-line-arguments], the generated program will execute the program using only a single thread. This option is most useful for creating programs to run on bare-metal microprocessors that have no threading support. On such platforms, mutual exclusion is typically realized by disabling interrupts.

    \n

    The execution strategy is to have two queues of pending accessor invocations, one that is sorted by\ntag (the event queue) and one that is sorted by priority (the reaction queue).\nExecution proceeds as follows:

    \n
      \n
    1. \n

      At initialization, an event for each timer is put on the event queue and logical time is initialized to the current time, represented as the number of nanoseconds elapsed since January 1, 1970.

      \n
    2. \n
    3. \n

      At each logical time, pull all events from event queue that have the same earliest tag, find the reactions that these events trigger, and put them on the reaction queue. If there are no events on the event queue, then exit the program (unless the --keepalive true (command-line argument)[/docs/handbook/target-declaration#command-line-arguments] is given).

      \n
    4. \n
    5. \n

      Wait until physical time matches or exceeds that earliest timestamp (unless the --fast true (command-line argument)[/docs/handbook/target-declaration#command-line-arguments] is given). Then advance logical time to match that earliest timestamp.

      \n
    6. \n
    7. \n

      Execute reactions in order of priority from the reaction queue. These reactions may produce outputs, which results in more events getting put on the reaction queue. Those reactions are assured of having lower priority than the reaction that is executing. If a reaction calls lf_schedule(), an event will be put on the event queue, not the reaction queue.

      \n
    8. \n
    9. \n

      When the reaction queue is empty, go to 2.

      \n
    10. \n
    \n
    \n
    \n

    Unlike the C target, the Cpp target implements more of the analysis and setup of a Lingua Franca in the runtime system. The runtime system is define in the reactor-cpp repository on GitHub. See that repo for details.

    \n
    \n
    \n

    The Python target is built on top of the C runtime to enable maximum efficiency where possible. It uses the single-threaded C runtime by default but will switch to the multi-threaded C runtime if a physical action is detected. The threading target property can be used to override this behavior.

    \n

    Running lfc on a XXX.lf program that uses the Python target specification on a\nLinux machine will create the following files (other operating systems will have\na slightly different structure and/or files):

    \n
    ├── src\n│   └── XXX.lf\n└── src-gen\n    └── XXX\n        ###### Files related to the Python C extension module for XXX ######\n        ├── build               # Temporary files for setuptools\n        ├── core                # Core C runtime files\n        ├── ctarget.c           # C target API implementations\n        ├── ctarget.h           # C target API definitions\n        ├── LinguaFrancaXXX*.so # The Python C extension module for XXX\n        ├── pythontarget.c      # Python target API implementations\n        ├── pythontarget.h      # Python target API definitions\n        ├── setup.py            # Setup file used to build the Python C extension\n        ├── XXX.c               # Source code of the Python C extension\n        ###### Files containing the Python code ######\n        └── XXX.py              # Python file containing reactors and reaction code\n
    \n

    There are two major components in the src-gen/XXX directory that together enable the execution of a Python target application:

    \n
      \n
    • An XXX.py file containing the user code (e.g., reactor definitions and reactions).
    • \n
    • The source code for a Python C extension module called LinguaFrancaXXX containing the C runtime, as well as hooks to execute the user-defined reactions.
    • \n
    \n

    The interactions between the src-gen/XXX/XXX.py file and the LinguaFrancaXXX module are explained below.

    \n

    The XXX.py file containing user code

    \n

    The XXX.py file contains all the reactor definitions in the form of Python classes. The contents of a reactor are converted as follows:

    \n
      \n
    • Each Reaction in a reactor definition will be converted to a class method.
    • \n
    • Each Parameter will be converted to a class property to make it read-only.
    • \n
    • Each State variable will be converted to an instance variable.
    • \n
    • Each trigger and effect will be converted to an object passed as a method function argument to reaction methods, allowing the body of the reaction to access them.
    • \n
    • Each reactor Preamble will be put in the class definition verbatim.
    • \n
    \n

    Finally, each reactor class instantiation will be converted to a Python object class instantiation.

    \n

    For example, imagine the following program:

    \n
    # src/XXX.lf\ntarget Python;\nreactor Foo(bar(0)) {\n  preamble {=\n    import random\n  =}\n  state baz\n  input _in\n  logical action act\n  reaction(_in, act) {=\n    # Body of the reaction\n    self.random.seed() # Note the use of self\n  =}\n}\nmain reactor {\n  foo = new Foo()\n}\n
    \n

    Th reactor Foo and its instance, foo, will be converted to

    \n
    # src-gen/XXX/XXX.py\n...\n# Python class for reactor Foo\nclass _Foo:\n\n    # From the preamble, verbatim:\n    import random\n    def __init__(self, **kwargs):\n        #Define parameters and their default values\n        self._bar = 0\n        # Handle parameters that are set in instantiation\n        self.__dict__.update(kwargs)\n\n        # Define state variables\n        self.baz = None\n\n    @property\n    def bar(self):\n        return self._bar\n\n    def reaction_function_0(self , _in, act):\n        # Body of the reaction\n        self.random.seed() # Note the use of self\n        return 0\n\n\n# Instantiate classes\nxxx_lf = [None] * 1\nxxx_foo_lf = [None] * 1\n# Start initializing XXX of class XXX\nfor xxx_i in range(1):\n    bank_index = xxx_i\n    xxx_lf[0] = _XXX(\n        _bank_index = 0,\n    )\n    # Start initializing XXX.foo of class Foo\n    for xxx_foo_i in range(1):\n        bank_index = xxx_foo_i\n        xxx_foo_lf[0] = _Foo(\n            _bank_index = 0,\n            _bar=0,\n        )\n...\n
    \n

    The generated LinguaFrancaXXX Python module (a C extension module)

    \n

    The rest of the files in src-gen/XXX form a Python C extension\nmodule\ncalled LinguaFrancaXXX that can be built by executing python3 setup.py build_ext --inplace in the src-gen/XXX/ folder. In this case, Python will\nread the instructions in the src-gen/XXX/setup.py file and build a\nLinguaFrancaXXX module in src-gen/XXX/. The --inplace flag puts the\ncompiled extension (the LinguaFrancaXXX*.so in the example above) in the\nsrc-gen directory alongside the XXX.py file.

    \n

    As mentioned before, the LinguaFrancaXXX module is separate from\nsrc-gen/XXX/XXX.py but interacts with it. Next, we explain this interaction.

    \n

    Interactions between XXX.py and LinguaFrancaXXX

    \n

    The LinguaFrancaXXX module is imported in src-gen/XXX/XXX.py:

    \n
    from LinguaFrancaXXX import *\n
    \n

    This is done to enable the main function in src-gen/XXX/XXX.py to make a call to the start() function, which is part of the generated (and installed) LinguaFrancaXXX module. This function will start the main event handling loop of the C runtime.

    \n

    From then on, LinguaFrancaXXX will call reactions that are defined in src-gen/XXX/XXX.py when needed.

    \n

    The LinguaFrancaBase package

    \n

    LinguaFrancaBase is a package that contains several helper methods and definitions that are necessary for the Python target to work. This module is installable via python3 -m pip install LinguaFrancaBase but is automatically installed if needed during the installation of LinguaFrancaXXX. The source code of this package can be found on GitHub.

    \n

    This package’s modules are imported in the XXX.py program:

    \n
    from LinguaFrancaBase.constants import * #Useful constants\nfrom LinguaFrancaBase.functions import * #Useful helper functions\nfrom LinguaFrancaBase.classes import * #Useful classes\n
    \n

    Already imported Python modules

    \n

    The following packages are already imported and thus do not need to be re-imported by the user:

    \n
    import os\nimport sys\nimport copy\n
    \n
    \n
    \n

    When a TypeScript reactor is compiled, the generated code is placed inside a project directory. This is because there are two steps of compilation. First, the Lingua Franca compiler generates a TypeScript project from the TypeScript reactor code. Second, the Lingua Franca compiler runs a TypeScript compiler on the generated TypeScript project to produce executable JavaScript. This is illustrated below:

    \n
    Lingua Franca (.lf) ==> TypeScript (.ts) ==> JavaScript (.js)\n
    \n

    Assuming the directory containing our Lingua Franca file Foo.lf is named TS, the compiler will generate the following:

    \n
      \n
    1. TS/package.json
    2. \n
    3. TS/node_modules
    4. \n
    5. TS/Foo/tsconfig.json
    6. \n
    7. TS/Foo/babel.config.js
    8. \n
    9. TS/Foo/src/
    10. \n
    11. TS/Foo/dist/
    12. \n
    \n

    Items 1, 3, and 4 are configuration files for the generated project. Item 2 is a node_modules directory with contents specified by item 1. Item 5 is the directory for generated TypeScript code. Item 6 is the directory for compiled JavaScript code. In addition to the generated code for your Lingua Franca program, items 5 and 6 include libraries from the reactor-ts submodule.

    \n

    The Lingua Franca compiler automatically invokes other programs as it compiles a Lingua Franca (.lf) file to a Node.js executable JavaScript (.js) file. The files package.json, babel.config.js, and tsconfig.json are used to configure the behavior of those other programs. Whenever you compile a .lf file for the first time, the Lingua Franca compiler will copy default versions of these configuration files into the new project so the other programs can run. The Lingua Franca compiler will only copy a default configuration file into a project if that file is not already present in the generated project. This means you, the reactor programmer, may safely modify these configuration files to control the finer points of compilation. Beware, other generated files in the project’s src and dist directories may be overwritten by the compiler.

    \n

    package.json

    \n

    Node.js uses a package.json file to describe metadata relevant to a Node project. This includes a list of project dependencies (i.e. modules) used by the project. When the Lingua Franca compiler copies a default package.json file into a Lingua Franca project that doesn’t already have a package.json, the compiler runs the command npm install to create a node_modules directory. The default package.json only lists dependencies for the reactor-ts submodule. Follow these instructions to modify package.json if you want to use other Node modules in your reactors.

    \n

    tsconfig.json

    \n

    After generating a TypeScript program from a .lf file, the Lingua Franca compiler uses the TypeScript compiler tsc to run a type check. The behavior of tsc is configured by the tsconfig.json file. You probably won’t need to modify tsconfig.json, but you can if you know what you’re doing.

    \n

    babel.config.js

    \n

    If the tsc type check was successful, the Lingua Franca compiler uses babel to compile the generated TypeScript code into JavaScript. (This blog post articulates the advantages of using babel over tsc to generate JavaScript.) There are many different flavors of JavaScript and the babel.config.js file specifies exactly what babel should generate. This is the file to edit if you want the Lingua Franca compiler to produce a different version of JavaScript as its final output.

    \n

    Debugging Type Errors

    \n

    Let’s take the minimal reactor example, and intentionally break it by adding a type error into the reaction.

    \n
    target TypeScript;\nmain reactor ReactionTypeError {\n    timer t;\n    reaction(t) {=\n        let foo:number = "THIS IS NOT A NUMBER";\n        console.log("Hello World.");\n    =}\n}\n
    \n

    This reactor will not compile, and should you attempt to compile it you will get an output from the compiler which looks something like this:

    \n
    --- Standard output from command:\nsrc/ReactionTypeError.ts(23,25): error TS2322: Type '"THIS IS NOT A NUMBER"' is not assignable to type 'number'.\n\n--- End of standard output.
    \n

    In particular the output

    \n
    src/ReactionTypeError.ts(23,25): error TS2322: Type '"THIS IS NOT A NUMBER"' is not assignable to type 'number'.
    \n

    identifies the problem: surprisingly, the string \"THIS IS NOT A NUMBER\" is not a number. However the line information (23,25) is a little confusing because it points to the location of the type error in the generated .ts file ReactionTypeError/src/ReactionTypeError.ts not in the original .lf file ReactionTypeError.lf. The .ts files produced by the TypeScript code generator are quite readable if you are familiar with the reactor-ts submodule, but even if you aren’t familiar it is not too difficult to track down the problem. Just open ReactionTypeError/src/ReactionTypeError.ts in your favorite text editor (we recommend Visual Studio for its excellent TypeScript integration) and look at line 23.

    \n
    14        this.addReaction(\n15            new Triggers(this.t),\n16            new Args(this.t),\n17            function (this, __t: Readable<Tag>) {\n18                // =============== START react prologue\n19                const util = this.util;\n20                let t = __t.get();\n21                // =============== END react prologue\n22                try {\n23                    let foo:number = "THIS IS NOT A NUMBER";\n24                    console.log("Hello World.");\n25                } finally {\n26                    // =============== START react epilogue\n27\n28                    // =============== END react epilogue\n29                }\n30            }\n31        );\n
    \n

    There (inside the try block) we can find the problematic reaction code. Reaction code is copied verbatim into generated .ts files.

    \n

    It can be a bit harder to interpret type errors outside of reaction code, but most type error messages are still relatively clear. For example if you attempt to connect a reactor output to an incompatibly typed input like:

    \n
    target TypeScript;\nmain reactor ConnectionError {\n    s = new Sender();\n    r = new Receiver();\n    s.foo -> r.bar;\n}\nreactor Sender {\n    output foo:number;\n}\nreactor Receiver {\n    input bar:string;\n}\n
    \n

    you should get an error like

    \n
    --- Standard output from command:\nsrc/InputTypeError.ts(36,23): error TS2345: Argument of type 'OutPort<number>' is not assignable to parameter of type 'Port<string>'.\n  Types of property 'value' are incompatible.\n    Type 'number | undefined' is not assignable to type 'string | undefined'.\n      Type 'number' is not assignable to type 'string | undefined'.\n\n--- End of standard output.
    \n

    The key message being Argument of type 'OutPort<number>' is not assignable to parameter of type 'Port<string>'.

    \n

    One last tip: if you attempt to reference a port, action, timer etc. named foo that isn’t declared in the triggers, uses, or effects declaration of the reaction, you will get the error Cannot find name 'foo' in the reaction body.

    \n

    Utility Function Reference

    \n

    These utility functions may be called within a TypeScript reaction:

    \n

    util.requestShutdown(): void Ends execution after one microstep. See Stopping Execution.

    \n

    util.getCurrentTag(): Tag Gets the current (logical) tag. See Tags.

    \n

    util.getCurrentLogicalTime(): TimeValue Gets the current logical TimeValue. See Time.

    \n

    util.getCurrentPhysicalTime(): TimeValue Gets the current physical TimeValue. See Time.

    \n

    util.getElapsedLogicalTime(): TimeValue Gets the elapsed logical TimeValue from execution start. See Time.

    \n

    util.getElapsedPhysicalTime(): TimeValue Gets the elapsed physical TimeValue from execution start. See Time.

    \n

    util.success(): void Invokes the reactor-ts App’s default success callback. FIXME: Currently doesn’t do anything in Lingua Franca.

    \n

    util.failure(): void Invokes the reactor-ts App’s default failure callback. Throws an error.

    \n

    Building Reactor-ts Documentation

    \n

    To build and view proper documentation for time.ts (and other reactor-ts libraries), install typedoc and run

    \n
    typedoc --out docs src\n
    \n

    from the root of the reactor-ts. You probably already have the reactor-ts submodule at

    \n
    lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/
    \n

    You should see an output like.

    \n
    Using TypeScript 3.8.3 from /usr/local/lib/node_modules/typescript/lib\nRendering [========================================] 100%\n\nDocumentation generated at /Users/<username>/git/lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/docs
    \n

    Open that path in a browser with /index.html appended to the end like

    \n
    /Users/<username>/git/lingua-franca/xtext/org.icyphy.linguafranca/src/lib/TS/reactor-ts/docs/index.html
    \n

    to navigate the docs.

    \n
    \n
    \n

    Target Properties

    \n

    Target properties may be mentioned like so:

    \n
    target Rust {\n    // enables single-file project layout\n    single-file-project: false,\n    // timeout for the execution. The program will shutdown at most after the specified duration.\n    timeout: 3 sec,\n\n    cargo-features: ["cli"]\n}\n
    \n

    See Target Declaration for the full list of supported target properties.

    \n

    The Executable

    \n

    The executable name is the name of the main reactor transformed to snake_case: main reactor RustProgram will generate rust_program. See Command-Line Arguments for details.

    \n

    File layout

    \n

    The Rust code generator generates a Cargo project with a classical layout:

    \n
    ├── Cargo.lock\n├── Cargo.toml\n├── src\n│   ├── main.rs\n│   └── reactors\n│       ├── mod.rs\n|       ├── ...\n|\n└── target\n    ├── ...
    \n

    The module structure is as follows:

    \n
      \n
    • the crate has a module reactors
    • \n
    • each LF reactor has its own submodule of reactors. For instance, Minimal.lf will generate minimal.rs. The name is transformed to snake_case.
    • \n
    \n

    This means that to refer to the contents of another reactor module, e.g. that of Other.lf, you have to write super::other::Foo. This is relevant to access preamble items.

    \n

    Single-file layout

    \n

    The Rust target supports an alternative file layout, where all reactors are generated into the main.rs file, making the project fit in a single file (excluding Cargo.toml). The module structure is unchanged: the file still contains a mod reactors { ... } within which each reactor has its mod foo { ... }. You can thus change the layout without having to update any LF code.

    \n

    Set the target property single-file-project: true to use this layout.

    \n

    Note: this alternative layout is provided for the purposes of making self-contained benchmark files. Generating actual runnable benchmarks from an LF file may be explored in the future.

    \n

    Specifying dependencies

    \n

    The Rust code generator leverages Cargo to allow LF programs to profit from Rust’s large package ecosystem. The code generator may also link support files written in pure Rust into the generated crate. Target properties are used to achieve all this.

    \n

    Adding cargo dependencies

    \n

    The cargo-dependencies target property may be used to specify dependencies on crates coming from crates.io. Here’s an example:

    \n
    target Rust {\n   cargo-dependencies: {\n      termcolor: "0.8"\n   }\n};\n
    \n

    The value of the cargo-dependencies property is a map of crate identifiers to a dependency-spec. An informal example follows:

    \n
    cargo-dependencies: {\n   // Name-of-the-crate: "version"\n   rand: "0.8",\n   // Equivalent to using an explicit map:\n   rand: {\n     version: "0.8"\n   },\n   // The map allows specifying more details\n   rand: {\n     // A path to a local unpublished crate.\n     // Note 'path' is mutually exclusive with 'git'.\n     path: "/home/me/Git/local-rand-clone"\n   },\n   rand: {\n     // A URL to a git repo\n     git: "https://github.com/me/rand",\n     // Specify an explicit Git revision number\n     rev: "abcdef1234"\n   },\n   rand: {\n     version: "0.8",\n     // you can specify cargo features\n     features: ["some-cargo-feature",]\n   }\n}\n
    \n

    When a dependency-spec is specified as an object, its key-value pairs correspond directly to those of a Cargo dependency specification. For instance for the following dependency spec:

    \n
       rand: {\n     version: "0.8",\n     // you can specify cargo features\n     features: ["some-cargo-feature",]\n   }\n
    \n

    we add the following to the generated Cargo.toml:

    \n
    [dependencies.rand]\nversion = "0.8"\nfeatures = ["some-cargo-feature"]\n
    \n

    Not all keys are necessarily supported though, e.g. the registry key is not supported (yet).

    \n

    Configuring the runtime

    \n

    The runtime crate can be configured just like other crates, using the cargo-dependencies target property, e.g.:

    \n
    cargo-dependencies: {\n   reactor_rt: {\n     features: ["parallel-runtime"]\n   }\n}\n
    \n

    The dependency is always included, with defaults picked by LFC. The location information (path/git/version key) is optional.\nSee reactor_rt for the supported features.

    \n

    Linking support files

    \n

    You can link-in additional rust modules using the rust-include target property:

    \n
    target Rust {\n  rust-include: ["foo.rs"]\n};\n
    \n

    The property is a list of paths (relative to the directory containing the .lf file). Each path should either point to a Rust file (.rs), or a directory that contains a mod.rs file. Each of those will be copied to the src directory of the generated Cargo project, and linked in to the main.rs file.

    \n

    To refer to the included module, you can use e.g. crate::foo if your module is named foo.

    \n

    Generation scheme

    \n

    Each reactor generates its own struct which contains state variables. For instance,

    \n\n\n\n\n\n\n\n\n\n\n\n\n\n
    LFGenerated Rust
    \n
    reactor SomeReactor {\n  state field: u32(0)\n}\n
    \n
    \n
    struct SomeReactor {\n  field: u32\n}\n
    \n
    \n

    In the following we refer to that struct as the state struct.

    \n

    Reactions

    \n

    Reactions are each generated in a separate method of the reactor struct. Reaction names are unspecified and may be mangled to prevent explicit calling. The parameters of that method are

    \n
      \n
    • &mut self: the state struct described above,
    • \n
    • ctx: &mut ReactionCtx: the context object for the reaction execution,
    • \n
    • For each dependency, a parameter is generated.\n
        \n
      • If the dependency is a component of this reactor, the name of the parameter is just the name of the component
      • \n
      • If the dependency is a port of a child reactor, the name of the parameter is <name of the child instance>__<name of the port>, e.g. child__out for child.out.
      • \n
      • The type of the parameter depends on the kind of dependency and of component:
      • \n
      \n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
      ComponentUse/trigger dependencyEffect dependency
      \n\n\n

      Port of type T

      \n
      \n

      &ReadablePort<T>

      \n
      \n

      WritablePort<T>

      \n
      \n

      Logical action of type T

      \n
      \n

      &LogicalAction<T>

      \n
      \n

      &mut LogicalAction<T>

      \n
      \n

      Physical action of type T

      \n
      \n

      &PhysicalActionRef<T>

      \n
      \n

      &mut PhysicalActionRef<T>

      \n
      Timer\n

      &Timer

      \n
      \n

      n/a

      \n
      \n

      Port bank of type T

      \n
      \n

      &ReadablePortBank<T>

      \n
      \n

      WritablePortBank<T>

      \n
      \n

      Undeclared dependencies, and dependencies on timers and startup or shutdown, do not generate a parameter.

      \n

      The ReactionCtx object is a mediator to manipulate all those dependency objects. It has methods to set ports, schedule actions, retrieve the current logical time, etc.

      \n

      For instance:

      \n
      reactor Source {\n    output out: i32;\n    reaction(startup) -> out {=\n        ctx.set(out, 76600)\n    =}\n}\n
      \n

      In this example, the context object ctx is used to set a port to a value. The port is in scope as out.

      \n
      \n

      ⚠ TODO when the runtime crate is public link to the docs, they should be the most exhaustive documentation.

      \n
      \n
    \n````","headings":[{"value":"Overview","depth":2},{"value":"Requirements","depth":2},{"value":"Limitations","depth":2},{"value":"The Target Specification","depth":2},{"value":"Parameters and State Variables","depth":2},{"value":"Array Values for Parameters","depth":3},{"value":"Array Values for States","depth":3},{"value":"States and Parameters with Struct Values","depth":3},{"value":"Array-Valued Parameters","depth":3},{"value":"State Variables","depth":3},{"value":"Array Expressions for State Variables and Parameters","depth":3},{"value":"Assigning Arbitrary Initial Expressions to State Variables and Parameters","depth":3},{"value":"Array or Object Parameters","depth":3},{"value":"Inputs and Outputs","depth":2},{"value":"Sending and Receiving Data","depth":3},{"value":"Persistent Inputs","depth":3},{"value":"Fixed Length Array Inputs and Outputs","depth":3},{"value":"Variable Length Array Inputs and Outputs","depth":3},{"value":"Dynamically Allocated Data","depth":3},{"value":"Mutable Inputs","depth":3},{"value":"String Types","depth":3},{"value":"Macros For Setting Output Values","depth":3},{"value":"Sending and Receiving Large Data Types","depth":3},{"value":"Sending and Receiving Objects","depth":3},{"value":"Sending and Receiving Custom Types","depth":3},{"value":"Time","depth":2},{"value":"Tags","depth":3},{"value":"Summary of Time Functions","depth":3},{"value":"Actions","depth":2},{"value":"Zero-Delay Actions","depth":3},{"value":"Actions With Values","depth":3},{"value":"Zero-Delay Actions","depth":3},{"value":"Actions With Values","depth":3},{"value":"Schedule Functions","depth":3},{"value":"Stopping Execution","depth":2},{"value":"Log and Debug Information","depth":2},{"value":"Libraries Available to Programmers","depth":2},{"value":"Libraries Available in All Programs","depth":4},{"value":"Standard C Libraries","depth":4},{"value":"Available Libraries Requiring #include","depth":4},{"value":"Available Libraries Requiring #include, a files entry, and a cmake-include","depth":4},{"value":"Scheduler Target Property","depth":2},{"value":"Target Implementation Details","depth":2},{"value":"Included Libraries","depth":3},{"value":"Multithreaded Implementation","depth":3},{"value":"Single Threaded Implementation","depth":3},{"value":"The XXX.py file containing user code","depth":3},{"value":"The generated LinguaFrancaXXX Python module (a C extension module)","depth":3},{"value":"Interactions between XXX.py and LinguaFrancaXXX","depth":3},{"value":"The LinguaFrancaBase package","depth":3},{"value":"Already imported Python modules","depth":3},{"value":"package.json","depth":3},{"value":"tsconfig.json","depth":3},{"value":"babel.config.js","depth":3},{"value":"Debugging Type Errors","depth":3},{"value":"Utility Function Reference","depth":3},{"value":"Building Reactor-ts Documentation","depth":3},{"value":"Target Properties","depth":3},{"value":"The Executable","depth":3},{"value":"File layout","depth":3},{"value":"Single-file layout","depth":4},{"value":"Specifying dependencies","depth":3},{"value":"Adding cargo dependencies","depth":4},{"value":"Configuring the runtime","depth":4},{"value":"Linking support files","depth":4},{"value":"Generation scheme","depth":3},{"value":"Reactions","depth":4}],"frontmatter":{"permalink":"/docs/handbook/target-language-details","title":"Target Language Details","oneline":"Detailed reference for each target langauge.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Expressions","oneline":"Expressions in Lingua Franca.","permalink":"/docs/handbook/expressions"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Target Declaration","oneline":"The target declaration and its parameters in Lingua Franca.","permalink":"/docs/handbook/target-declaration"}}}},"pageContext":{"id":"3-target-language-details","slug":"/docs/handbook/target-language-details","repoPath":"/packages/documentation/copy/en/reference/Target Language Details.md","previousID":"24e3b5ae-ac22-5a79-956a-4258d40ae77c","nextID":"de456861-0847-5726-aba2-da3ba779bd9d","lang":"en","modifiedTime":"2023-11-08T00:42:45.528Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/termination/page-data.json b/page-data/docs/handbook/termination/page-data.json index ce312029c..4cd4397e3 100644 --- a/page-data/docs/handbook/termination/page-data.json +++ b/page-data/docs/handbook/termination/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/termination","result":{"data":{"markdownRemark":{"id":"6f15c9ee-d99f-5526-99a6-3dd6bce05820","excerpt":"Shutdown Reactions There are several mechanisms for terminating a Lingua Franca in an orderly fashion.\nAll of these mechanisms result in a final tag at which…","html":"

    Shutdown Reactions

    \n

    There are several mechanisms for terminating a Lingua Franca in an orderly fashion.\nAll of these mechanisms result in a final tag at which any reaction that declares $shutdown$ as a trigger will be invoked (recall that a tag is a tuple (logical time, microstep)). Other reactions may also be invoked at this final tag, and the order in which reactions are invoked will be constrained by the normal precedence rules.

    \n

    If a reaction triggered by $shutdown$ produces outputs, then downstream reactors will also be invoked at the final tag. If the reaction schedules any actions by calling schedule(), those will be ignored. In fact, any event after the final tag will be ignored. After the completion of the final tag, the program will exit.

    \n

    There are four ways to terminate a program:

    \n
      \n
    • Timeout: The program specifies the last logical time at which reactions should be triggered.
    • \n
    • Starvation: At the conclusion of some tag, there are no events in the event queue at future tags.
    • \n
    • Stop request: Some reaction requests that the program terminate.
    • \n
    • External signal: Program is terminated externally using operating services like control-C or kill.
    • \n
    \n

    We address each of these in turn.

    \n

    Timeout

    \n

    The target property timeout specifies the last logical time at which reactions should be triggered. The last invocation of reactions will be at tag (timeout, 0).

    \n

    There is a significant subtlety when using physical connections, which are connections using the syntax ~>. Such connections specify that the tag at the receiving end will be based on the physical time at which the message is received. If the tag assigned at the receiving end is greater than the final tag, then the message is lost. Hence, messages sent near the timeout time are likely to be lost!

    \n

    Starvation

    \n

    If a Lingua Franca program has no physical actions, and if at any time during execution there are no future events waiting to be processed, then there is no possibility for any more reactions to occur and the program will exit. This situation is called starvation. If there is a timer anywhere in the program with a period, then this condition never occurs.

    \n

    One subtlety is that reactions triggered by $shutdown$ will be invoked one microstep later than the last tag at which there was an event. They cannot be invoked at the same tag because it is only after that last tag has completed that the runtime system can be sure that there are no future events. It would not be correct to trigger the $shutdown$ reactions at that point because it would be impossible to respect the required reaction ordering.

    \n
    \n

    Starvation termination is not currently implemented for federated execution. You will need to use one of the other mechanisms to terminate a federated program.

    \n
    \n

    Stop Request

    \n

    If a reaction calls the built-in request_stop() function, then it is requesting that the program cease execution as soon as possible. This cessation will normally occur in the next microstep. The current tag will be completed as normal. Then the tag will be advanced by one microstep, and reactions triggered by $shutdown$ will be executed, along with any other reactions with triggers at that tag, with all reactions executed in precedence order.

    \n
    \n

    In a federated execution, things are more complicated. In general, it is not possible to cease execution in the next microstep because this would mean that every federate has a communication channel to every other with delay equal to one microstep. This does not create a causality loop, but it means that all federates have to advance time in lockstep, which creates a global barrier synchronization that would likely kill performance. It would also make decentralized coordination impossible because the safe-to-process (STP) threshold for all federates would diverge to infinity.

    \n

    For centralized coordination, when a reaction in a federate calls request_stop(), the federate sends a STOP_REQUEST message to the RTI with its current timestamp t as a payload and completes execution of any other reactions triggered at the current tag. It then blocks, waiting for a STOP_GRANTED message with a timestamp payload s. If s > t, then it sets timeout = s and continues executing, using the timeout mechanism (see above) to stop. If s = t, then schedules the shutdown phase to occur one microstep later, as in the unfederated case.

    \n

    When the RTI receives a STOP_REQUEST message from a federate, it forwards it to all other federates and waits for a reply from all. Each reply will have a timestamp payload. The RTI chooses s, the largest of these timestamps, and sends a STOP_GRANTED message to all federates with payload s.

    \n

    When a federate receives a STOP_REQUEST message, it replies with its current logical time t, completes its current tag (if one is progress), and blocks, waiting for a STOP_GRANTED message from the RTI. When it gets the reply with payload s, if s > t, then it sets timeout = s and continues executing, using the timeout mechanism (see above) to stop. If s = t, then it schedules the shutdown phase to occur one microstep later, as in the unfederated case.

    \n
    \n

    External Signal

    \n

    A control-C or other kill signal to a running Lingua Franca program will cause execution to stop immediately.

    \n
    \n

    For federated programs, each federate and the RTI catches external signals to shut down in an orderly way.

    \n

    When a federate gets such an external signal (e.g. control-C), it sends a RESIGN message to the RTI and an EOF (end of file) on each socket connection to another federate. It then closes all sockets and shuts down. The RTI and all other federates should continue running until some other termination condition occurs.

    \n

    When the RTI gets such an external signal (e.g. control-C), it broadcasts a STOP_REQUEST message to all federates, waits for their replies (with a timeout in case the federate or the network has failed), chooses the maximum timestamp s on the replies, broadcasts a STOP_GRANTED message to all federates with payload s, and waits for LOGICAL_TIME_COMPLETE messages as above.

    \n
    ","headings":[{"value":"Shutdown Reactions","depth":2},{"value":"Timeout","depth":2},{"value":"Starvation","depth":2},{"value":"Stop Request","depth":2},{"value":"External Signal","depth":2}],"frontmatter":{"permalink":"/docs/handbook/termination","title":"Termination","oneline":"Terminating a Lingua Franca execution.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","permalink":"/docs/handbook/distributed-execution"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"id":"1-termination","slug":"/docs/handbook/termination","repoPath":"/packages/documentation/copy/en/topics/Termination.md","previousID":"79d9c9b2-eee4-5652-9541-c483de60119e","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/termination","result":{"data":{"markdownRemark":{"id":"6f15c9ee-d99f-5526-99a6-3dd6bce05820","excerpt":"Shutdown Reactions There are several mechanisms for terminating a Lingua Franca in an orderly fashion.\nAll of these mechanisms result in a final tag at which…","html":"

    Shutdown Reactions

    \n

    There are several mechanisms for terminating a Lingua Franca in an orderly fashion.\nAll of these mechanisms result in a final tag at which any reaction that declares $shutdown$ as a trigger will be invoked (recall that a tag is a tuple (logical time, microstep)). Other reactions may also be invoked at this final tag, and the order in which reactions are invoked will be constrained by the normal precedence rules.

    \n

    If a reaction triggered by $shutdown$ produces outputs, then downstream reactors will also be invoked at the final tag. If the reaction schedules any actions by calling schedule(), those will be ignored. In fact, any event after the final tag will be ignored. After the completion of the final tag, the program will exit.

    \n

    There are four ways to terminate a program:

    \n
      \n
    • Timeout: The program specifies the last logical time at which reactions should be triggered.
    • \n
    • Starvation: At the conclusion of some tag, there are no events in the event queue at future tags.
    • \n
    • Stop request: Some reaction requests that the program terminate.
    • \n
    • External signal: Program is terminated externally using operating services like control-C or kill.
    • \n
    \n

    We address each of these in turn.

    \n

    Timeout

    \n

    The target property timeout specifies the last logical time at which reactions should be triggered. The last invocation of reactions will be at tag (timeout, 0).

    \n

    There is a significant subtlety when using physical connections, which are connections using the syntax ~>. Such connections specify that the tag at the receiving end will be based on the physical time at which the message is received. If the tag assigned at the receiving end is greater than the final tag, then the message is lost. Hence, messages sent near the timeout time are likely to be lost!

    \n

    Starvation

    \n

    If a Lingua Franca program has no physical actions, and if at any time during execution there are no future events waiting to be processed, then there is no possibility for any more reactions to occur and the program will exit. This situation is called starvation. If there is a timer anywhere in the program with a period, then this condition never occurs.

    \n

    One subtlety is that reactions triggered by $shutdown$ will be invoked one microstep later than the last tag at which there was an event. They cannot be invoked at the same tag because it is only after that last tag has completed that the runtime system can be sure that there are no future events. It would not be correct to trigger the $shutdown$ reactions at that point because it would be impossible to respect the required reaction ordering.

    \n
    \n

    Starvation termination is not currently implemented for federated execution. You will need to use one of the other mechanisms to terminate a federated program.

    \n
    \n

    Stop Request

    \n

    If a reaction calls the built-in request_stop() function, then it is requesting that the program cease execution as soon as possible. This cessation will normally occur in the next microstep. The current tag will be completed as normal. Then the tag will be advanced by one microstep, and reactions triggered by $shutdown$ will be executed, along with any other reactions with triggers at that tag, with all reactions executed in precedence order.

    \n
    \n

    In a federated execution, things are more complicated. In general, it is not possible to cease execution in the next microstep because this would mean that every federate has a communication channel to every other with delay equal to one microstep. This does not create a causality loop, but it means that all federates have to advance time in lockstep, which creates a global barrier synchronization that would likely kill performance. It would also make decentralized coordination impossible because the safe-to-process (STP) threshold for all federates would diverge to infinity.

    \n

    For centralized coordination, when a reaction in a federate calls request_stop(), the federate sends a STOP_REQUEST message to the RTI with its current timestamp t as a payload and completes execution of any other reactions triggered at the current tag. It then blocks, waiting for a STOP_GRANTED message with a timestamp payload s. If s > t, then it sets timeout = s and continues executing, using the timeout mechanism (see above) to stop. If s = t, then schedules the shutdown phase to occur one microstep later, as in the unfederated case.

    \n

    When the RTI receives a STOP_REQUEST message from a federate, it forwards it to all other federates and waits for a reply from all. Each reply will have a timestamp payload. The RTI chooses s, the largest of these timestamps, and sends a STOP_GRANTED message to all federates with payload s.

    \n

    When a federate receives a STOP_REQUEST message, it replies with its current logical time t, completes its current tag (if one is progress), and blocks, waiting for a STOP_GRANTED message from the RTI. When it gets the reply with payload s, if s > t, then it sets timeout = s and continues executing, using the timeout mechanism (see above) to stop. If s = t, then it schedules the shutdown phase to occur one microstep later, as in the unfederated case.

    \n
    \n

    External Signal

    \n

    A control-C or other kill signal to a running Lingua Franca program will cause execution to stop immediately.

    \n
    \n

    For federated programs, each federate and the RTI catches external signals to shut down in an orderly way.

    \n

    When a federate gets such an external signal (e.g. control-C), it sends a RESIGN message to the RTI and an EOF (end of file) on each socket connection to another federate. It then closes all sockets and shuts down. The RTI and all other federates should continue running until some other termination condition occurs.

    \n

    When the RTI gets such an external signal (e.g. control-C), it broadcasts a STOP_REQUEST message to all federates, waits for their replies (with a timeout in case the federate or the network has failed), chooses the maximum timestamp s on the replies, broadcasts a STOP_GRANTED message to all federates with payload s, and waits for LOGICAL_TIME_COMPLETE messages as above.

    \n
    ","headings":[{"value":"Shutdown Reactions","depth":2},{"value":"Timeout","depth":2},{"value":"Starvation","depth":2},{"value":"Stop Request","depth":2},{"value":"External Signal","depth":2}],"frontmatter":{"permalink":"/docs/handbook/termination","title":"Termination","oneline":"Terminating a Lingua Franca execution.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Distributed Execution","oneline":"Distributed Execution (preliminary)","permalink":"/docs/handbook/distributed-execution"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"id":"1-termination","slug":"/docs/handbook/termination","repoPath":"/packages/documentation/copy/en/topics/Termination.md","previousID":"79d9c9b2-eee4-5652-9541-c483de60119e","lang":"en","modifiedTime":"2023-11-08T00:42:45.532Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/time-and-timers/page-data.json b/page-data/docs/handbook/time-and-timers/page-data.json index 8b2ec5eec..8db70b693 100644 --- a/page-data/docs/handbook/time-and-timers/page-data.json +++ b/page-data/docs/handbook/time-and-timers/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/time-and-timers","result":{"data":{"markdownRemark":{"id":"051bc602-4435-5af0-930a-f471f905219e","excerpt":"$page-showing-target$ Logical Time A key property of Lingua Franca is logical time. All events occur at an instant in logical time. By default, the runtime…","html":"

    $page-showing-target$

    \n

    Logical Time

    \n

    A key property of Lingua Franca is logical time. All events occur at an instant in logical time. By default, the runtime system does its best to align logical time with physical time, which is some measurement of time on the execution platform. The lag is defined to be physical time minus logical time, and the goal of the runtime system is maintain a small non-negative lag.

    \n

    The lag is allowed to go negative only if the fast target property or the --fast command-line argument is set to true. In that case, the program will execute as fast as possible with no regard to physical time.

    \n
    \n

    In Lingua Franca, $time$ is a data type.\nA parameter, state variable, port, or action may have type $time$.\nIn the C target, time values internally have type instant_t or interval_t,\nboth of which are (usually) equivalent to the C type long long.\nIn the C++ target, time values internally have the type std::chrono::nanoseconds. For details, see the Target Language Details.\nIn the Rust target, time values internally have type FIXME.

    \n
    \n

    Time Values

    \n

    A time value is given with units (unless the value is 0, in which case the units can be omitted).\nThe allowable units are:

    \n
      \n
    • For nanoseconds: ns, nsec, or nsecs
    • \n
    • For microseconds: us, usec, or usecs
    • \n
    • For milliseconds: ms, msec, or msecs
    • \n
    • For seconds: s, sec, secs, second, or seconds
    • \n
    • For minutes: min, minute, mins, or minutes
    • \n
    • For hours: h, hour, or hours
    • \n
    • For days: d, day, or days
    • \n
    • For weeks: week or weeks
    • \n
    \n

    The following example illustrates using time values for parameters and state variables:

    \n

    $start(SlowingClock)$

    \n
    target C\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state interval: time = start\n  logical action a\n  reaction(startup) -> a {=\n    lf_schedule(a, self->start);\n  =}\n  reaction(a) -> a {=\n    instant_t elapsed_logical_time = lf_time_logical_elapsed();\n    printf("Logical time since start: %lld nsec.\\n",\n        elapsed_logical_time\n    );\n    self->interval += self->incr;\n    lf_schedule(a, self->interval);\n  =}\n}\n
    \n
    target Cpp\nmain reactor SlowingClock(start: time(100 ms), incr: time(100 ms)) {\n  state interval: time(start)\n  logical action a\n  reaction(startup) -> a {=\n    a.schedule(start);\n  =}\n  reaction(a) -> a {=\n    auto elapsed_logical_time = get_elapsed_logical_time();\n    std::cout << "Logical time since start: " << elapsed_logical_time << " nsec" << std::endl;\n    interval += incr;\n    a.schedule(interval);\n  =}\n}\n
    \n
    target Python\nmain reactor SlowingClock(start = 100 ms, incr = 100 ms) {\n  state interval = start\n  logical action a\n  reaction(startup) -> a {=\n    a.schedule(self.start)\n  =}\n  reaction(a) -> a {=\n    elapsed_logical_time = lf.time.logical_elapsed()\n    print(\n        f"Logical time since start: {elapsed_logical_time} nsec."\n    )\n    self.interval += self.incr\n    a.schedule(self.interval)\n  =}\n}\n
    \n
    target TypeScript\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state interval: time = start\n  logical action a\n  reaction(startup) -> a {=\n    actions.a.schedule(start, null);\n  =}\n  reaction(a) -> a {=\n    console.log(`Logical time since start: ${util.getElapsedLogicalTime()}`)\n    interval = interval.add(incr)\n    actions.a.schedule(interval, null)\n  =}\n}\n
    \n
    target Rust\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state start = start\n  state incr = incr\n  state interval: time = start\n  state expected_time: time()\n  logical action a\n  reaction(startup) -> a {=\n    ctx.schedule(a, After(self.start));\n  =}\n  reaction(a) -> a {=\n    println!(\n        "Logical time since start: {} nsec.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n    self.interval += self.incr;\n    ctx.schedule(a, After(self.interval));\n    self.expected_time += self.interval;\n  =}\n}\n
    \n

    $end(SlowingClock)$

    \n

    This has two time parameters, start and incr, each with default value 100 ms and type $time$. This parameter is used to initialize the interval state variable, which also stores a time. The $logical$ $action$ a, explained in Actions, is used to schedule events to occur at time start after program startup and then at intervals that are increased each time by incr. The result of executing this program will look like this:

    \n
    Logical time since start: 100000000 nsec.\nLogical time since start: 300000000 nsec.\nLogical time since start: 600000000 nsec.\nLogical time since start: 1000000000 nsec.\n...
    \n

    Timers

    \n

    The simplest use of logical time in Lingua Franca is to invoke a reaction periodically. This is done by first declaring a $timer$ using this syntax:

    \n
      timer <name>(<offset>, <period>)\n
    \n

    The <period>, which is optional, specifies the time interval between timer events. The <offset>, which is also optional, specifies the (logical) time interval between when the program starts executing and the first timer event. If no period is given, then the timer event occurs only once. If neither an offset nor a period is specified, then one timer event occurs at program start, simultaneous with the $startup$ event.

    \n

    The period and offset are given by a number and a units, for example, 10 ms. See the expressions documentation for allowable units. Consider the following example:

    \n

    $start(Timer)$

    \n
    target C\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    printf("Logical time is %lld.\\n", lf_time_logical());\n  =}\n}\n
    \n
    target Cpp\nmain reactor Timer {\n  timer t(0, 1 s)\n  reaction(t) {=\n    std::cout << "Logical time is: " << get_logical_time() << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(f"Logical time is {lf.time.logical()}.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    console.log(`Logical time is ${util.getCurrentLogicalTime()}.`)\n  =}\n}\n
    \n
    target Rust\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    println!(\n        "Logical time is {}.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Timer)$

    \n

    This specifies a timer named t that will first trigger at the start of execution and then repeatedly trigger at intervals of one second. Notice that the time units can be left off if the value is zero.

    \n

    This target provides a built-in function for retrieving the logical time at which the reaction is invoked,\nget_logical_time()\nFIXME\nlf.time.logical()\nutil.getCurrentLogicalTime()\nFIXME.\nOn most platforms (with the exception of some embedded platforms), the returned value is a 64-bit number representing the number of nanoseconds that have elapsed since January 1, 1970. Executing the above displays something like the following:

    \n
    Logical time is 1648402121312985000.\nLogical time is 1648402122312985000.\nLogical time is 1648402123312985000.\n...
    \n

    The output lines appear at one second intervals unless the fast option has been specified.

    \n

    Elapsed Time

    \n

    The times above are a bit hard to read, so, for convenience, each target provides a built-in function to retrieve the elapsed time. For example:

    \n

    $start(TimeElapsed)$

    \n
    target C\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    printf(\n        "Elapsed logical time is %lld.\\n",\n        lf_time_logical_elapsed()\n    );\n  =}\n}\n
    \n
    target Cpp\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    std::cout << "Elapsed logical time is " << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    print(\n        f"Elapsed logical time is {lf.time.logical_elapsed()}."\n    )\n  =}\n}\n
    \n
    target TypeScript\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    console.log(`Elapsed logical time is ${util.getElapsedLogicalTime()}`)\n  =}\n}\n
    \n
    target Rust\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    println!(\n        "Elapsed logical time is {}.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(TimeElapsed)$

    \n

    See the Target Language Details for the full set of functions provided for accessing time values.

    \n

    Executing this program will produce something like this:

    \n
    Elapsed logical time is 0.\nElapsed logical time is 1000000000.\nElapsed logical time is 2000000000.\n...
    \n

    Comparing Logical and Physical Times

    \n

    The following program compares logical and physical times:

    \n

    $start(TimeLag)$

    \n
    target C\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    interval_t t = lf_time_logical_elapsed();\n    interval_t T = lf_time_physical_elapsed();\n    printf(\n        "Elapsed logical time: %lld, physical time: %lld, lag: %lld\\n",\n        t, T, T-t\n    );\n  =}\n}\n
    \n
    target Cpp\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    auto logical_time = get_elapsed_logical_time();\n    auto physical_time = get_elapsed_physical_time();\n    std::cout << "Elapsed logical time: " << logical_time\n        << " physical time: " << physical_time\n        << " lag: " << physical_time - logical_time <<  std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    t = lf.time.logical_elapsed()\n    T = lf.time.physical_elapsed()\n    print(\n        f"Elapsed logical time: {t}, physical time: {T}, lag: {T-t}"\n    )\n  =}\n}\n
    \n
    target TypeScript\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    const t = util.getElapsedLogicalTime()\n    const T = util.getElapsedPhysicalTime()\n    console.log(`Elapsed logical time: ${t}, physical time: ${T}, lag: ${T.subtract(t)}`)\n  =}\n}\n
    \n
    target Rust\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    let t = ctx.get_elapsed_logical_time();\n    let T = ctx.get_elapsed_physical_time();\n    println!(\n      "Elapsed logical time: {}, physical time: {}, lag: {}",\n      t.as_nanos(),\n      T.as_nanos(),\n      (T-t).as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(TimeLag)$

    \n

    Execution will show something like this:

    \n
    Elapsed logical time: 0, physical time: 855000, lag: 855000\nElapsed logical time: 1000000000, physical time: 1004714000, lag: 4714000\nElapsed logical time: 2000000000, physical time: 2004663000, lag: 4663000\nElapsed logical time: 3000000000, physical time: 3000210000, lag: 210000\n...
    \n

    In this case, the lag varies from a few hundred microseconds to a small number of milliseconds. The amount of lag will depend on the execution platform.

    \n

    Simultaneity and Instantaneity

    \n

    If two timers have the same offset and period, then their events are logically simultaneous. No observer will be able to see that one timer has triggered and the other has not.

    \n

    A reaction is always invoked at a well-defined logical time, and logical time does not advance during its execution. Any output produced by the reaction will be logically simultaneous with the input. In other words, reactions are logically instantaneous (for an exception, see Logical Execution Time). Physical time, however, does elapse during execution of a reaction.

    \n

    Timeout

    \n

    By default, a Lingua Franca program will terminate when there are no more events to process. If there is a timer with a non-zero period, then there will always be more events to process, so the default execution will be unbounded. To specify a finite execution horizon, you can either specify a timeout target property or a --timeout command-line option. For example, the following timeout property will cause the above timer with a period of one second to terminate after 11 events:

    \n
    target C {\n  timeout: 10 s\n}\n
    \n
    target Cpp {\n  timeout: 10 s\n}\n
    \n
    target Python {\n  timeout: 10 s\n}\n
    \n
    target TypeScript {\n  timeout: 10 s\n}\n
    \n
    target Rust {\n  timeout: 10 s\n}\n
    \n

    Startup and Shutdown

    \n

    To cause a reaction to be invoked at the start of execution, a special startup trigger is provided:

    \n
    reactor Foo {\n  reaction(startup) {=\n    ... perform initialization ...\n  =}\n}\n
    \n

    The startup trigger is equivalent to a timer with no offset or period.

    \n

    To cause a reaction to be invoked at the end of execution, a special shutdown trigger is provided. Consider the following reactor, commonly used to build regression tests:

    \n

    $start(TestCount)$

    \n
    target C\nreactor TestCount(start: int = 0, stride: int = 1, num_inputs: int = 1) {\n  state count: int = start\n  state inputs_received: int = 0\n  input x: int\n  reaction(x) {=\n    printf("Received %d.\\n", x->value);\n    if (x->value != self->count) {\n      printf("ERROR: Expected %d.\\n", self->count);\n      exit(1);\n    }\n    self->count += self->stride;\n    self->inputs_received++;\n  =}\n  reaction(shutdown) {=\n    printf("Shutdown invoked.\\n");\n    if (self->inputs_received != self->num_inputs) {\n      printf("ERROR: Expected to receive %d inputs, but got %d.\\n",\n          self->num_inputs,\n          self->inputs_received\n      );\n      exit(2);\n    }\n  =}\n}\n
    \n
    target Cpp\nreactor TestCount(start: int = 0, stride: int = 1, num_inputs: int = 1) {\n  state count: int = start\n  state inputs_received: int = 0\n  input x: int\n  reaction(x) {=\n    auto value = *x.get();\n    std::cout << "Received " <<  value << std::endl;\n    if (value != count) {\n      std::cerr << "ERROR: Expected: "<< count << std::endl;\n      exit(1);\n    }\n    count += stride;\n    inputs_received++;\n  =}\n  reaction(shutdown) {=\n    std::cout << "Shutdown invoked." << std::endl;\n    if (inputs_received != num_inputs) {\n      std::cerr << "ERROR: Expected to receive " << num_inputs\n          << " inputs, but got " << inputs_received << std::endl;\n      exit(2);\n    }\n  =}\n}\n
    \n
    target Python\nreactor TestCount(start=0, stride=1, num_inputs=1) {\n  state count = start\n  state inputs_received = 0\n  input x\n  reaction(x) {=\n    print(f"Received {x.value}.")\n    if x.value != self.count:\n      sys.stderr.write(f"ERROR: Expected {self.count}.\\n")\n      exit(1)\n    self.count += self.stride\n    self.inputs_received += 1\n  =}\n  reaction(shutdown) {=\n    print("Shutdown invoked.")\n    if self.inputs_received != self.num_inputs:\n      sys.stderr.write(\n          f"ERROR: Expected to receive {self.num_inputs} inputs, but got {self.inputs_received}.\\n"\n      )\n      exit(2)\n  =}\n}\n
    \n
    target TypeScript\nreactor TestCount(start: number = 0, stride: number = 1, numInputs: number = 1) {\n  state count: number = start\n  state inputsReceived: number = 0\n  input x: number\n  reaction(x) {=\n    console.log(`Received ${x}`)\n    if (x != count) {\n      console.error(`ERROR: Expected ${count}.`)\n      process.exit(1)\n    }\n    count += stride;\n    inputsReceived++\n  =}\n  reaction(shutdown) {=\n    console.log("Shutdown invoked.")\n    if (inputsReceived != numInputs) {\n      console.error(`ERROR: Expected to receive ${numInputs}, but got ${inputsReceived}.`)\n      process.exit(2)\n    }\n  =}\n}\n
    \n
    target Rust\nreactor TestCount(start: u32 = 0, stride: u32 = 1, num_inputs: u32 = 1) {\n  state stride = stride\n  state num_inputs = num_inputs\n  state count: u32 = start\n  state inputs_received: u32 = 0\n  input x: u32\n  reaction(x) {=\n    let x = ctx.get(x).unwrap();\n    println!("Received {}.", x);\n    if x != self.count {\n      println!("ERROR: Expected {}.", self.count);\n      std::process::exit(1);\n    }\n    self.count += self.stride;\n    self.inputs_received += 1;\n  =}\n  reaction(shutdown) {=\n    println!("Shutdown invoked.");\n    if self.inputs_received != self.num_inputs {\n      println!(\n          "ERROR: Expected to receive {} inputs, but got {}.",\n          self.num_inputs,\n          self.inputs_received\n      );\n      std::process::exit(2);\n    }\n  =}\n}\n
    \n

    $end(TestCount)$

    \n

    This reactor tests its inputs against expected values, which are expected to start with the value given by the start parameter and increase by stride with each successive input. It expects to receive a total of num_inputs input events. It checks the total number of inputs received in its $shutdown$ reaction.

    \n

    The shutdown trigger typically occurs at microstep 0, but may occur at a larger microstep. See Superdense Time and Termination.

    ","headings":[{"value":"Logical Time","depth":2},{"value":"Time Values","depth":2},{"value":"Timers","depth":2},{"value":"Elapsed Time","depth":2},{"value":"Comparing Logical and Physical Times","depth":2},{"value":"Simultaneity and Instantaneity","depth":2},{"value":"Timeout","depth":2},{"value":"Startup and Shutdown","depth":2}],"frontmatter":{"permalink":"/docs/handbook/time-and-timers","title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","permalink":"/docs/handbook/parameters-and-state-variables"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","permalink":"/docs/handbook/composing-reactors"}}}},"pageContext":{"id":"1-time-and-timers","slug":"/docs/handbook/time-and-timers","repoPath":"/packages/documentation/copy/en/topics/Time and Timers.md","previousID":"20781702-b6a5-5a16-b4d8-b4c45cd76fa3","nextID":"834f9d0d-f7c6-5732-8c60-bad1954701f7","lang":"en","modifiedTime":"2023-11-03T01:50:26.537Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/time-and-timers","result":{"data":{"markdownRemark":{"id":"051bc602-4435-5af0-930a-f471f905219e","excerpt":"$page-showing-target$ Logical Time A key property of Lingua Franca is logical time. All events occur at an instant in logical time. By default, the runtime…","html":"

    $page-showing-target$

    \n

    Logical Time

    \n

    A key property of Lingua Franca is logical time. All events occur at an instant in logical time. By default, the runtime system does its best to align logical time with physical time, which is some measurement of time on the execution platform. The lag is defined to be physical time minus logical time, and the goal of the runtime system is maintain a small non-negative lag.

    \n

    The lag is allowed to go negative only if the fast target property or the --fast command-line argument is set to true. In that case, the program will execute as fast as possible with no regard to physical time.

    \n
    \n

    In Lingua Franca, $time$ is a data type.\nA parameter, state variable, port, or action may have type $time$.\nIn the C target, time values internally have type instant_t or interval_t,\nboth of which are (usually) equivalent to the C type long long.\nIn the C++ target, time values internally have the type std::chrono::nanoseconds. For details, see the Target Language Details.\nIn the Rust target, time values internally have type FIXME.

    \n
    \n

    Time Values

    \n

    A time value is given with units (unless the value is 0, in which case the units can be omitted).\nThe allowable units are:

    \n
      \n
    • For nanoseconds: ns, nsec, or nsecs
    • \n
    • For microseconds: us, usec, or usecs
    • \n
    • For milliseconds: ms, msec, or msecs
    • \n
    • For seconds: s, sec, secs, second, or seconds
    • \n
    • For minutes: min, minute, mins, or minutes
    • \n
    • For hours: h, hour, or hours
    • \n
    • For days: d, day, or days
    • \n
    • For weeks: week or weeks
    • \n
    \n

    The following example illustrates using time values for parameters and state variables:

    \n

    $start(SlowingClock)$

    \n
    target C\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state interval: time = start\n  logical action a\n  reaction(startup) -> a {=\n    lf_schedule(a, self->start);\n  =}\n  reaction(a) -> a {=\n    instant_t elapsed_logical_time = lf_time_logical_elapsed();\n    printf("Logical time since start: %lld nsec.\\n",\n        elapsed_logical_time\n    );\n    self->interval += self->incr;\n    lf_schedule(a, self->interval);\n  =}\n}\n
    \n
    target Cpp\nmain reactor SlowingClock(start: time(100 ms), incr: time(100 ms)) {\n  state interval: time(start)\n  logical action a\n  reaction(startup) -> a {=\n    a.schedule(start);\n  =}\n  reaction(a) -> a {=\n    auto elapsed_logical_time = get_elapsed_logical_time();\n    std::cout << "Logical time since start: " << elapsed_logical_time << " nsec" << std::endl;\n    interval += incr;\n    a.schedule(interval);\n  =}\n}\n
    \n
    target Python\nmain reactor SlowingClock(start = 100 ms, incr = 100 ms) {\n  state interval = start\n  logical action a\n  reaction(startup) -> a {=\n    a.schedule(self.start)\n  =}\n  reaction(a) -> a {=\n    elapsed_logical_time = lf.time.logical_elapsed()\n    print(\n        f"Logical time since start: {elapsed_logical_time} nsec."\n    )\n    self.interval += self.incr\n    a.schedule(self.interval)\n  =}\n}\n
    \n
    target TypeScript\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state interval: time = start\n  logical action a\n  reaction(startup) -> a {=\n    actions.a.schedule(start, null);\n  =}\n  reaction(a) -> a {=\n    console.log(`Logical time since start: ${util.getElapsedLogicalTime()}`)\n    interval = interval.add(incr)\n    actions.a.schedule(interval, null)\n  =}\n}\n
    \n
    target Rust\nmain reactor SlowingClock(start: time = 100 ms, incr: time = 100 ms) {\n  state start = start\n  state incr = incr\n  state interval: time = start\n  state expected_time: time()\n  logical action a\n  reaction(startup) -> a {=\n    ctx.schedule(a, After(self.start));\n  =}\n  reaction(a) -> a {=\n    println!(\n        "Logical time since start: {} nsec.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n    self.interval += self.incr;\n    ctx.schedule(a, After(self.interval));\n    self.expected_time += self.interval;\n  =}\n}\n
    \n

    $end(SlowingClock)$

    \n

    This has two time parameters, start and incr, each with default value 100 ms and type $time$. This parameter is used to initialize the interval state variable, which also stores a time. The $logical$ $action$ a, explained in Actions, is used to schedule events to occur at time start after program startup and then at intervals that are increased each time by incr. The result of executing this program will look like this:

    \n
    Logical time since start: 100000000 nsec.\nLogical time since start: 300000000 nsec.\nLogical time since start: 600000000 nsec.\nLogical time since start: 1000000000 nsec.\n...
    \n

    Timers

    \n

    The simplest use of logical time in Lingua Franca is to invoke a reaction periodically. This is done by first declaring a $timer$ using this syntax:

    \n
      timer <name>(<offset>, <period>)\n
    \n

    The <period>, which is optional, specifies the time interval between timer events. The <offset>, which is also optional, specifies the (logical) time interval between when the program starts executing and the first timer event. If no period is given, then the timer event occurs only once. If neither an offset nor a period is specified, then one timer event occurs at program start, simultaneous with the $startup$ event.

    \n

    The period and offset are given by a number and a units, for example, 10 ms. See the expressions documentation for allowable units. Consider the following example:

    \n

    $start(Timer)$

    \n
    target C\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    printf("Logical time is %lld.\\n", lf_time_logical());\n  =}\n}\n
    \n
    target Cpp\nmain reactor Timer {\n  timer t(0, 1 s)\n  reaction(t) {=\n    std::cout << "Logical time is: " << get_logical_time() << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    print(f"Logical time is {lf.time.logical()}.")\n  =}\n}\n
    \n
    target TypeScript\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    console.log(`Logical time is ${util.getCurrentLogicalTime()}.`)\n  =}\n}\n
    \n
    target Rust\nmain reactor Timer {\n  timer t(0, 1 sec)\n  reaction(t) {=\n    println!(\n        "Logical time is {}.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(Timer)$

    \n

    This specifies a timer named t that will first trigger at the start of execution and then repeatedly trigger at intervals of one second. Notice that the time units can be left off if the value is zero.

    \n

    This target provides a built-in function for retrieving the logical time at which the reaction is invoked,\nget_logical_time()\nFIXME\nlf.time.logical()\nutil.getCurrentLogicalTime()\nFIXME.\nOn most platforms (with the exception of some embedded platforms), the returned value is a 64-bit number representing the number of nanoseconds that have elapsed since January 1, 1970. Executing the above displays something like the following:

    \n
    Logical time is 1648402121312985000.\nLogical time is 1648402122312985000.\nLogical time is 1648402123312985000.\n...
    \n

    The output lines appear at one second intervals unless the fast option has been specified.

    \n

    Elapsed Time

    \n

    The times above are a bit hard to read, so, for convenience, each target provides a built-in function to retrieve the elapsed time. For example:

    \n

    $start(TimeElapsed)$

    \n
    target C\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    printf(\n        "Elapsed logical time is %lld.\\n",\n        lf_time_logical_elapsed()\n    );\n  =}\n}\n
    \n
    target Cpp\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    std::cout << "Elapsed logical time is " << get_elapsed_logical_time() << std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    print(\n        f"Elapsed logical time is {lf.time.logical_elapsed()}."\n    )\n  =}\n}\n
    \n
    target TypeScript\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    console.log(`Elapsed logical time is ${util.getElapsedLogicalTime()}`)\n  =}\n}\n
    \n
    target Rust\nmain reactor TimeElapsed {\n  timer t(0, 1 s)\n  reaction(t) {=\n    println!(\n        "Elapsed logical time is {}.",\n        ctx.get_elapsed_logical_time().as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(TimeElapsed)$

    \n

    See the Target Language Details for the full set of functions provided for accessing time values.

    \n

    Executing this program will produce something like this:

    \n
    Elapsed logical time is 0.\nElapsed logical time is 1000000000.\nElapsed logical time is 2000000000.\n...
    \n

    Comparing Logical and Physical Times

    \n

    The following program compares logical and physical times:

    \n

    $start(TimeLag)$

    \n
    target C\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    interval_t t = lf_time_logical_elapsed();\n    interval_t T = lf_time_physical_elapsed();\n    printf(\n        "Elapsed logical time: %lld, physical time: %lld, lag: %lld\\n",\n        t, T, T-t\n    );\n  =}\n}\n
    \n
    target Cpp\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    auto logical_time = get_elapsed_logical_time();\n    auto physical_time = get_elapsed_physical_time();\n    std::cout << "Elapsed logical time: " << logical_time\n        << " physical time: " << physical_time\n        << " lag: " << physical_time - logical_time <<  std::endl;\n  =}\n}\n
    \n
    target Python\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    t = lf.time.logical_elapsed()\n    T = lf.time.physical_elapsed()\n    print(\n        f"Elapsed logical time: {t}, physical time: {T}, lag: {T-t}"\n    )\n  =}\n}\n
    \n
    target TypeScript\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    const t = util.getElapsedLogicalTime()\n    const T = util.getElapsedPhysicalTime()\n    console.log(`Elapsed logical time: ${t}, physical time: ${T}, lag: ${T.subtract(t)}`)\n  =}\n}\n
    \n
    target Rust\nmain reactor TimeLag {\n  timer t(0, 1 s)\n  reaction(t) {=\n    let t = ctx.get_elapsed_logical_time();\n    let T = ctx.get_elapsed_physical_time();\n    println!(\n      "Elapsed logical time: {}, physical time: {}, lag: {}",\n      t.as_nanos(),\n      T.as_nanos(),\n      (T-t).as_nanos(),\n    );\n  =}\n}\n
    \n

    $end(TimeLag)$

    \n

    Execution will show something like this:

    \n
    Elapsed logical time: 0, physical time: 855000, lag: 855000\nElapsed logical time: 1000000000, physical time: 1004714000, lag: 4714000\nElapsed logical time: 2000000000, physical time: 2004663000, lag: 4663000\nElapsed logical time: 3000000000, physical time: 3000210000, lag: 210000\n...
    \n

    In this case, the lag varies from a few hundred microseconds to a small number of milliseconds. The amount of lag will depend on the execution platform.

    \n

    Simultaneity and Instantaneity

    \n

    If two timers have the same offset and period, then their events are logically simultaneous. No observer will be able to see that one timer has triggered and the other has not.

    \n

    A reaction is always invoked at a well-defined logical time, and logical time does not advance during its execution. Any output produced by the reaction will be logically simultaneous with the input. In other words, reactions are logically instantaneous (for an exception, see Logical Execution Time). Physical time, however, does elapse during execution of a reaction.

    \n

    Timeout

    \n

    By default, a Lingua Franca program will terminate when there are no more events to process. If there is a timer with a non-zero period, then there will always be more events to process, so the default execution will be unbounded. To specify a finite execution horizon, you can either specify a timeout target property or a --timeout command-line option. For example, the following timeout property will cause the above timer with a period of one second to terminate after 11 events:

    \n
    target C {\n  timeout: 10 s\n}\n
    \n
    target Cpp {\n  timeout: 10 s\n}\n
    \n
    target Python {\n  timeout: 10 s\n}\n
    \n
    target TypeScript {\n  timeout: 10 s\n}\n
    \n
    target Rust {\n  timeout: 10 s\n}\n
    \n

    Startup and Shutdown

    \n

    To cause a reaction to be invoked at the start of execution, a special startup trigger is provided:

    \n
    reactor Foo {\n  reaction(startup) {=\n    ... perform initialization ...\n  =}\n}\n
    \n

    The startup trigger is equivalent to a timer with no offset or period.

    \n

    To cause a reaction to be invoked at the end of execution, a special shutdown trigger is provided. Consider the following reactor, commonly used to build regression tests:

    \n

    $start(TestCount)$

    \n
    target C\nreactor TestCount(start: int = 0, stride: int = 1, num_inputs: int = 1) {\n  state count: int = start\n  state inputs_received: int = 0\n  input x: int\n  reaction(x) {=\n    printf("Received %d.\\n", x->value);\n    if (x->value != self->count) {\n      printf("ERROR: Expected %d.\\n", self->count);\n      exit(1);\n    }\n    self->count += self->stride;\n    self->inputs_received++;\n  =}\n  reaction(shutdown) {=\n    printf("Shutdown invoked.\\n");\n    if (self->inputs_received != self->num_inputs) {\n      printf("ERROR: Expected to receive %d inputs, but got %d.\\n",\n          self->num_inputs,\n          self->inputs_received\n      );\n      exit(2);\n    }\n  =}\n}\n
    \n
    target Cpp\nreactor TestCount(start: int = 0, stride: int = 1, num_inputs: int = 1) {\n  state count: int = start\n  state inputs_received: int = 0\n  input x: int\n  reaction(x) {=\n    auto value = *x.get();\n    std::cout << "Received " <<  value << std::endl;\n    if (value != count) {\n      std::cerr << "ERROR: Expected: "<< count << std::endl;\n      exit(1);\n    }\n    count += stride;\n    inputs_received++;\n  =}\n  reaction(shutdown) {=\n    std::cout << "Shutdown invoked." << std::endl;\n    if (inputs_received != num_inputs) {\n      std::cerr << "ERROR: Expected to receive " << num_inputs\n          << " inputs, but got " << inputs_received << std::endl;\n      exit(2);\n    }\n  =}\n}\n
    \n
    target Python\nreactor TestCount(start=0, stride=1, num_inputs=1) {\n  state count = start\n  state inputs_received = 0\n  input x\n  reaction(x) {=\n    print(f"Received {x.value}.")\n    if x.value != self.count:\n      sys.stderr.write(f"ERROR: Expected {self.count}.\\n")\n      exit(1)\n    self.count += self.stride\n    self.inputs_received += 1\n  =}\n  reaction(shutdown) {=\n    print("Shutdown invoked.")\n    if self.inputs_received != self.num_inputs:\n      sys.stderr.write(\n          f"ERROR: Expected to receive {self.num_inputs} inputs, but got {self.inputs_received}.\\n"\n      )\n      exit(2)\n  =}\n}\n
    \n
    target TypeScript\nreactor TestCount(start: number = 0, stride: number = 1, numInputs: number = 1) {\n  state count: number = start\n  state inputsReceived: number = 0\n  input x: number\n  reaction(x) {=\n    console.log(`Received ${x}`)\n    if (x != count) {\n      console.error(`ERROR: Expected ${count}.`)\n      process.exit(1)\n    }\n    count += stride;\n    inputsReceived++\n  =}\n  reaction(shutdown) {=\n    console.log("Shutdown invoked.")\n    if (inputsReceived != numInputs) {\n      console.error(`ERROR: Expected to receive ${numInputs}, but got ${inputsReceived}.`)\n      process.exit(2)\n    }\n  =}\n}\n
    \n
    target Rust\nreactor TestCount(start: u32 = 0, stride: u32 = 1, num_inputs: u32 = 1) {\n  state stride = stride\n  state num_inputs = num_inputs\n  state count: u32 = start\n  state inputs_received: u32 = 0\n  input x: u32\n  reaction(x) {=\n    let x = ctx.get(x).unwrap();\n    println!("Received {}.", x);\n    if x != self.count {\n      println!("ERROR: Expected {}.", self.count);\n      std::process::exit(1);\n    }\n    self.count += self.stride;\n    self.inputs_received += 1;\n  =}\n  reaction(shutdown) {=\n    println!("Shutdown invoked.");\n    if self.inputs_received != self.num_inputs {\n      println!(\n          "ERROR: Expected to receive {} inputs, but got {}.",\n          self.num_inputs,\n          self.inputs_received\n      );\n      std::process::exit(2);\n    }\n  =}\n}\n
    \n

    $end(TestCount)$

    \n

    This reactor tests its inputs against expected values, which are expected to start with the value given by the start parameter and increase by stride with each successive input. It expects to receive a total of num_inputs input events. It checks the total number of inputs received in its $shutdown$ reaction.

    \n

    The shutdown trigger typically occurs at microstep 0, but may occur at a larger microstep. See Superdense Time and Termination.

    ","headings":[{"value":"Logical Time","depth":2},{"value":"Time Values","depth":2},{"value":"Timers","depth":2},{"value":"Elapsed Time","depth":2},{"value":"Comparing Logical and Physical Times","depth":2},{"value":"Simultaneity and Instantaneity","depth":2},{"value":"Timeout","depth":2},{"value":"Startup and Shutdown","depth":2}],"frontmatter":{"permalink":"/docs/handbook/time-and-timers","title":"Time and Timers","oneline":"Time and timers in Lingua Franca.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Parameters and State Variables","oneline":"Parameters and state variables in Lingua Franca.","permalink":"/docs/handbook/parameters-and-state-variables"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Composing Reactors","oneline":"Composing reactors in Lingua Franca.","permalink":"/docs/handbook/composing-reactors"}}}},"pageContext":{"id":"1-time-and-timers","slug":"/docs/handbook/time-and-timers","repoPath":"/packages/documentation/copy/en/topics/Time and Timers.md","previousID":"20781702-b6a5-5a16-b4d8-b4c45cd76fa3","nextID":"834f9d0d-f7c6-5732-8c60-bad1954701f7","lang":"en","modifiedTime":"2023-11-08T00:42:45.532Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/timing-analysis/page-data.json b/page-data/docs/handbook/timing-analysis/page-data.json index d9fc4aaab..79d99facb 100644 --- a/page-data/docs/handbook/timing-analysis/page-data.json +++ b/page-data/docs/handbook/timing-analysis/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/timing-analysis","result":{"data":{"markdownRemark":{"id":"791b4a7b-0b3a-528e-a7be-2cb254c96f50","excerpt":"Examples Precision-Timed Actuation (discussion Dec 2018) Given a time unit c, H3 reacts sporadically >= 100c (e.g., 10, 120, 230, …) H4 reacts periodically with…","html":"

    Examples

    \n

    Precision-Timed Actuation (discussion Dec 2018)

    \n

    Given a time unit c,

    \n
      \n
    • H3 reacts sporadically >= 100c (e.g., 10, 120, 230, …)
    • \n
    • H4 reacts periodically with period 50c (e.g., 0, 50, 100, …)
    • \n
    • Delay adds 100c to the timestamp of each incoming event
    • \n
    • Actuate shall start executing H5 before r.t. clock exceeds time stamp of incoming events
    • \n
    \n
    +--------+\n|        |          +--------+     +-------+     +---------+\n|   H3   +----------> H1     |     |       |     |         |\n|        |          |        +-----> Delay +-----> Actuate |\n+--------+    +-----> H2     |     |  100  |     |   (H5)  |\n              |     +--------+     +-------+     +---------+\n              |\n+--------+    |\n|        |    |\n|   H4   +----+\n|        |\n+--------+
    \n

    We can construct a dependency graph:

    \n
    H3 ---> H1 ---> H2 ---> H5\n            |\nH4 ---------+
    \n

    A feasible schedule requires that:

    \n
      \n
    • WCET(H3) + WCET(H1) + WCET(H2) <= 100c
    • \n
    • WCET(H4) + WCET(H1) + WCET(H2) <= 100c
    • \n
    \n

    Preemption Example

    \n
              T = 1s          C = 300ms\n       +---------+      +----------+\n       |         |      |   Corr   |\n       |   GPS   +------> r1       +----+\n       |         |      |          |    |\n       +---------+      +----------+    |     +----------------+         D = 100ms\n                                        |     |                |        +--------+\n         T = 100ms                      +-----> r2             |        |        |\n       +---------+                            |       Ctrl     +-------->  Act.  |\n       |         |           +----------------> r3             |        |        |\n       |  IMU    +-----------+                |                |        +--------+\n       |         |                            +----------------+\n       +---------+
    \n

    This example needs the following:

    \n
      \n
    • r3 needs to preempt r1.
    • \n
    • The event from GPS needs a delay of 300ms between Corr and Ctrl, so Ctrl never sees an older event.
    • \n
    \n

    If we want to avoid preemption, as this hurts WCET analysis:

    \n
      \n
    • Split reactor Corr. into three (or more) reactors and add a delay of 100 ms after each one.
    • \n
    \n

    For both solutions, the scheduler needs a “safe to process” analysis for reaction r3 to execute while r1 is\nstill executing for an older time-stamped event.

    \n

    Preemption can be avoided when there are enough cores (or hardware threads in PRET) available to execute r1 and r3 concurrently.

    ","headings":[{"value":"Examples","depth":1},{"value":"Precision-Timed Actuation (discussion Dec 2018)","depth":2},{"value":"Preemption Example","depth":2}],"frontmatter":{"permalink":"/docs/handbook/timing-analysis","title":"Timing Analysis","oneline":"Timing Analysis.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/timing-analysis","repoPath":"/packages/documentation/copy/en/less-developed/Timing Analysis.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/timing-analysis","result":{"data":{"markdownRemark":{"id":"791b4a7b-0b3a-528e-a7be-2cb254c96f50","excerpt":"Examples Precision-Timed Actuation (discussion Dec 2018) Given a time unit c, H3 reacts sporadically >= 100c (e.g., 10, 120, 230, …) H4 reacts periodically with…","html":"

    Examples

    \n

    Precision-Timed Actuation (discussion Dec 2018)

    \n

    Given a time unit c,

    \n
      \n
    • H3 reacts sporadically >= 100c (e.g., 10, 120, 230, …)
    • \n
    • H4 reacts periodically with period 50c (e.g., 0, 50, 100, …)
    • \n
    • Delay adds 100c to the timestamp of each incoming event
    • \n
    • Actuate shall start executing H5 before r.t. clock exceeds time stamp of incoming events
    • \n
    \n
    +--------+\n|        |          +--------+     +-------+     +---------+\n|   H3   +----------> H1     |     |       |     |         |\n|        |          |        +-----> Delay +-----> Actuate |\n+--------+    +-----> H2     |     |  100  |     |   (H5)  |\n              |     +--------+     +-------+     +---------+\n              |\n+--------+    |\n|        |    |\n|   H4   +----+\n|        |\n+--------+
    \n

    We can construct a dependency graph:

    \n
    H3 ---> H1 ---> H2 ---> H5\n            |\nH4 ---------+
    \n

    A feasible schedule requires that:

    \n
      \n
    • WCET(H3) + WCET(H1) + WCET(H2) <= 100c
    • \n
    • WCET(H4) + WCET(H1) + WCET(H2) <= 100c
    • \n
    \n

    Preemption Example

    \n
              T = 1s          C = 300ms\n       +---------+      +----------+\n       |         |      |   Corr   |\n       |   GPS   +------> r1       +----+\n       |         |      |          |    |\n       +---------+      +----------+    |     +----------------+         D = 100ms\n                                        |     |                |        +--------+\n         T = 100ms                      +-----> r2             |        |        |\n       +---------+                            |       Ctrl     +-------->  Act.  |\n       |         |           +----------------> r3             |        |        |\n       |  IMU    +-----------+                |                |        +--------+\n       |         |                            +----------------+\n       +---------+
    \n

    This example needs the following:

    \n
      \n
    • r3 needs to preempt r1.
    • \n
    • The event from GPS needs a delay of 300ms between Corr and Ctrl, so Ctrl never sees an older event.
    • \n
    \n

    If we want to avoid preemption, as this hurts WCET analysis:

    \n
      \n
    • Split reactor Corr. into three (or more) reactors and add a delay of 100 ms after each one.
    • \n
    \n

    For both solutions, the scheduler needs a “safe to process” analysis for reaction r3 to execute while r1 is\nstill executing for an older time-stamped event.

    \n

    Preemption can be avoided when there are enough cores (or hardware threads in PRET) available to execute r1 and r3 concurrently.

    ","headings":[{"value":"Examples","depth":1},{"value":"Precision-Timed Actuation (discussion Dec 2018)","depth":2},{"value":"Preemption Example","depth":2}],"frontmatter":{"permalink":"/docs/handbook/timing-analysis","title":"Timing Analysis","oneline":"Timing Analysis.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/timing-analysis","repoPath":"/packages/documentation/copy/en/less-developed/Timing Analysis.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/tools/page-data.json b/page-data/docs/handbook/tools/page-data.json index 54d175461..5a43c429f 100644 --- a/page-data/docs/handbook/tools/page-data.json +++ b/page-data/docs/handbook/tools/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/tools","result":{"data":{"markdownRemark":{"id":"61cd9ca1-e842-5a49-9f75-b0310f8cec4e","excerpt":"IDE integration The idea is to build a language server to facilitate the integration with a variety of editors/IDEs. See Language Server Protocol (LSP) for more…","html":"

    IDE integration

    \n

    The idea is to build a language server to facilitate the integration with a variety of editors/IDEs. See Language Server Protocol (LSP) for more information.

    \n
                     +-------------------------------------+\n+--------+       |  +----------+         +----------+  |\n|        +-------|-->    LF    +--------->  Target  |  |\n| Editor |  src  |  | Compiler | gen src | Compiler |  |\n|        <-------|--+          <---------+          |  |\n+--------+  err  |  +----------+ gen err +----------+  |\n                 |          Language Server            |\n                 +-------------------------------------+
    \n

    If the LF compiler encounters any syntax errors, it will report them to the editor (the language client). If the LF code compiles, the output will be sent to the target compiler. If the target compiler reports any errors, these, too, will be reported to the editor via the language server. The tricky part is to match target language errors to LF source locations; the language server will have to do some bookkeeping.

    ","headings":[{"value":"IDE integration","depth":1}],"frontmatter":{"permalink":"/docs/handbook/tools","title":"Tools","oneline":"LF Tools.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/tools","repoPath":"/packages/documentation/copy/en/less-developed/Tools.md","lang":"en","modifiedTime":"2023-11-03T01:50:26.533Z"}},"staticQueryHashes":[]} \ No newline at end of file +{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/tools","result":{"data":{"markdownRemark":{"id":"61cd9ca1-e842-5a49-9f75-b0310f8cec4e","excerpt":"IDE integration The idea is to build a language server to facilitate the integration with a variety of editors/IDEs. See Language Server Protocol (LSP) for more…","html":"

    IDE integration

    \n

    The idea is to build a language server to facilitate the integration with a variety of editors/IDEs. See Language Server Protocol (LSP) for more information.

    \n
                     +-------------------------------------+\n+--------+       |  +----------+         +----------+  |\n|        +-------|-->    LF    +--------->  Target  |  |\n| Editor |  src  |  | Compiler | gen src | Compiler |  |\n|        <-------|--+          <---------+          |  |\n+--------+  err  |  +----------+ gen err +----------+  |\n                 |          Language Server            |\n                 +-------------------------------------+
    \n

    If the LF compiler encounters any syntax errors, it will report them to the editor (the language client). If the LF code compiles, the output will be sent to the target compiler. If the target compiler reports any errors, these, too, will be reported to the editor via the language server. The tricky part is to match target language errors to LF source locations; the language server will have to do some bookkeeping.

    ","headings":[{"value":"IDE integration","depth":1}],"frontmatter":{"permalink":"/docs/handbook/tools","title":"Tools","oneline":"LF Tools.","preamble":""}},"prev":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}},"next":{"childMarkdownRemark":{"frontmatter":{"title":"Contributing","oneline":"Contribute to Lingua Franca.","permalink":"/docs/handbook/contributing"}}}},"pageContext":{"slug":"/docs/handbook/tools","repoPath":"/packages/documentation/copy/en/less-developed/Tools.md","lang":"en","modifiedTime":"2023-11-08T00:42:45.520Z"}},"staticQueryHashes":[]} \ No newline at end of file diff --git a/page-data/docs/handbook/tracing/page-data.json b/page-data/docs/handbook/tracing/page-data.json index 43418f849..e6937c9f0 100644 --- a/page-data/docs/handbook/tracing/page-data.json +++ b/page-data/docs/handbook/tracing/page-data.json @@ -1 +1 @@ -{"componentChunkName":"component---src-templates-documentation-tsx","path":"/docs/handbook/tracing","result":{"data":{"markdownRemark":{"id":"7f0a051c-f9a3-5d83-b042-6408ba55e8a0","excerpt":"Tracing is a powerful tool when it comes to analysis and debugging of applications. Unfortunately, most tracing tools that are readily available are designed…","html":"

    Tracing is a powerful tool when it comes to analysis and debugging of applications. Unfortunately, most tracing tools that are readily available are designed specifically for analyzing processes, threads and system calls. Specialized tools are required to enable analysis that is tailored to an alternative model of computation such as Reactors. The tools should be capable of understanding the fundamental concepts of the model, such as the distinction between logical and physical time, as well as structural units such as reactors and reactions. This page gives an overview of the currently supported trace mechanism, as well as an outline of alternative tools that could be useful in the future.

    \n

    Tracing is different from logging. Logging produces human-readable output in textual form and incurs significant overhead. Tracing produces binary data that must be further processed to be useful and is designed to have minimal impact on the execution time of a program.

    \n

    Tracing is currently supported in the C, Python, and C++ targets. The mechanism used in C and Python is different from that used in C++. Tracing in C++ requires third-party tools that may only be available in Linux. Tracing in C and Python does not require any third-party tools.

    \n
    \n

    Tracing in C++

    \n

    Tracing in the C++ target of Lingua Franca is based on three third-party tools. LTTng is a Linux tool used to instrument the Lingua Franca program and to record traces in the CTF, which minimizes the overhead of instrumentation. Chrome (or Chromium) has a build in trace viewer that is used to visualize the recorded trace data in a reactor-specific way. Since the Chrome trace-viewer cannot read CTF traces directly, we use Babeltrace2 to convert the recorded CTF trace to a JSON file that the Google trace viewer can load.

    \n

    Usage

    \n

    Some helper scripts that we will use below, can be found in the reactor-cpp repository.

    \n
      \n
    1. \n

      Build and install the user space tools of LTTng (lttng-ust) as described here. On Arch, there is a community package available pacman -Sy lttng-ust. On Ubuntu, you need to install lttng-tools, lttng-modules-dkms, and liblttng-ust-dev

      \n
    2. \n
    3. \n

      Build and install Babeltrace2 and its python bindings as described here. In most cases, the following steps should work:

      \n
        \n
      1. git clone --branch v2.0.4 git@github.com:efficios/babeltrace.git
      2. \n
      3. cd babeltrace
      4. \n
      5. ./bootstrap
      6. \n
      7. ./configure --prefix=/path/to/preferred/install/location --enable-python-bindings --disable-debug-info --disable-man-pages
      8. \n
      9. make install
      10. \n
      \n
    4. \n
    5. \n

      Make sure babeltrace is available on your path:

      \n
        \n
      1. export PYTHONPATH=${PYTHONPATH}:/path/to/preferred/install/location/lib/python3.8/site-packages
      2. \n
      3. export LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/path/to/preferred/install/location/lib
      4. \n
      \n
    6. \n
    7. \n

      Modify the target declaration of your Lingua Franca program to enable tracing:

      \n
    8. \n
    \n
    target Cpp {\n    tracing: true\n};\n
    \n
      \n
    1. Build the Lingua Franca program. The current build process is not very robust and does not automatically rebuild the reactor-cpp framework with tracing support if an earlier build exist. Be sure to remove all build artifacts rm -r bin build include lib share src-gen before triggering a new build. Alternatively, if you compile with lfc, you can add -c to the command to clean before building.
    2. \n
    3. Start a LTTng user space session by simply running the start_tracing.sh script. This will print the directory in which the recorded traces will be placed.
    4. \n
    5. Run your instrumented Lingua Franca application.
    6. \n
    7. Stop the LTTng session using stop_tracing.sh.
    8. \n
    9. Convert the recorded CTF trace to a JSON file using ctf_to_json.py <lttng-session-dir>. <lttng-session-dir> is the output directory reported by start_tracing.sh. By default, this produces a file trace.json. Optionally, the default output file can be overridden using -o or --output.
    10. \n
    11. Open Chrome (or Chromium) and go to about://tracing. Load the previously generated JSON file to visualize it.
    12. \n
    \n

    The Trace View

    \n

    \"Screenshot_20200512_165849\"

    \n

    The trace visualization consists of two parts. The upper part (labeled Execution), shows the physical time at which reactions are executed by the scheduler or by its worker threads. The lower parts show the reactors of the program. For each reactor, all scheduled actions (red markers) and all triggered reactions (blue markers) are visualized by their logical time. All elements in the trace view can be clicked on to display more detailed information.

    \n

    Supporting Tracing in Other Targets

    \n

    The same mechanism as described above can be used to trace Lingua Franca applications in other target languages. The key is to instrument the target runtime in order to produce CTF traces. LTTng comes with support for multiple languages. As a fallback solution, C can be used to define the trace points which then can be used from the target language through a foreign function interface. It should also be considered, to use an alternative library in the target language that is capable of producing CTF traces. The only requirement is that the generated CTF events have a similar structure, as it is currently used in the C++ target. See trace.hh in reactor-cpp to get an overview of the available trace points.

    \n

    Trace Viewers

    \n

    This section gives a brief overview of trace viewers that could be applicable for tracing Lingua Franca applications.

    \n

    Google Trace Viewer

    \n

    The Google Trace Viewer is the only viewer currently supported. Since it reads JSON files, it is easy to use and a conversion script can easily tailor the trace data such that it is correctly displayed by the viewer. Documentation of the JSON trace format can be found here. There is also a list of available color codes. The approach of using LTTng for tracing, a converter and Google Trace Viewer can also be used to sample and visualize data live. This is shown in the Scalapus project.

    \n

    Trace Compass

    \n

    Trace Compass is based on Eclipse and has native support for CTF traces. Due to the Eclipse support, Trace Compass is a natural candidate for integration with the Lingua Franca IDE. However, Trace Compass is tailored for the visualization of kernel traces or running user processes. Out of the box, it cannot make sense of reactor traces. There are various ways to customize Trace Compass in order to properly display the data, but they are difficult to use and/or not well documented.

    \n
      \n
    1. Custom trace analysis and views can be defined in an XML format. This probably works well for smaller tasks, but programming a mildly complex analysis in XML without proper error checking and debugging appears ridiculous.
    2. \n
    3. Custom analysis and views can also be programmed in scripting languages such as Python or Java Script. However, the tools seem not to be ready for production. I was not able to get the Python support working. Using Java Script the tools worked for simple tasks, but I was not able to come to a satisfactory results due to lack of or incomplete documentation and low customizability.
    4. \n
    5. Writing a Trace Compass Plugin. This seems to be the most promising approach, but probably requires a considerable amount of work. Writing such a plugin and integrating it with the Lingua Franca IDE could be a nice student project though.
    6. \n
    \n

    Vampir

    \n

    Vampir is another powerful tracing tool that is mainly developed at TU Dresden. It targets mostly HPC applications, and I am not sure if it can be adjusted to display specific information other than details of processes and threads.

    \n
    \n
    \n

    Tracing in C and Python

    \n

    The C and Python tracing mechanism depends only on the availability of the pthread library. Like C++ tracing, tracing is enabled by a target parameter:

    \n
    target C {\n    tracing: true\n};\n
    \n

    Once it is enabled, when the compiled program, say Foo.lf, is executed, a trace file is created, Foo.lft (the extension is for “Lingua Franca trace”). If you wish to customize the root name of the trace file, you can specify the following target property instead:

    \n
    target C {\n    tracing: {trace-file-name: "Bar"}\n};\n
    \n

    This will result in the trace file being named Bar.lft, regardless of the name of the .lf file.

    \n

    The trace file is a binary file. It is not human readable. There are utilities for reading it:

    \n
      \n
    • trace_to_csv: This program creates a text file with one line per traced event in comma-separated list format.
    • \n
    • trace_to_chrome: This program creates a text file in JSON format that is suitable for reading into the same Google Trace Viewer, which runs in Chrome, as used above in C++ tracing.
    • \n
    • trace_to_influxdb : This program will send the traced event to a running InfluxDB database server.
    • \n
    • fedsd: This program creates a timed sequence diagram showing the interactions between components of a federated program (see Tracing Federated Programs below).
    • \n
    \n

    These four programs are located in reactor-c at lingua-franca/core/src/main/resources/lib/c/reactor-c/util/tracing. Running sudo make install in that directory will put executables into usr/local/bin.

    \n

    Consider for example the ThreadedThreaded.lf test, which executes a number of heavyweight computations in parallel on multiple cores. If you enable tracing as shown above and run the program, a ThreadedTheread.lft file will appear. Running

    \n
       trace_to_csv ThreadedThreaded\n
    \n

    will create a file called ThreadedThreaded.csv that looks like this:

    \n
    Event, Reactor, Reaction, Worker, Elapsed Logical Time, Microstep, Elapsed Physical Time, Trigger, Extra Delay\nSchedule called, a, 0, 0, 0, 0, 704000, a.t, 0\nSchedule called, a, 0, 0, 0, 0, 704000, a.t, 200000000\nSchedule called, a, 0, 0, 200000000, 0, 177916000, a.t, 200000000\n...\nReaction starts, a, 0, 1, 0, 0, 765000, NO TRIGGER, 0\nReaction ends, a, 0, 1, 0, 0, 765000, NO TRIGGER, 0\nReaction starts, t[0], 0, 1, 0, 0, 793000, NO TRIGGER, 0\nReaction ends, t[0], 0, 1, 0, 0, 177520000, NO TRIGGER, 0\nReaction starts, t[3], 0, 1, 200000000, 0, 177955000, NO TRIGGER, 0\nReaction ends, t[3], 0, 1, 200000000, 0, 348602000, NO TRIGGER, 0
    \n

    The first line defines each of the columns. For example, the second line records a call to lf_schedule() for reactor named a, with no associated reaction, in worker thread 0, at (elapsed) logical time 0 with microstep 0. The call occurred at (elapsed) physical time\n704 microseconds and is scheduling the trigger named a.t (a timer) with extra delay 0. This file can be imported into any spreadsheet program and sorted and analyzed.

    \n

    The trace_to_csv utility will also create a summary file called ThreadedThreaded_summary.csv that looks like this after importing in Excel:

    \n

    \n \n \n

    \n

    If you call

    \n
       trace_to_chrome ThreadedThreaded\n
    \n

    then a ThreadedThreaded.json file is created. To visualize the data, point your Chrome browser to chrome://tracing/. Click on the Load button and select the .json file that you just created. The result should look something like this:

    \n

    \n \n \n

    \n

    The tan-colored regions whose labels start with “A” and “W” represent time spent advancing logical time and waiting for activity on the reaction queue, respectively. When logical time advances, unless you have specified the -fast option, one of the worker threads blocks execution until physical time catches up with logical time. The remaining worker threads block waiting for reactions that are ready to execute appear on the reaction queue.

    \n

    The JSON trace format can be found here. There is also a list of available color codes.

    \n

    User-Defined Tracepoints

    \n

    Users can add their own tracepoints in order to provide low-overhead recording of events and events with values that occur during the execution of reactions. To do this, the first step is to register the trace event in a startup reaction as follows:

    \n
        reaction(startup) {=\n        if (!register_user_trace_event("Description of event")) {\n            fprintf(stderr, "ERROR: Failed to register trace event.\\n");\n            exit(1);\n        }\n    =}\n
    \n

    The description of the event is an arbitrary string, but the string must be unique. All events with the same description will be collected together in any display of events.

    \n

    To then actually record an event, in a reaction, call tracepoint_user_event, passing it the same string. E.g.,

    \n
    \treaction(in) -> out {=\n\t    ...\n\t    tracepoint_user_event("Description of event");\n\t    ...\n\t=}\n
    \n

    You can also pass a value to the trace. The type of the value is long long, so it can be a time value or an int. For example,

    \n
    \treaction(in) -> out {=\n\t    ...\n\t    tracepoint_user_value("Description of event", 42);\n\t    ...\n\t=}\n
    \n

    An example of a Chrome display of a run of the Tracing regression test is here:

    \n

    \n \n \n

    \n

    In this image, “Number of Destination invocations” is an event description to\nwhich values 1 through 10 were passed. This results in the shaded value plot\nshown first. The other four rows are just pure events (with no value). They are\nshown by (extremely) thin lines positioned at the physical time of the\noccurrence of the event. Dragging the mouse over those thin lines shows further\ndetails about the event in the window below.

    \n

    Tracing Federated Programs

    \n

    When the tracing target parameter is set to true in a federated program, then each federate plus the RTI will generate a binary trace file. The utility fedsd generates an HTML file containing an SVG graphic that shows the messages exchanged between components over time. Like the other utilities, fedsd is defined in lingua_franca/util/tracing and installed using make install.

    \n

    Consider the following LF program:

    \n\"Feedback\n

    Setting tracing: true in this program and running it produces four .lft files. Running fedsd on those files:

    \n
       fedsd
    \n

    results in converting the files to .csv files and then generating a trace_svg.html file. Opening that file reveals a trace, the beginning of which looks like this:

    \n

    \n \n \n

    \n

    If you call

    \n
       trace_to_chrome ThreadedThreaded\n
    \n

    then a ThreadedThreaded.json file is created. To visualize the data, point your Chrome browser to chrome://tracing/. Click on the Load button and select the .json file that you just created. The result should look something like this:

    \n

    \n \n \n

    \n

    The tan-colored regions whose labels start with “A” and “W” represent time spent advancing logical time and waiting for activity on the reaction queue, respectively. When logical time advances, unless you have specified the -fast option, one of the worker threads blocks execution until physical time catches up with logical time. The remaining worker threads block waiting for reactions that are ready to execute appear on the reaction queue.

    \n

    The JSON trace format can be found here. There is also a list of available color codes.

    \n

    User-Defined Tracepoints

    \n

    Users can add their own tracepoints in order to provide low-overhead recording of events and events with values that occur during the execution of reactions. To do this, the first step is to register the trace event in a startup reaction as follows:

    \n
        reaction(startup) {=\n        if (!register_user_trace_event("Description of event")) {\n            fprintf(stderr, "ERROR: Failed to register trace event.\\n");\n            exit(1);\n        }\n    =}\n
    \n

    The description of the event is an arbitrary string, but the string must be unique. All events with the same description will be collected together in any display of events.

    \n

    To then actually record an event, in a reaction, call tracepoint_user_event, passing it the same string. E.g.,

    \n
    \treaction(in) -> out {=\n\t    ...\n\t    tracepoint_user_event("Description of event");\n\t    ...\n\t=}\n
    \n

    You can also pass a value to the trace. The type of the value is long long, so it can be a time value or an int. For example,

    \n
    \treaction(in) -> out {=\n\t    ...\n\t    tracepoint_user_value("Description of event", 42);\n\t    ...\n\t=}\n
    \n

    An example of a Chrome display of a run of the Tracing regression test is here:

    \n

    \n \n \n

    \n

    In this image, “Number of Destination invocations” is an event description to\nwhich values 1 through 10 were passed. This results in the shaded value plot\nshown first. The other four rows are just pure events (with no value). They are\nshown by (extremely) thin lines positioned at the physical time of the\noccurrence of the event. Dragging the mouse over those thin lines shows further\ndetails about the event in the window below.

    \n

    Tracing Federated Programs

    \n

    When the tracing target parameter is set to true in a federated program, then each federate plus the RTI will generate a binary trace file. The utility fedsd generates an HTML file containing an SVG graphic that shows the messages exchanged between components over time. Like the other utilities, fedsd is defined in lingua_franca/util/tracing and installed using make install.

    \n

    Consider the following LF program:

    \n\"Feedback\n

    Setting tracing: true in this program and running it produces four .lft files. Running fedsd on those files:

    \n
       fedsd
    \n

    results in converting the files to .csv files and then generating a trace_svg.html file. Opening that file reveals a trace, the beginning of which looks like this:

    \n

    \n

    Skip to main content

    Note: Lingua Franca is an evolving language, and the older papers below may use a syntax that does not match the current syntax. Nevertheless, these papers are useful for understanding the principles.

    Published Papers

    In reverse chronological order:

    Presentations

    Press Coverage

    Learning Resources

    \ No newline at end of file +
    Skip to main content

    Note: Lingua Franca is an evolving language, and the older papers below may use a syntax that does not match the current syntax. Nevertheless, these papers are useful for understanding the principles.

    Published Papers

    In reverse chronological order:

    Presentations

    Press Coverage

    Learning Resources

    \ No newline at end of file diff --git a/sitemap/sitemap-0.xml b/sitemap/sitemap-0.xml index a9ad50de5..98442c14b 100644 --- a/sitemap/sitemap-0.xml +++ b/sitemap/sitemap-0.xml @@ -1 +1 @@ -https://www.lf-lang.org/docs/handbook/contributingdaily0.7https://www.lf-lang.org/docs/handbook/eclipse-oomphdaily0.7https://www.lf-lang.org/docs/handbook/intellijdaily0.7https://www.lf-lang.org/docs/handbook/running-benchmarksdaily0.7https://www.lf-lang.org/docs/handbook/website-developmentdaily0.7https://www.lf-lang.org/docs/handbook/developer-setupdaily0.7https://www.lf-lang.org/docs/handbook/regression-testsdaily0.7https://www.lf-lang.org/docs/handbook/arduinodaily0.7https://www.lf-lang.org/docs/handbook/proof-importdaily0.7https://www.lf-lang.org/docs/handbook/logical-execution-timedaily0.7https://www.lf-lang.org/docs/handbook/related-workdaily0.7https://www.lf-lang.org/docs/handbook/timing-analysisdaily0.7https://www.lf-lang.org/docs/handbook/toolsdaily0.7https://www.lf-lang.org/docs/handbook/language-specificationdaily0.7https://www.lf-lang.org/docs/handbook/generic-types-interfaces-inheritancedaily0.7https://www.lf-lang.org/docs/handbook/import-systemdaily0.7https://www.lf-lang.org/docs/handbook/reactors-on-patmosdaily0.7https://www.lf-lang.org/docs/handbook/featuresdaily0.7https://www.lf-lang.org/docs/handbook/containerized-executiondaily0.7https://www.lf-lang.org/docs/handbook/expressionsdaily0.7https://www.lf-lang.org/docs/handbook/securitydaily0.7https://www.lf-lang.org/docs/handbook/zephyrdaily0.7https://www.lf-lang.org/docs/handbook/target-declarationdaily0.7https://www.lf-lang.org/docs/handbook/code-extensiondaily0.7https://www.lf-lang.org/docs/handbook/tracingdaily0.7https://www.lf-lang.org/docs/handbook/a-first-reactordaily0.7https://www.lf-lang.org/docs/handbook/actionsdaily0.7https://www.lf-lang.org/docs/handbook/causality-loopsdaily0.7https://www.lf-lang.org/docs/handbook/composing-reactorsdaily0.7https://www.lf-lang.org/docs/handbook/deadlinesdaily0.7https://www.lf-lang.org/docs/handbook/command-line-toolsdaily0.7https://www.lf-lang.org/docs/handbook/distributed-executiondaily0.7https://www.lf-lang.org/docs/handbook/extending-reactorsdaily0.7https://www.lf-lang.org/docs/handbook/genericsdaily0.7https://www.lf-lang.org/docs/handbook/troubleshootingdaily0.7https://www.lf-lang.org/docs/handbook/epoch-idedaily0.7https://www.lf-lang.org/docs/handbook/inputs-and-outputsdaily0.7https://www.lf-lang.org/docs/handbook/methodsdaily0.7https://www.lf-lang.org/docs/handbook/modal-modelsdaily0.7https://www.lf-lang.org/docs/handbook/multiports-and-banksdaily0.7https://www.lf-lang.org/docs/handbook/overviewdaily0.7https://www.lf-lang.org/docs/handbook/parameters-and-state-variablesdaily0.7https://www.lf-lang.org/docs/handbook/preamblesdaily0.7https://www.lf-lang.org/docs/handbook/reaction-declarationsdaily0.7https://www.lf-lang.org/docs/handbook/reactionsdaily0.7https://www.lf-lang.org/docs/handbook/superdense-timedaily0.7https://www.lf-lang.org/docs/handbook/terminationdaily0.7https://www.lf-lang.org/docs/handbook/time-and-timersdaily0.7https://www.lf-lang.org/docs/handbook/tutorial-videodaily0.7https://www.lf-lang.org/docs/handbook/target-language-detailsdaily0.7https://www.lf-lang.org/communitydaily0.7https://www.lf-lang.org/downloaddaily0.7https://www.lf-lang.org/emptydaily0.7https://www.lf-lang.org/daily0.7https://www.lf-lang.org/publications-and-presentationsdaily0.7https://www.lf-lang.org/docs/daily0.7https://www.lf-lang.org/docs/handbook/daily0.7 \ No newline at end of file +https://www.lf-lang.org/docs/handbook/contributingdaily0.7https://www.lf-lang.org/docs/handbook/eclipse-oomphdaily0.7https://www.lf-lang.org/docs/handbook/intellijdaily0.7https://www.lf-lang.org/docs/handbook/running-benchmarksdaily0.7https://www.lf-lang.org/docs/handbook/website-developmentdaily0.7https://www.lf-lang.org/docs/handbook/developer-setupdaily0.7https://www.lf-lang.org/docs/handbook/regression-testsdaily0.7https://www.lf-lang.org/docs/handbook/arduinodaily0.7https://www.lf-lang.org/docs/handbook/proof-importdaily0.7https://www.lf-lang.org/docs/handbook/logical-execution-timedaily0.7https://www.lf-lang.org/docs/handbook/zephyrdaily0.7https://www.lf-lang.org/docs/handbook/related-workdaily0.7https://www.lf-lang.org/docs/handbook/timing-analysisdaily0.7https://www.lf-lang.org/docs/handbook/toolsdaily0.7https://www.lf-lang.org/docs/handbook/language-specificationdaily0.7https://www.lf-lang.org/docs/handbook/generic-types-interfaces-inheritancedaily0.7https://www.lf-lang.org/docs/handbook/import-systemdaily0.7https://www.lf-lang.org/docs/handbook/reactors-on-patmosdaily0.7https://www.lf-lang.org/docs/handbook/featuresdaily0.7https://www.lf-lang.org/docs/handbook/expressionsdaily0.7https://www.lf-lang.org/docs/handbook/securitydaily0.7https://www.lf-lang.org/docs/handbook/containerized-executiondaily0.7https://www.lf-lang.org/docs/handbook/target-declarationdaily0.7https://www.lf-lang.org/docs/handbook/tracingdaily0.7https://www.lf-lang.org/docs/handbook/code-extensiondaily0.7https://www.lf-lang.org/docs/handbook/a-first-reactordaily0.7https://www.lf-lang.org/docs/handbook/actionsdaily0.7https://www.lf-lang.org/docs/handbook/causality-loopsdaily0.7https://www.lf-lang.org/docs/handbook/composing-reactorsdaily0.7https://www.lf-lang.org/docs/handbook/deadlinesdaily0.7https://www.lf-lang.org/docs/handbook/distributed-executiondaily0.7https://www.lf-lang.org/docs/handbook/extending-reactorsdaily0.7https://www.lf-lang.org/docs/handbook/genericsdaily0.7https://www.lf-lang.org/docs/handbook/inputs-and-outputsdaily0.7https://www.lf-lang.org/docs/handbook/methodsdaily0.7https://www.lf-lang.org/docs/handbook/modal-modelsdaily0.7https://www.lf-lang.org/docs/handbook/command-line-toolsdaily0.7https://www.lf-lang.org/docs/handbook/epoch-idedaily0.7https://www.lf-lang.org/docs/handbook/multiports-and-banksdaily0.7https://www.lf-lang.org/docs/handbook/troubleshootingdaily0.7https://www.lf-lang.org/docs/handbook/overviewdaily0.7https://www.lf-lang.org/docs/handbook/parameters-and-state-variablesdaily0.7https://www.lf-lang.org/docs/handbook/preamblesdaily0.7https://www.lf-lang.org/docs/handbook/reaction-declarationsdaily0.7https://www.lf-lang.org/docs/handbook/reactionsdaily0.7https://www.lf-lang.org/docs/handbook/superdense-timedaily0.7https://www.lf-lang.org/docs/handbook/terminationdaily0.7https://www.lf-lang.org/docs/handbook/time-and-timersdaily0.7https://www.lf-lang.org/docs/handbook/tutorial-videodaily0.7https://www.lf-lang.org/docs/handbook/target-language-detailsdaily0.7https://www.lf-lang.org/communitydaily0.7https://www.lf-lang.org/downloaddaily0.7https://www.lf-lang.org/emptydaily0.7https://www.lf-lang.org/daily0.7https://www.lf-lang.org/publications-and-presentationsdaily0.7https://www.lf-lang.org/docs/daily0.7https://www.lf-lang.org/docs/handbook/daily0.7 \ No newline at end of file diff --git a/webpack-runtime-e4cdf7e8b18db5d0f606.js b/webpack-runtime-959e45056baa6f96280c.js similarity index 97% rename from webpack-runtime-e4cdf7e8b18db5d0f606.js rename to webpack-runtime-959e45056baa6f96280c.js index d645f70ce..4dbe3c54b 100644 --- a/webpack-runtime-e4cdf7e8b18db5d0f606.js +++ b/webpack-runtime-959e45056baa6f96280c.js @@ -1,2 +1,2 @@ -!function(){"use strict";var e,t,n,r,o,a={},i={};function c(e){var t=i[e];if(void 0!==t)return t.exports;var n=i[e]={exports:{}};return a[e](n,n.exports,c),n.exports}c.m=a,e=[],c.O=function(t,n,r,o){if(!n){var a=1/0;for(f=0;f=o)&&Object.keys(c.O).every((function(e){return c.O[e](n[u])}))?n.splice(u--,1):(i=!1,o0&&e[f-1][2]>o;f--)e[f]=e[f-1];e[f]=[n,r,o]},c.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return c.d(t,{a:t}),t},c.d=function(e,t){for(var n in t)c.o(t,n)&&!c.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})},c.f={},c.e=function(e){return Promise.all(Object.keys(c.f).reduce((function(t,n){return c.f[n](e,t),t}),[]))},c.u=function(e){return{56:"component---src-templates-pages-index-tsx",175:"component---src-templates-pages-docs-handbook-index-tsx",208:"90c0253ea39b81a5e49d137436e825ead0ef33a7",248:"component---src-templates-pages-download-tsx",306:"component---src-templates-pages-empty-tsx",351:"commons",517:"component---src-templates-documentation-tsx",532:"styles",533:"component---src-templates-pages-community-tsx",542:"component---src-templates-pages-publications-and-presentations-tsx",616:"component---src-templates-pages-docs-index-tsx"}[e]+"-"+{56:"0a7639f51e33385eb8f9",175:"4aa0b654197d1d5fae53",208:"7299e993a3830cab1f9e",248:"dbe46be65ee8b6b2b60f",306:"167cd7a4c2900b63ac14",351:"c37c3719903bf568c839",517:"04d26b5abab5434898b4",532:"9fa542d257a161f411e7",533:"311a2a5f18f0af45971a",542:"f09ec94dd1ed2f74a804",616:"b914ffbd927f75fc8fee"}[e]+".js"},c.miniCssF=function(e){return"styles.8b00b2cda1c9b4bc348f.css"},c.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),c.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},t={},n="lingua-franca:",c.l=function(e,r,o,a){if(t[e])t[e].push(r);else{var i,u;if(void 0!==o)for(var s=document.getElementsByTagName("script"),f=0;f=o)&&Object.keys(c.O).every((function(e){return c.O[e](n[u])}))?n.splice(u--,1):(i=!1,o0&&e[f-1][2]>o;f--)e[f]=e[f-1];e[f]=[n,r,o]},c.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return c.d(t,{a:t}),t},c.d=function(e,t){for(var n in t)c.o(t,n)&&!c.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})},c.f={},c.e=function(e){return Promise.all(Object.keys(c.f).reduce((function(t,n){return c.f[n](e,t),t}),[]))},c.u=function(e){return{56:"component---src-templates-pages-index-tsx",175:"component---src-templates-pages-docs-handbook-index-tsx",208:"90c0253ea39b81a5e49d137436e825ead0ef33a7",248:"component---src-templates-pages-download-tsx",306:"component---src-templates-pages-empty-tsx",351:"commons",517:"component---src-templates-documentation-tsx",532:"styles",533:"component---src-templates-pages-community-tsx",542:"component---src-templates-pages-publications-and-presentations-tsx",616:"component---src-templates-pages-docs-index-tsx"}[e]+"-"+{56:"0a7639f51e33385eb8f9",175:"4aa0b654197d1d5fae53",208:"7299e993a3830cab1f9e",248:"dbe46be65ee8b6b2b60f",306:"167cd7a4c2900b63ac14",351:"192078f8b1bf52769f18",517:"b5d67be0887eb73cf7b6",532:"9fa542d257a161f411e7",533:"311a2a5f18f0af45971a",542:"f09ec94dd1ed2f74a804",616:"b914ffbd927f75fc8fee"}[e]+".js"},c.miniCssF=function(e){return"styles.8b00b2cda1c9b4bc348f.css"},c.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),c.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},t={},n="lingua-franca:",c.l=function(e,r,o,a){if(t[e])t[e].push(r);else{var i,u;if(void 0!==o)for(var s=document.getElementsByTagName("script"),f=0;f 0 && deferred[i - 1][2] > priority; i--) deferred[i] = deferred[i - 1];\n\t\tdeferred[i] = [chunkIds, fn, priority];\n\t\treturn;\n\t}\n\tvar notFulfilled = Infinity;\n\tfor (var i = 0; i < deferred.length; i++) {\n\t\tvar chunkIds = deferred[i][0];\n\t\tvar fn = deferred[i][1];\n\t\tvar priority = deferred[i][2];\n\t\tvar fulfilled = true;\n\t\tfor (var j = 0; j < chunkIds.length; j++) {\n\t\t\tif ((priority & 1 === 0 || notFulfilled >= priority) && Object.keys(__webpack_require__.O).every(function(key) { return __webpack_require__.O[key](chunkIds[j]); })) {\n\t\t\t\tchunkIds.splice(j--, 1);\n\t\t\t} else {\n\t\t\t\tfulfilled = false;\n\t\t\t\tif(priority < notFulfilled) notFulfilled = priority;\n\t\t\t}\n\t\t}\n\t\tif(fulfilled) {\n\t\t\tdeferred.splice(i--, 1)\n\t\t\tvar r = fn();\n\t\t\tif (r !== undefined) result = r;\n\t\t}\n\t}\n\treturn result;\n};","var inProgress = {};\nvar dataWebpackPrefix = \"lingua-franca:\";\n// loadScript function to load a script via script tag\n__webpack_require__.l = function(url, done, key, chunkId) {\n\tif(inProgress[url]) { inProgress[url].push(done); return; }\n\tvar script, needAttach;\n\tif(key !== undefined) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tfor(var i = 0; i < scripts.length; i++) {\n\t\t\tvar s = scripts[i];\n\t\t\tif(s.getAttribute(\"src\") == url || s.getAttribute(\"data-webpack\") == dataWebpackPrefix + key) { script = s; break; }\n\t\t}\n\t}\n\tif(!script) {\n\t\tneedAttach = true;\n\t\tscript = document.createElement('script');\n\n\t\tscript.charset = 'utf-8';\n\t\tscript.timeout = 120;\n\t\tif (__webpack_require__.nc) {\n\t\t\tscript.setAttribute(\"nonce\", __webpack_require__.nc);\n\t\t}\n\t\tscript.setAttribute(\"data-webpack\", dataWebpackPrefix + key);\n\t\tscript.src = url;\n\t}\n\tinProgress[url] = [done];\n\tvar onScriptComplete = function(prev, event) {\n\t\t// avoid mem leaks in IE.\n\t\tscript.onerror = script.onload = null;\n\t\tclearTimeout(timeout);\n\t\tvar doneFns = inProgress[url];\n\t\tdelete inProgress[url];\n\t\tscript.parentNode && script.parentNode.removeChild(script);\n\t\tdoneFns && doneFns.forEach(function(fn) { return fn(event); });\n\t\tif(prev) return prev(event);\n\t}\n\t;\n\tvar timeout = setTimeout(onScriptComplete.bind(null, undefined, { type: 'timeout', target: script }), 120000);\n\tscript.onerror = onScriptComplete.bind(null, script.onerror);\n\tscript.onload = onScriptComplete.bind(null, script.onload);\n\tneedAttach && document.head.appendChild(script);\n};","var createStylesheet = function(chunkId, fullhref, resolve, reject) {\n\tvar linkTag = document.createElement(\"link\");\n\n\tlinkTag.rel = \"stylesheet\";\n\tlinkTag.type = \"text/css\";\n\tvar onLinkComplete = function(event) {\n\t\t// avoid mem leaks.\n\t\tlinkTag.onerror = linkTag.onload = null;\n\t\tif (event.type === 'load') {\n\t\t\tresolve();\n\t\t} else {\n\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n\t\t\tvar realHref = event && event.target && event.target.href || fullhref;\n\t\t\tvar err = new Error(\"Loading CSS chunk \" + chunkId + \" failed.\\n(\" + realHref + \")\");\n\t\t\terr.code = \"CSS_CHUNK_LOAD_FAILED\";\n\t\t\terr.type = errorType;\n\t\t\terr.request = realHref;\n\t\t\tlinkTag.parentNode.removeChild(linkTag)\n\t\t\treject(err);\n\t\t}\n\t}\n\tlinkTag.onerror = linkTag.onload = onLinkComplete;\n\tlinkTag.href = fullhref;\n\n\tdocument.head.appendChild(linkTag);\n\treturn linkTag;\n};\nvar findStylesheet = function(href, fullhref) {\n\tvar existingLinkTags = document.getElementsByTagName(\"link\");\n\tfor(var i = 0; i < existingLinkTags.length; i++) {\n\t\tvar tag = existingLinkTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\") || tag.getAttribute(\"href\");\n\t\tif(tag.rel === \"stylesheet\" && (dataHref === href || dataHref === fullhref)) return tag;\n\t}\n\tvar existingStyleTags = document.getElementsByTagName(\"style\");\n\tfor(var i = 0; i < existingStyleTags.length; i++) {\n\t\tvar tag = existingStyleTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\");\n\t\tif(dataHref === href || dataHref === fullhref) return tag;\n\t}\n};\nvar loadStylesheet = function(chunkId) {\n\treturn new Promise(function(resolve, reject) {\n\t\tvar href = __webpack_require__.miniCssF(chunkId);\n\t\tvar fullhref = __webpack_require__.p + href;\n\t\tif(findStylesheet(href, fullhref)) return resolve();\n\t\tcreateStylesheet(chunkId, fullhref, resolve, reject);\n\t});\n}\n// object to store loaded CSS chunks\nvar installedCssChunks = {\n\t658: 0\n};\n\n__webpack_require__.f.miniCss = function(chunkId, promises) {\n\tvar cssChunks = {\"532\":1};\n\tif(installedCssChunks[chunkId]) promises.push(installedCssChunks[chunkId]);\n\telse if(installedCssChunks[chunkId] !== 0 && cssChunks[chunkId]) {\n\t\tpromises.push(installedCssChunks[chunkId] = loadStylesheet(chunkId).then(function() {\n\t\t\tinstalledCssChunks[chunkId] = 0;\n\t\t}, function(e) {\n\t\t\tdelete installedCssChunks[chunkId];\n\t\t\tthrow e;\n\t\t}));\n\t}\n};\n\n// no hmr","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = function(module) {\n\tvar getter = module && module.__esModule ?\n\t\tfunction() { return module['default']; } :\n\t\tfunction() { return module; };\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = function(exports, definition) {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.f = {};\n// This file contains only the entry chunk.\n// The chunk loading function for additional chunks\n__webpack_require__.e = function(chunkId) {\n\treturn Promise.all(Object.keys(__webpack_require__.f).reduce(function(promises, key) {\n\t\t__webpack_require__.f[key](chunkId, promises);\n\t\treturn promises;\n\t}, []));\n};","// This function allow to reference async chunks\n__webpack_require__.u = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"\" + {\"56\":\"component---src-templates-pages-index-tsx\",\"175\":\"component---src-templates-pages-docs-handbook-index-tsx\",\"208\":\"90c0253ea39b81a5e49d137436e825ead0ef33a7\",\"248\":\"component---src-templates-pages-download-tsx\",\"306\":\"component---src-templates-pages-empty-tsx\",\"351\":\"commons\",\"517\":\"component---src-templates-documentation-tsx\",\"532\":\"styles\",\"533\":\"component---src-templates-pages-community-tsx\",\"542\":\"component---src-templates-pages-publications-and-presentations-tsx\",\"616\":\"component---src-templates-pages-docs-index-tsx\"}[chunkId] + \"-\" + {\"56\":\"0a7639f51e33385eb8f9\",\"175\":\"4aa0b654197d1d5fae53\",\"208\":\"7299e993a3830cab1f9e\",\"248\":\"dbe46be65ee8b6b2b60f\",\"306\":\"167cd7a4c2900b63ac14\",\"351\":\"c37c3719903bf568c839\",\"517\":\"04d26b5abab5434898b4\",\"532\":\"9fa542d257a161f411e7\",\"533\":\"311a2a5f18f0af45971a\",\"542\":\"f09ec94dd1ed2f74a804\",\"616\":\"b914ffbd927f75fc8fee\"}[chunkId] + \".js\";\n};","// This function allow to reference all chunks\n__webpack_require__.miniCssF = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"\" + \"styles\" + \".\" + \"8b00b2cda1c9b4bc348f\" + \".css\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = function(obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }","// define __esModule on exports\n__webpack_require__.r = function(exports) {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","__webpack_require__.p = \"/\";","// no baseURI\n\n// object to store loaded and loading chunks\n// undefined = chunk not loaded, null = chunk preloaded/prefetched\n// [resolve, reject, Promise] = chunk loading, 0 = chunk loaded\nvar installedChunks = {\n\t658: 0\n};\n\n__webpack_require__.f.j = function(chunkId, promises) {\n\t\t// JSONP chunk loading for javascript\n\t\tvar installedChunkData = __webpack_require__.o(installedChunks, chunkId) ? installedChunks[chunkId] : undefined;\n\t\tif(installedChunkData !== 0) { // 0 means \"already installed\".\n\n\t\t\t// a Promise means \"currently loading\".\n\t\t\tif(installedChunkData) {\n\t\t\t\tpromises.push(installedChunkData[2]);\n\t\t\t} else {\n\t\t\t\tif(!/^(532|658)$/.test(chunkId)) {\n\t\t\t\t\t// setup Promise in chunk cache\n\t\t\t\t\tvar promise = new Promise(function(resolve, reject) { installedChunkData = installedChunks[chunkId] = [resolve, reject]; });\n\t\t\t\t\tpromises.push(installedChunkData[2] = promise);\n\n\t\t\t\t\t// start chunk loading\n\t\t\t\t\tvar url = __webpack_require__.p + __webpack_require__.u(chunkId);\n\t\t\t\t\t// create error before stack unwound to get useful stacktrace later\n\t\t\t\t\tvar error = new Error();\n\t\t\t\t\tvar loadingEnded = function(event) {\n\t\t\t\t\t\tif(__webpack_require__.o(installedChunks, chunkId)) {\n\t\t\t\t\t\t\tinstalledChunkData = installedChunks[chunkId];\n\t\t\t\t\t\t\tif(installedChunkData !== 0) installedChunks[chunkId] = undefined;\n\t\t\t\t\t\t\tif(installedChunkData) {\n\t\t\t\t\t\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n\t\t\t\t\t\t\t\tvar realSrc = event && event.target && event.target.src;\n\t\t\t\t\t\t\t\terror.message = 'Loading chunk ' + chunkId + ' failed.\\n(' + errorType + ': ' + realSrc + ')';\n\t\t\t\t\t\t\t\terror.name = 'ChunkLoadError';\n\t\t\t\t\t\t\t\terror.type = errorType;\n\t\t\t\t\t\t\t\terror.request = realSrc;\n\t\t\t\t\t\t\t\tinstalledChunkData[1](error);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t\t__webpack_require__.l(url, loadingEnded, \"chunk-\" + chunkId, chunkId);\n\t\t\t\t} else installedChunks[chunkId] = 0;\n\t\t\t}\n\t\t}\n};\n\n// no prefetching\n\n// no preloaded\n\n// no HMR\n\n// no HMR manifest\n\n__webpack_require__.O.j = function(chunkId) { return installedChunks[chunkId] === 0; };\n\n// install a JSONP callback for chunk loading\nvar webpackJsonpCallback = function(parentChunkLoadingFunction, data) {\n\tvar chunkIds = data[0];\n\tvar moreModules = data[1];\n\tvar runtime = data[2];\n\t// add \"moreModules\" to the modules object,\n\t// then flag all \"chunkIds\" as loaded and fire callback\n\tvar moduleId, chunkId, i = 0;\n\tif(chunkIds.some(function(id) { return installedChunks[id] !== 0; })) {\n\t\tfor(moduleId in moreModules) {\n\t\t\tif(__webpack_require__.o(moreModules, moduleId)) {\n\t\t\t\t__webpack_require__.m[moduleId] = moreModules[moduleId];\n\t\t\t}\n\t\t}\n\t\tif(runtime) var result = runtime(__webpack_require__);\n\t}\n\tif(parentChunkLoadingFunction) parentChunkLoadingFunction(data);\n\tfor(;i < chunkIds.length; i++) {\n\t\tchunkId = chunkIds[i];\n\t\tif(__webpack_require__.o(installedChunks, chunkId) && installedChunks[chunkId]) {\n\t\t\tinstalledChunks[chunkId][0]();\n\t\t}\n\t\tinstalledChunks[chunkId] = 0;\n\t}\n\treturn __webpack_require__.O(result);\n}\n\nvar chunkLoadingGlobal = self[\"webpackChunklingua_franca\"] = self[\"webpackChunklingua_franca\"] || [];\nchunkLoadingGlobal.forEach(webpackJsonpCallback.bind(null, 0));\nchunkLoadingGlobal.push = webpackJsonpCallback.bind(null, chunkLoadingGlobal.push.bind(chunkLoadingGlobal));"],"names":["deferred","inProgress","dataWebpackPrefix","loadStylesheet","installedCssChunks","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","exports","module","__webpack_modules__","m","O","result","chunkIds","fn","priority","notFulfilled","Infinity","i","length","fulfilled","j","Object","keys","every","key","splice","r","n","getter","__esModule","d","a","definition","o","defineProperty","enumerable","get","f","e","chunkId","Promise","all","reduce","promises","u","miniCssF","g","globalThis","this","Function","window","obj","prop","prototype","hasOwnProperty","call","l","url","done","push","script","needAttach","scripts","document","getElementsByTagName","s","getAttribute","createElement","charset","timeout","nc","setAttribute","src","onScriptComplete","prev","event","onerror","onload","clearTimeout","doneFns","parentNode","removeChild","forEach","setTimeout","bind","type","target","head","appendChild","Symbol","toStringTag","value","p","resolve","reject","href","fullhref","existingLinkTags","dataHref","tag","rel","existingStyleTags","findStylesheet","linkTag","errorType","realHref","err","Error","code","request","createStylesheet","miniCss","then","installedChunks","installedChunkData","test","promise","error","realSrc","message","name","webpackJsonpCallback","parentChunkLoadingFunction","data","moreModules","runtime","some","id","chunkLoadingGlobal","self"],"sourceRoot":""} \ No newline at end of file +{"version":3,"file":"webpack-runtime-959e45056baa6f96280c.js","mappings":"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","sources":["webpack://lingua-franca/webpack/runtime/chunk loaded","webpack://lingua-franca/webpack/runtime/load script","webpack://lingua-franca/webpack/runtime/css loading","webpack://lingua-franca/webpack/bootstrap","webpack://lingua-franca/webpack/runtime/compat get default export","webpack://lingua-franca/webpack/runtime/define property getters","webpack://lingua-franca/webpack/runtime/ensure chunk","webpack://lingua-franca/webpack/runtime/get javascript chunk filename","webpack://lingua-franca/webpack/runtime/get mini-css chunk filename","webpack://lingua-franca/webpack/runtime/global","webpack://lingua-franca/webpack/runtime/hasOwnProperty shorthand","webpack://lingua-franca/webpack/runtime/make namespace object","webpack://lingua-franca/webpack/runtime/publicPath","webpack://lingua-franca/webpack/runtime/jsonp chunk loading"],"sourcesContent":["var deferred = [];\n__webpack_require__.O = function(result, chunkIds, fn, priority) {\n\tif(chunkIds) {\n\t\tpriority = priority || 0;\n\t\tfor(var i = deferred.length; i > 0 && deferred[i - 1][2] > priority; i--) deferred[i] = deferred[i - 1];\n\t\tdeferred[i] = [chunkIds, fn, priority];\n\t\treturn;\n\t}\n\tvar notFulfilled = Infinity;\n\tfor (var i = 0; i < deferred.length; i++) {\n\t\tvar chunkIds = deferred[i][0];\n\t\tvar fn = deferred[i][1];\n\t\tvar priority = deferred[i][2];\n\t\tvar fulfilled = true;\n\t\tfor (var j = 0; j < chunkIds.length; j++) {\n\t\t\tif ((priority & 1 === 0 || notFulfilled >= priority) && Object.keys(__webpack_require__.O).every(function(key) { return __webpack_require__.O[key](chunkIds[j]); })) {\n\t\t\t\tchunkIds.splice(j--, 1);\n\t\t\t} else {\n\t\t\t\tfulfilled = false;\n\t\t\t\tif(priority < notFulfilled) notFulfilled = priority;\n\t\t\t}\n\t\t}\n\t\tif(fulfilled) {\n\t\t\tdeferred.splice(i--, 1)\n\t\t\tvar r = fn();\n\t\t\tif (r !== undefined) result = r;\n\t\t}\n\t}\n\treturn result;\n};","var inProgress = {};\nvar dataWebpackPrefix = \"lingua-franca:\";\n// loadScript function to load a script via script tag\n__webpack_require__.l = function(url, done, key, chunkId) {\n\tif(inProgress[url]) { inProgress[url].push(done); return; }\n\tvar script, needAttach;\n\tif(key !== undefined) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tfor(var i = 0; i < scripts.length; i++) {\n\t\t\tvar s = scripts[i];\n\t\t\tif(s.getAttribute(\"src\") == url || s.getAttribute(\"data-webpack\") == dataWebpackPrefix + key) { script = s; break; }\n\t\t}\n\t}\n\tif(!script) {\n\t\tneedAttach = true;\n\t\tscript = document.createElement('script');\n\n\t\tscript.charset = 'utf-8';\n\t\tscript.timeout = 120;\n\t\tif (__webpack_require__.nc) {\n\t\t\tscript.setAttribute(\"nonce\", __webpack_require__.nc);\n\t\t}\n\t\tscript.setAttribute(\"data-webpack\", dataWebpackPrefix + key);\n\t\tscript.src = url;\n\t}\n\tinProgress[url] = [done];\n\tvar onScriptComplete = function(prev, event) {\n\t\t// avoid mem leaks in IE.\n\t\tscript.onerror = script.onload = null;\n\t\tclearTimeout(timeout);\n\t\tvar doneFns = inProgress[url];\n\t\tdelete inProgress[url];\n\t\tscript.parentNode && script.parentNode.removeChild(script);\n\t\tdoneFns && doneFns.forEach(function(fn) { return fn(event); });\n\t\tif(prev) return prev(event);\n\t}\n\t;\n\tvar timeout = setTimeout(onScriptComplete.bind(null, undefined, { type: 'timeout', target: script }), 120000);\n\tscript.onerror = onScriptComplete.bind(null, script.onerror);\n\tscript.onload = onScriptComplete.bind(null, script.onload);\n\tneedAttach && document.head.appendChild(script);\n};","var createStylesheet = function(chunkId, fullhref, resolve, reject) {\n\tvar linkTag = document.createElement(\"link\");\n\n\tlinkTag.rel = \"stylesheet\";\n\tlinkTag.type = \"text/css\";\n\tvar onLinkComplete = function(event) {\n\t\t// avoid mem leaks.\n\t\tlinkTag.onerror = linkTag.onload = null;\n\t\tif (event.type === 'load') {\n\t\t\tresolve();\n\t\t} else {\n\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n\t\t\tvar realHref = event && event.target && event.target.href || fullhref;\n\t\t\tvar err = new Error(\"Loading CSS chunk \" + chunkId + \" failed.\\n(\" + realHref + \")\");\n\t\t\terr.code = \"CSS_CHUNK_LOAD_FAILED\";\n\t\t\terr.type = errorType;\n\t\t\terr.request = realHref;\n\t\t\tlinkTag.parentNode.removeChild(linkTag)\n\t\t\treject(err);\n\t\t}\n\t}\n\tlinkTag.onerror = linkTag.onload = onLinkComplete;\n\tlinkTag.href = fullhref;\n\n\tdocument.head.appendChild(linkTag);\n\treturn linkTag;\n};\nvar findStylesheet = function(href, fullhref) {\n\tvar existingLinkTags = document.getElementsByTagName(\"link\");\n\tfor(var i = 0; i < existingLinkTags.length; i++) {\n\t\tvar tag = existingLinkTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\") || tag.getAttribute(\"href\");\n\t\tif(tag.rel === \"stylesheet\" && (dataHref === href || dataHref === fullhref)) return tag;\n\t}\n\tvar existingStyleTags = document.getElementsByTagName(\"style\");\n\tfor(var i = 0; i < existingStyleTags.length; i++) {\n\t\tvar tag = existingStyleTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\");\n\t\tif(dataHref === href || dataHref === fullhref) return tag;\n\t}\n};\nvar loadStylesheet = function(chunkId) {\n\treturn new Promise(function(resolve, reject) {\n\t\tvar href = __webpack_require__.miniCssF(chunkId);\n\t\tvar fullhref = __webpack_require__.p + href;\n\t\tif(findStylesheet(href, fullhref)) return resolve();\n\t\tcreateStylesheet(chunkId, fullhref, resolve, reject);\n\t});\n}\n// object to store loaded CSS chunks\nvar installedCssChunks = {\n\t658: 0\n};\n\n__webpack_require__.f.miniCss = function(chunkId, promises) {\n\tvar cssChunks = {\"532\":1};\n\tif(installedCssChunks[chunkId]) promises.push(installedCssChunks[chunkId]);\n\telse if(installedCssChunks[chunkId] !== 0 && cssChunks[chunkId]) {\n\t\tpromises.push(installedCssChunks[chunkId] = loadStylesheet(chunkId).then(function() {\n\t\t\tinstalledCssChunks[chunkId] = 0;\n\t\t}, function(e) {\n\t\t\tdelete installedCssChunks[chunkId];\n\t\t\tthrow e;\n\t\t}));\n\t}\n};\n\n// no hmr","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = function(module) {\n\tvar getter = module && module.__esModule ?\n\t\tfunction() { return module['default']; } :\n\t\tfunction() { return module; };\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = function(exports, definition) {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.f = {};\n// This file contains only the entry chunk.\n// The chunk loading function for additional chunks\n__webpack_require__.e = function(chunkId) {\n\treturn Promise.all(Object.keys(__webpack_require__.f).reduce(function(promises, key) {\n\t\t__webpack_require__.f[key](chunkId, promises);\n\t\treturn promises;\n\t}, []));\n};","// This function allow to reference async chunks\n__webpack_require__.u = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"\" + {\"56\":\"component---src-templates-pages-index-tsx\",\"175\":\"component---src-templates-pages-docs-handbook-index-tsx\",\"208\":\"90c0253ea39b81a5e49d137436e825ead0ef33a7\",\"248\":\"component---src-templates-pages-download-tsx\",\"306\":\"component---src-templates-pages-empty-tsx\",\"351\":\"commons\",\"517\":\"component---src-templates-documentation-tsx\",\"532\":\"styles\",\"533\":\"component---src-templates-pages-community-tsx\",\"542\":\"component---src-templates-pages-publications-and-presentations-tsx\",\"616\":\"component---src-templates-pages-docs-index-tsx\"}[chunkId] + \"-\" + {\"56\":\"0a7639f51e33385eb8f9\",\"175\":\"4aa0b654197d1d5fae53\",\"208\":\"7299e993a3830cab1f9e\",\"248\":\"dbe46be65ee8b6b2b60f\",\"306\":\"167cd7a4c2900b63ac14\",\"351\":\"192078f8b1bf52769f18\",\"517\":\"b5d67be0887eb73cf7b6\",\"532\":\"9fa542d257a161f411e7\",\"533\":\"311a2a5f18f0af45971a\",\"542\":\"f09ec94dd1ed2f74a804\",\"616\":\"b914ffbd927f75fc8fee\"}[chunkId] + \".js\";\n};","// This function allow to reference all chunks\n__webpack_require__.miniCssF = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"\" + \"styles\" + \".\" + \"8b00b2cda1c9b4bc348f\" + \".css\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = function(obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }","// define __esModule on exports\n__webpack_require__.r = function(exports) {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","__webpack_require__.p = \"/\";","// no baseURI\n\n// object to store loaded and loading chunks\n// undefined = chunk not loaded, null = chunk preloaded/prefetched\n// [resolve, reject, Promise] = chunk loading, 0 = chunk loaded\nvar installedChunks = {\n\t658: 0\n};\n\n__webpack_require__.f.j = function(chunkId, promises) {\n\t\t// JSONP chunk loading for javascript\n\t\tvar installedChunkData = __webpack_require__.o(installedChunks, chunkId) ? installedChunks[chunkId] : undefined;\n\t\tif(installedChunkData !== 0) { // 0 means \"already installed\".\n\n\t\t\t// a Promise means \"currently loading\".\n\t\t\tif(installedChunkData) {\n\t\t\t\tpromises.push(installedChunkData[2]);\n\t\t\t} else {\n\t\t\t\tif(!/^(532|658)$/.test(chunkId)) {\n\t\t\t\t\t// setup Promise in chunk cache\n\t\t\t\t\tvar promise = new Promise(function(resolve, reject) { installedChunkData = installedChunks[chunkId] = [resolve, reject]; });\n\t\t\t\t\tpromises.push(installedChunkData[2] = promise);\n\n\t\t\t\t\t// start chunk loading\n\t\t\t\t\tvar url = __webpack_require__.p + __webpack_require__.u(chunkId);\n\t\t\t\t\t// create error before stack unwound to get useful stacktrace later\n\t\t\t\t\tvar error = new Error();\n\t\t\t\t\tvar loadingEnded = function(event) {\n\t\t\t\t\t\tif(__webpack_require__.o(installedChunks, chunkId)) {\n\t\t\t\t\t\t\tinstalledChunkData = installedChunks[chunkId];\n\t\t\t\t\t\t\tif(installedChunkData !== 0) installedChunks[chunkId] = undefined;\n\t\t\t\t\t\t\tif(installedChunkData) {\n\t\t\t\t\t\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n\t\t\t\t\t\t\t\tvar realSrc = event && event.target && event.target.src;\n\t\t\t\t\t\t\t\terror.message = 'Loading chunk ' + chunkId + ' failed.\\n(' + errorType + ': ' + realSrc + ')';\n\t\t\t\t\t\t\t\terror.name = 'ChunkLoadError';\n\t\t\t\t\t\t\t\terror.type = errorType;\n\t\t\t\t\t\t\t\terror.request = realSrc;\n\t\t\t\t\t\t\t\tinstalledChunkData[1](error);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t\t__webpack_require__.l(url, loadingEnded, \"chunk-\" + chunkId, chunkId);\n\t\t\t\t} else installedChunks[chunkId] = 0;\n\t\t\t}\n\t\t}\n};\n\n// no prefetching\n\n// no preloaded\n\n// no HMR\n\n// no HMR manifest\n\n__webpack_require__.O.j = function(chunkId) { return installedChunks[chunkId] === 0; };\n\n// install a JSONP callback for chunk loading\nvar webpackJsonpCallback = function(parentChunkLoadingFunction, data) {\n\tvar chunkIds = data[0];\n\tvar moreModules = data[1];\n\tvar runtime = data[2];\n\t// add \"moreModules\" to the modules object,\n\t// then flag all \"chunkIds\" as loaded and fire callback\n\tvar moduleId, chunkId, i = 0;\n\tif(chunkIds.some(function(id) { return installedChunks[id] !== 0; })) {\n\t\tfor(moduleId in moreModules) {\n\t\t\tif(__webpack_require__.o(moreModules, moduleId)) {\n\t\t\t\t__webpack_require__.m[moduleId] = moreModules[moduleId];\n\t\t\t}\n\t\t}\n\t\tif(runtime) var result = runtime(__webpack_require__);\n\t}\n\tif(parentChunkLoadingFunction) parentChunkLoadingFunction(data);\n\tfor(;i < chunkIds.length; i++) {\n\t\tchunkId = chunkIds[i];\n\t\tif(__webpack_require__.o(installedChunks, chunkId) && installedChunks[chunkId]) {\n\t\t\tinstalledChunks[chunkId][0]();\n\t\t}\n\t\tinstalledChunks[chunkId] = 0;\n\t}\n\treturn __webpack_require__.O(result);\n}\n\nvar chunkLoadingGlobal = self[\"webpackChunklingua_franca\"] = self[\"webpackChunklingua_franca\"] || [];\nchunkLoadingGlobal.forEach(webpackJsonpCallback.bind(null, 0));\nchunkLoadingGlobal.push = webpackJsonpCallback.bind(null, chunkLoadingGlobal.push.bind(chunkLoadingGlobal));"],"names":["deferred","inProgress","dataWebpackPrefix","loadStylesheet","installedCssChunks","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","exports","module","__webpack_modules__","m","O","result","chunkIds","fn","priority","notFulfilled","Infinity","i","length","fulfilled","j","Object","keys","every","key","splice","r","n","getter","__esModule","d","a","definition","o","defineProperty","enumerable","get","f","e","chunkId","Promise","all","reduce","promises","u","miniCssF","g","globalThis","this","Function","window","obj","prop","prototype","hasOwnProperty","call","l","url","done","push","script","needAttach","scripts","document","getElementsByTagName","s","getAttribute","createElement","charset","timeout","nc","setAttribute","src","onScriptComplete","prev","event","onerror","onload","clearTimeout","doneFns","parentNode","removeChild","forEach","setTimeout","bind","type","target","head","appendChild","Symbol","toStringTag","value","p","resolve","reject","href","fullhref","existingLinkTags","dataHref","tag","rel","existingStyleTags","findStylesheet","linkTag","errorType","realHref","err","Error","code","request","createStylesheet","miniCss","then","installedChunks","installedChunkData","test","promise","error","realSrc","message","name","webpackJsonpCallback","parentChunkLoadingFunction","data","moreModules","runtime","some","id","chunkLoadingGlobal","self"],"sourceRoot":""} \ No newline at end of file diff --git a/webpack.stats.json b/webpack.stats.json index 35c40af39..6b6afeaa5 100644 --- a/webpack.stats.json +++ b/webpack.stats.json @@ -1 +1 @@ -{"name":"build-javascript","namedChunkGroups":{"polyfill":{"name":"polyfill","assets":[{"name":"webpack-runtime-e4cdf7e8b18db5d0f606.js","size":4959},{"name":"polyfill-8093e63a736b03ae8dca.js","size":85019}],"filteredAssets":0,"assetsSize":89978,"filteredAuxiliaryAssets":2,"auxiliaryAssetsSize":224667},"app":{"name":"app","assets":[{"name":"webpack-runtime-e4cdf7e8b18db5d0f606.js","size":4959},{"name":"framework-fffeb206b88cc7e067cd.js","size":140366},{"name":"app-91a8e178c6b8a94bf17c.js","size":153277}],"filteredAssets":0,"assetsSize":298602,"filteredAuxiliaryAssets":3,"auxiliaryAssetsSize":969241},"component---src-templates-documentation-tsx":{"name":"component---src-templates-documentation-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","size":7108},{"name":"component---src-templates-documentation-tsx-04d26b5abab5434898b4.js","size":62427}],"filteredAssets":0,"assetsSize":243018,"filteredAuxiliaryAssets":5,"auxiliaryAssetsSize":523587},"component---src-templates-pages-community-tsx":{"name":"component---src-templates-pages-community-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-community-tsx-311a2a5f18f0af45971a.js","size":9305}],"filteredAssets":0,"assetsSize":182788,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":407008},"component---src-templates-pages-docs-handbook-index-tsx":{"name":"component---src-templates-pages-docs-handbook-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-docs-handbook-index-tsx-4aa0b654197d1d5fae53.js","size":7482}],"filteredAssets":0,"assetsSize":180965,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":408330},"component---src-templates-pages-docs-index-tsx":{"name":"component---src-templates-pages-docs-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-docs-index-tsx-b914ffbd927f75fc8fee.js","size":9369}],"filteredAssets":0,"assetsSize":182852,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":412739},"component---src-templates-pages-download-tsx":{"name":"component---src-templates-pages-download-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-download-tsx-dbe46be65ee8b6b2b60f.js","size":9536}],"filteredAssets":0,"assetsSize":183019,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":403045},"component---src-templates-pages-empty-tsx":{"name":"component---src-templates-pages-empty-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-empty-tsx-167cd7a4c2900b63ac14.js","size":615}],"filteredAssets":0,"assetsSize":174098,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":390858},"component---src-templates-pages-index-tsx":{"name":"component---src-templates-pages-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","size":7108},{"name":"component---src-templates-pages-index-tsx-0a7639f51e33385eb8f9.js","size":7073}],"filteredAssets":0,"assetsSize":187664,"filteredAuxiliaryAssets":6,"auxiliaryAssetsSize":500688},"component---src-templates-pages-publications-and-presentations-tsx":{"name":"component---src-templates-pages-publications-and-presentations-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-c37c3719903bf568c839.js","size":108820},{"name":"component---src-templates-pages-publications-and-presentations-tsx-f09ec94dd1ed2f74a804.js","size":17116}],"filteredAssets":0,"assetsSize":190599,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":412437}},"assetsByChunkName":{"polyfill":["webpack-runtime-e4cdf7e8b18db5d0f606.js","polyfill-8093e63a736b03ae8dca.js"],"app":["webpack-runtime-e4cdf7e8b18db5d0f606.js","framework-fffeb206b88cc7e067cd.js","app-91a8e178c6b8a94bf17c.js"],"component---src-templates-documentation-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","component---src-templates-documentation-tsx-04d26b5abab5434898b4.js"],"component---src-templates-pages-community-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-community-tsx-311a2a5f18f0af45971a.js"],"component---src-templates-pages-docs-handbook-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-docs-handbook-index-tsx-4aa0b654197d1d5fae53.js"],"component---src-templates-pages-docs-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-docs-index-tsx-b914ffbd927f75fc8fee.js"],"component---src-templates-pages-download-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-download-tsx-dbe46be65ee8b6b2b60f.js"],"component---src-templates-pages-empty-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-empty-tsx-167cd7a4c2900b63ac14.js"],"component---src-templates-pages-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","component---src-templates-pages-index-tsx-0a7639f51e33385eb8f9.js"],"component---src-templates-pages-publications-and-presentations-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-c37c3719903bf568c839.js","component---src-templates-pages-publications-and-presentations-tsx-f09ec94dd1ed2f74a804.js"]},"childAssetsByChunkName":{}} \ No newline at end of file +{"name":"build-javascript","namedChunkGroups":{"polyfill":{"name":"polyfill","assets":[{"name":"webpack-runtime-959e45056baa6f96280c.js","size":4959},{"name":"polyfill-8093e63a736b03ae8dca.js","size":85019}],"filteredAssets":0,"assetsSize":89978,"filteredAuxiliaryAssets":2,"auxiliaryAssetsSize":224667},"app":{"name":"app","assets":[{"name":"webpack-runtime-959e45056baa6f96280c.js","size":4959},{"name":"framework-fffeb206b88cc7e067cd.js","size":140366},{"name":"app-91a8e178c6b8a94bf17c.js","size":153277}],"filteredAssets":0,"assetsSize":298602,"filteredAuxiliaryAssets":3,"auxiliaryAssetsSize":969241},"component---src-templates-documentation-tsx":{"name":"component---src-templates-documentation-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","size":7108},{"name":"component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js","size":62673}],"filteredAssets":0,"assetsSize":243264,"filteredAuxiliaryAssets":5,"auxiliaryAssetsSize":523587},"component---src-templates-pages-community-tsx":{"name":"component---src-templates-pages-community-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-community-tsx-311a2a5f18f0af45971a.js","size":9305}],"filteredAssets":0,"assetsSize":182788,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":407008},"component---src-templates-pages-docs-handbook-index-tsx":{"name":"component---src-templates-pages-docs-handbook-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-docs-handbook-index-tsx-4aa0b654197d1d5fae53.js","size":7482}],"filteredAssets":0,"assetsSize":180965,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":408330},"component---src-templates-pages-docs-index-tsx":{"name":"component---src-templates-pages-docs-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-docs-index-tsx-b914ffbd927f75fc8fee.js","size":9369}],"filteredAssets":0,"assetsSize":182852,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":412739},"component---src-templates-pages-download-tsx":{"name":"component---src-templates-pages-download-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-download-tsx-dbe46be65ee8b6b2b60f.js","size":9536}],"filteredAssets":0,"assetsSize":183019,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":403045},"component---src-templates-pages-empty-tsx":{"name":"component---src-templates-pages-empty-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-empty-tsx-167cd7a4c2900b63ac14.js","size":615}],"filteredAssets":0,"assetsSize":174098,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":390858},"component---src-templates-pages-index-tsx":{"name":"component---src-templates-pages-index-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","size":7108},{"name":"component---src-templates-pages-index-tsx-0a7639f51e33385eb8f9.js","size":7073}],"filteredAssets":0,"assetsSize":187664,"filteredAuxiliaryAssets":6,"auxiliaryAssetsSize":500688},"component---src-templates-pages-publications-and-presentations-tsx":{"name":"component---src-templates-pages-publications-and-presentations-tsx","assets":[{"name":"styles.8b00b2cda1c9b4bc348f.css","size":64663},{"name":"commons-192078f8b1bf52769f18.js","size":108820},{"name":"component---src-templates-pages-publications-and-presentations-tsx-f09ec94dd1ed2f74a804.js","size":17116}],"filteredAssets":0,"assetsSize":190599,"filteredAuxiliaryAssets":4,"auxiliaryAssetsSize":412437}},"assetsByChunkName":{"polyfill":["webpack-runtime-959e45056baa6f96280c.js","polyfill-8093e63a736b03ae8dca.js"],"app":["webpack-runtime-959e45056baa6f96280c.js","framework-fffeb206b88cc7e067cd.js","app-91a8e178c6b8a94bf17c.js"],"component---src-templates-documentation-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","component---src-templates-documentation-tsx-b5d67be0887eb73cf7b6.js"],"component---src-templates-pages-community-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-community-tsx-311a2a5f18f0af45971a.js"],"component---src-templates-pages-docs-handbook-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-docs-handbook-index-tsx-4aa0b654197d1d5fae53.js"],"component---src-templates-pages-docs-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-docs-index-tsx-b914ffbd927f75fc8fee.js"],"component---src-templates-pages-download-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-download-tsx-dbe46be65ee8b6b2b60f.js"],"component---src-templates-pages-empty-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-empty-tsx-167cd7a4c2900b63ac14.js"],"component---src-templates-pages-index-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","90c0253ea39b81a5e49d137436e825ead0ef33a7-7299e993a3830cab1f9e.js","component---src-templates-pages-index-tsx-0a7639f51e33385eb8f9.js"],"component---src-templates-pages-publications-and-presentations-tsx":["styles.8b00b2cda1c9b4bc348f.css","commons-192078f8b1bf52769f18.js","component---src-templates-pages-publications-and-presentations-tsx-f09ec94dd1ed2f74a804.js"]},"childAssetsByChunkName":{}} \ No newline at end of file